2016-02-12 23:52:43 +08:00
|
|
|
class Project < ActiveRecord::Base
|
|
|
|
include ArchivableModel, SearchableModel
|
|
|
|
|
|
|
|
enum visibility: { hidden: 0, visible: 1 }
|
|
|
|
|
2016-09-21 21:35:23 +08:00
|
|
|
auto_strip_attributes :name, nullify: false
|
2016-02-12 23:52:43 +08:00
|
|
|
validates :name,
|
2016-10-05 23:45:20 +08:00
|
|
|
length: { minimum: Constants::NAME_MIN_LENGTH,
|
|
|
|
maximum: Constants::NAME_MAX_LENGTH },
|
2017-01-24 23:34:21 +08:00
|
|
|
uniqueness: { scope: :team, case_sensitive: false }
|
2016-02-12 23:52:43 +08:00
|
|
|
validates :visibility, presence: true
|
2017-01-24 23:34:21 +08:00
|
|
|
validates :team, presence: true
|
2016-02-12 23:52:43 +08:00
|
|
|
|
|
|
|
belongs_to :created_by, foreign_key: 'created_by_id', class_name: 'User'
|
|
|
|
belongs_to :last_modified_by, foreign_key: 'last_modified_by_id', class_name: 'User'
|
|
|
|
belongs_to :archived_by, foreign_key: 'archived_by_id', class_name: 'User'
|
|
|
|
belongs_to :restored_by, foreign_key: 'restored_by_id', class_name: 'User'
|
|
|
|
has_many :user_projects, inverse_of: :project
|
|
|
|
has_many :users, through: :user_projects
|
2016-07-22 16:43:55 +08:00
|
|
|
has_many :experiments, inverse_of: :project
|
2016-02-12 23:52:43 +08:00
|
|
|
has_many :project_comments, inverse_of: :project
|
|
|
|
has_many :comments, through: :project_comments
|
|
|
|
has_many :activities, inverse_of: :project
|
|
|
|
has_many :tags, inverse_of: :project
|
|
|
|
has_many :reports, inverse_of: :project, dependent: :destroy
|
|
|
|
has_many :report_elements, inverse_of: :project, dependent: :destroy
|
2017-01-24 23:34:21 +08:00
|
|
|
belongs_to :team, inverse_of: :projects
|
2016-02-12 23:52:43 +08:00
|
|
|
|
2017-01-11 21:56:47 +08:00
|
|
|
def self.search(
|
|
|
|
user,
|
|
|
|
include_archived,
|
|
|
|
query = nil,
|
|
|
|
page = 1,
|
2017-01-24 23:34:21 +08:00
|
|
|
current_team = nil
|
2017-01-11 21:56:47 +08:00
|
|
|
)
|
2016-07-21 19:11:15 +08:00
|
|
|
|
|
|
|
if query
|
2017-01-17 18:04:16 +08:00
|
|
|
a_query = '%' + query.strip.gsub('_', '\\_').gsub('%', '\\%') + '%'
|
2016-07-21 19:11:15 +08:00
|
|
|
else
|
|
|
|
a_query = query
|
|
|
|
end
|
|
|
|
|
2017-01-24 23:34:21 +08:00
|
|
|
if current_team
|
2016-02-12 23:52:43 +08:00
|
|
|
new_query = Project
|
2017-01-11 21:56:47 +08:00
|
|
|
.distinct
|
|
|
|
.joins(:user_projects)
|
2017-01-24 23:34:21 +08:00
|
|
|
.where('projects.team_id = ?',
|
|
|
|
current_team.id)
|
2017-01-11 21:56:47 +08:00
|
|
|
.where('projects.visibility = 1 OR user_projects.user_id = ?',
|
|
|
|
user.id)
|
|
|
|
.where_attributes_like(:name, a_query)
|
2017-01-20 00:09:25 +08:00
|
|
|
|
2017-01-16 23:50:18 +08:00
|
|
|
if include_archived
|
|
|
|
return new_query
|
|
|
|
else
|
|
|
|
return new_query.where('projects.archived = ?', false)
|
|
|
|
end
|
2016-02-12 23:52:43 +08:00
|
|
|
else
|
2017-01-24 23:34:21 +08:00
|
|
|
team_ids =
|
2017-01-25 00:08:48 +08:00
|
|
|
Team
|
2017-01-24 23:34:21 +08:00
|
|
|
.joins(:user_teams)
|
|
|
|
.where('user_teams.user_id = ?', user.id)
|
2017-01-11 21:56:47 +08:00
|
|
|
.select('id')
|
2016-02-12 23:52:43 +08:00
|
|
|
.distinct
|
2017-01-11 21:56:47 +08:00
|
|
|
|
|
|
|
if include_archived
|
|
|
|
new_query = Project
|
|
|
|
.distinct
|
|
|
|
.joins(:user_projects)
|
2017-01-24 23:34:21 +08:00
|
|
|
.where('projects.team_id IN (?)', team_ids)
|
2017-01-11 21:56:47 +08:00
|
|
|
.where(
|
|
|
|
'projects.visibility = 1 OR user_projects.user_id = ?',
|
|
|
|
user.id
|
|
|
|
)
|
|
|
|
.where_attributes_like(:name, a_query)
|
|
|
|
|
|
|
|
else
|
|
|
|
new_query = Project
|
|
|
|
.distinct
|
|
|
|
.joins(:user_projects)
|
2017-01-24 23:34:21 +08:00
|
|
|
.where('projects.team_id IN (?)', team_ids)
|
2017-01-11 21:56:47 +08:00
|
|
|
.where(
|
|
|
|
'projects.visibility = 1 OR user_projects.user_id = ?',
|
|
|
|
user.id
|
|
|
|
)
|
|
|
|
.where_attributes_like(:name, a_query)
|
|
|
|
.where('projects.archived = ?', false)
|
|
|
|
end
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# Show all results if needed
|
2016-10-05 23:45:20 +08:00
|
|
|
if page == Constants::SEARCH_NO_LIMIT
|
2016-02-12 23:52:43 +08:00
|
|
|
new_query
|
|
|
|
else
|
|
|
|
new_query
|
2016-10-05 23:45:20 +08:00
|
|
|
.limit(Constants::SEARCH_LIMIT)
|
|
|
|
.offset((page - 1) * Constants::SEARCH_LIMIT)
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-13 16:00:36 +08:00
|
|
|
def last_activities(count = Constants::ACTIVITY_AND_NOTIF_SEARCH_LIMIT)
|
2016-07-21 19:11:15 +08:00
|
|
|
activities.order(created_at: :desc).first(count)
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# Get project comments order by created_at time. Results are paginated
|
|
|
|
# using last comment id and per_page parameters.
|
2016-10-05 23:45:20 +08:00
|
|
|
def last_comments(last_id = 1, per_page = Constants::COMMENTS_SEARCH_LIMIT)
|
2016-10-13 17:05:11 +08:00
|
|
|
last_id = Constants::INFINITY if last_id <= 1
|
2016-09-27 15:00:02 +08:00
|
|
|
comments = Comment.joins(:project_comment)
|
|
|
|
.where(project_comments: { project_id: id })
|
|
|
|
.where('comments.id < ?', last_id)
|
|
|
|
.order(created_at: :desc)
|
|
|
|
.limit(per_page)
|
|
|
|
comments.reverse
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def unassigned_users
|
2016-07-21 19:11:15 +08:00
|
|
|
User
|
2017-01-24 23:34:21 +08:00
|
|
|
.joins('INNER JOIN user_teams ON users.id = user_teams.user_id')
|
|
|
|
.where('user_teams.team_id = ?', team)
|
2016-08-18 19:24:44 +08:00
|
|
|
.where.not(confirmed_at: nil)
|
|
|
|
.where('users.id NOT IN (?)',
|
|
|
|
UserProject.where(project: self).select(:user_id).distinct)
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def user_role(user)
|
|
|
|
unless self.users.include? user
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
return (self.user_projects.select { |up| up.user == user }).first.role
|
|
|
|
end
|
|
|
|
|
2016-12-23 22:40:01 +08:00
|
|
|
def active_experiments(sort_by = :new)
|
|
|
|
sort = case sort_by
|
|
|
|
when 'old' then { created_at: :asc }
|
|
|
|
when 'atoz' then { name: :asc }
|
|
|
|
when 'ztoa' then { name: :desc }
|
|
|
|
else { created_at: :desc }
|
|
|
|
end
|
|
|
|
experiments.is_archived(false).order(sort)
|
2016-07-29 21:47:41 +08:00
|
|
|
end
|
|
|
|
|
2016-08-08 20:19:35 +08:00
|
|
|
def archived_experiments
|
|
|
|
experiments.is_archived(true)
|
|
|
|
end
|
|
|
|
|
2016-07-25 17:25:10 +08:00
|
|
|
def project_my_modules
|
2016-07-26 14:44:09 +08:00
|
|
|
MyModule.where('"experiment_id" IN (?)', experiments.select(:id))
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def space_taken
|
|
|
|
st = 0
|
2016-07-22 20:31:09 +08:00
|
|
|
project_my_modules.find_each do |my_module|
|
2016-02-12 23:52:43 +08:00
|
|
|
st += my_module.space_taken
|
|
|
|
end
|
|
|
|
st
|
|
|
|
end
|
|
|
|
|
|
|
|
# Writes to user log.
|
|
|
|
def log(message)
|
|
|
|
final = "[%s] %s" % [name, message]
|
2017-01-24 23:34:21 +08:00
|
|
|
team.log(final)
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|
|
|
|
|
2016-08-03 18:57:51 +08:00
|
|
|
def assigned_samples
|
|
|
|
Sample.joins(:my_modules).where(my_modules: {
|
|
|
|
id: my_modules_ids.split(',')
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
def my_modules_ids
|
|
|
|
ids = active_experiments.map do |exp|
|
|
|
|
exp.my_modules.pluck(:id) if exp.my_modules
|
|
|
|
end
|
|
|
|
ids.delete_if { |i| i.flatten.empty? }
|
|
|
|
ids.join(', ')
|
|
|
|
end
|
2016-08-17 19:17:40 +08:00
|
|
|
|
|
|
|
def assigned_modules(user)
|
|
|
|
role = user_role(user)
|
|
|
|
if role.blank?
|
|
|
|
MyModule.none
|
|
|
|
elsif role == 'owner'
|
2016-08-18 16:55:29 +08:00
|
|
|
project_my_modules
|
|
|
|
.joins(:experiment)
|
|
|
|
.where('experiments.archived=false')
|
2016-08-18 17:53:57 +08:00
|
|
|
.where('my_modules.archived=false')
|
|
|
|
|
2016-08-17 19:17:40 +08:00
|
|
|
else
|
|
|
|
project_my_modules
|
|
|
|
.joins(:user_my_modules)
|
2016-08-18 16:55:29 +08:00
|
|
|
.joins(:experiment)
|
|
|
|
.where('experiments.archived=false AND user_my_modules.user_id IN (?)',
|
|
|
|
user.id)
|
2016-08-18 17:53:57 +08:00
|
|
|
.where('my_modules.archived=false')
|
2016-08-17 19:17:40 +08:00
|
|
|
.distinct
|
|
|
|
end
|
|
|
|
end
|
2016-02-12 23:52:43 +08:00
|
|
|
end
|