2009-05-10 10:54:31 +00:00
|
|
|
# Redmine - project management software
|
2016-03-13 10:30:10 +00:00
|
|
|
# Copyright (C) 2006-2016 Jean-Philippe Lang
|
2007-03-12 17:59:02 +00:00
|
|
|
#
|
|
|
|
|
# This program is free software; you can redistribute it and/or
|
|
|
|
|
# modify it under the terms of the GNU General Public License
|
|
|
|
|
# as published by the Free Software Foundation; either version 2
|
|
|
|
|
# of the License, or (at your option) any later version.
|
2011-08-21 01:54:24 +00:00
|
|
|
#
|
2007-03-12 17:59:02 +00:00
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
# GNU General Public License for more details.
|
2011-08-21 01:54:24 +00:00
|
|
|
#
|
2007-03-12 17:59:02 +00:00
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
|
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
|
|
class Member < ActiveRecord::Base
|
|
|
|
|
belongs_to :user
|
2009-09-12 08:36:46 +00:00
|
|
|
belongs_to :principal, :foreign_key => 'user_id'
|
|
|
|
|
has_many :member_roles, :dependent => :destroy
|
2016-07-14 07:15:13 +00:00
|
|
|
has_many :roles, lambda { distinct }, :through => :member_roles
|
2007-03-12 17:59:02 +00:00
|
|
|
belongs_to :project
|
|
|
|
|
|
2009-09-12 08:36:46 +00:00
|
|
|
validates_presence_of :principal, :project
|
2007-03-12 17:59:02 +00:00
|
|
|
validates_uniqueness_of :user_id, :scope => :project_id
|
2011-12-03 14:33:49 +00:00
|
|
|
validate :validate_role
|
2014-10-22 17:37:16 +00:00
|
|
|
attr_protected :id
|
2009-12-13 12:39:22 +00:00
|
|
|
|
2017-04-08 08:02:06 +00:00
|
|
|
before_destroy :set_issue_category_nil, :remove_from_project_default_assigned_to
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2015-06-18 19:12:17 +00:00
|
|
|
scope :active, lambda { joins(:principal).where(:users => {:status => Principal::STATUS_ACTIVE})}
|
|
|
|
|
|
2016-12-03 08:26:08 +00:00
|
|
|
# Sort by first role and principal
|
|
|
|
|
scope :sorted, lambda {
|
|
|
|
|
includes(:member_roles, :roles, :principal).
|
|
|
|
|
reorder("#{Role.table_name}.position").
|
|
|
|
|
order(Principal.fields_for_order_statement)
|
|
|
|
|
}
|
2017-01-07 10:28:59 +00:00
|
|
|
scope :sorted_by_project, lambda {
|
|
|
|
|
includes(:project).
|
|
|
|
|
reorder("#{Project.table_name}.lft")
|
|
|
|
|
}
|
2016-12-03 08:26:08 +00:00
|
|
|
|
2015-05-31 07:16:23 +00:00
|
|
|
alias :base_reload :reload
|
|
|
|
|
def reload(*args)
|
|
|
|
|
@managed_roles = nil
|
|
|
|
|
base_reload(*args)
|
|
|
|
|
end
|
|
|
|
|
|
2011-12-08 15:12:02 +00:00
|
|
|
def role
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def role=
|
|
|
|
|
end
|
|
|
|
|
|
2007-03-12 17:59:02 +00:00
|
|
|
def name
|
2007-06-29 17:27:27 +00:00
|
|
|
self.user.name
|
2006-07-29 19:54:22 +00:00
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2009-09-12 08:36:46 +00:00
|
|
|
alias :base_role_ids= :role_ids=
|
|
|
|
|
def role_ids=(arg)
|
|
|
|
|
ids = (arg || []).collect(&:to_i) - [0]
|
|
|
|
|
# Keep inherited roles
|
|
|
|
|
ids += member_roles.select {|mr| !mr.inherited_from.nil?}.collect(&:role_id)
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2009-09-12 08:36:46 +00:00
|
|
|
new_role_ids = ids - role_ids
|
|
|
|
|
# Add new roles
|
2014-10-23 21:46:40 +00:00
|
|
|
new_role_ids.each {|id| member_roles << MemberRole.new(:role_id => id, :member => self) }
|
2009-09-12 08:36:46 +00:00
|
|
|
# Remove roles (Rails' #role_ids= will not trigger MemberRole#on_destroy)
|
2009-12-13 12:39:22 +00:00
|
|
|
member_roles_to_destroy = member_roles.select {|mr| !ids.include?(mr.role_id)}
|
|
|
|
|
if member_roles_to_destroy.any?
|
|
|
|
|
member_roles_to_destroy.each(&:destroy)
|
|
|
|
|
end
|
2009-03-28 12:07:05 +00:00
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2008-01-25 10:31:06 +00:00
|
|
|
def <=>(member)
|
2015-06-21 18:43:12 +00:00
|
|
|
a, b = roles.sort, member.roles.sort
|
2012-02-12 14:54:30 +00:00
|
|
|
if a == b
|
|
|
|
|
if principal
|
|
|
|
|
principal <=> member.principal
|
|
|
|
|
else
|
|
|
|
|
1
|
|
|
|
|
end
|
2015-06-21 19:28:38 +00:00
|
|
|
elsif a.any?
|
|
|
|
|
b.any? ? a <=> b : -1
|
2012-02-12 14:54:30 +00:00
|
|
|
else
|
|
|
|
|
1
|
|
|
|
|
end
|
2009-09-12 08:36:46 +00:00
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Set member role ids ignoring any change to roles that
|
|
|
|
|
# user is not allowed to manage
|
2015-05-31 07:16:23 +00:00
|
|
|
def set_editable_role_ids(ids, user=User.current)
|
|
|
|
|
ids = (ids || []).collect(&:to_i) - [0]
|
|
|
|
|
editable_role_ids = user.managed_roles(project).map(&:id)
|
|
|
|
|
untouched_role_ids = self.role_ids - editable_role_ids
|
|
|
|
|
touched_role_ids = ids & editable_role_ids
|
|
|
|
|
self.role_ids = untouched_role_ids + touched_role_ids
|
2008-01-25 10:31:06 +00:00
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Returns true if one of the member roles is inherited
|
2015-05-31 07:16:23 +00:00
|
|
|
def any_inherited_role?
|
|
|
|
|
member_roles.any? {|mr| mr.inherited_from}
|
|
|
|
|
end
|
|
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Returns true if the member has the role and if it's inherited
|
2015-05-31 07:16:23 +00:00
|
|
|
def has_inherited_role?(role)
|
|
|
|
|
member_roles.any? {|mr| mr.role_id == role.id && mr.inherited_from.present?}
|
|
|
|
|
end
|
|
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Returns true if the member's role is editable by user
|
2015-05-31 07:16:23 +00:00
|
|
|
def role_editable?(role, user=User.current)
|
|
|
|
|
if has_inherited_role?(role)
|
|
|
|
|
false
|
2014-05-24 09:55:39 +00:00
|
|
|
else
|
2015-05-31 07:16:23 +00:00
|
|
|
user.managed_roles(project).include?(role)
|
2014-05-24 09:55:39 +00:00
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Returns true if the member is deletable by user
|
2015-05-31 07:16:23 +00:00
|
|
|
def deletable?(user=User.current)
|
|
|
|
|
if any_inherited_role?
|
|
|
|
|
false
|
|
|
|
|
else
|
|
|
|
|
roles & user.managed_roles(project) == roles
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Destroys the member
|
2015-05-31 07:16:23 +00:00
|
|
|
def destroy
|
|
|
|
|
member_roles.reload.each(&:destroy_without_member_removal)
|
|
|
|
|
super
|
|
|
|
|
end
|
|
|
|
|
|
2015-06-06 07:22:05 +00:00
|
|
|
# Returns true if the member is user or is a group
|
|
|
|
|
# that includes user
|
2009-12-26 15:46:12 +00:00
|
|
|
def include?(user)
|
|
|
|
|
if principal.is_a?(Group)
|
|
|
|
|
!user.nil? && user.groups.include?(principal)
|
|
|
|
|
else
|
2016-12-31 13:44:41 +00:00
|
|
|
self.principal == user
|
2009-12-26 15:46:12 +00:00
|
|
|
end
|
|
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2011-12-03 16:41:23 +00:00
|
|
|
def set_issue_category_nil
|
2014-07-29 17:49:53 +00:00
|
|
|
if user_id && project_id
|
2009-09-12 08:36:46 +00:00
|
|
|
# remove category based auto assignments for this member
|
2014-07-29 17:49:53 +00:00
|
|
|
IssueCategory.where(["project_id = ? AND assigned_to_id = ?", project_id, user_id]).
|
2014-01-08 04:58:13 +00:00
|
|
|
update_all("assigned_to_id = NULL")
|
2009-09-12 08:36:46 +00:00
|
|
|
end
|
2007-06-29 17:21:37 +00:00
|
|
|
end
|
2010-03-18 15:49:11 +00:00
|
|
|
|
2017-04-08 08:02:06 +00:00
|
|
|
def remove_from_project_default_assigned_to
|
|
|
|
|
if user_id && project && project.default_assigned_to_id == user_id
|
|
|
|
|
# remove project based auto assignments for this member
|
|
|
|
|
project.update_column(:default_assigned_to_id, nil)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2015-05-31 07:16:23 +00:00
|
|
|
# Returns the roles that the member is allowed to manage
|
|
|
|
|
# in the project the member belongs to
|
|
|
|
|
def managed_roles
|
|
|
|
|
@managed_roles ||= begin
|
|
|
|
|
if principal.try(:admin?)
|
|
|
|
|
Role.givable.to_a
|
|
|
|
|
else
|
|
|
|
|
members_management_roles = roles.select do |role|
|
|
|
|
|
role.has_permission?(:manage_members)
|
|
|
|
|
end
|
|
|
|
|
if members_management_roles.empty?
|
|
|
|
|
[]
|
|
|
|
|
elsif members_management_roles.any?(&:all_roles_managed?)
|
|
|
|
|
Role.givable.to_a
|
|
|
|
|
else
|
|
|
|
|
members_management_roles.map(&:managed_roles).reduce(&:|)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-03-08 20:36:22 +00:00
|
|
|
# Creates memberships for principal with the attributes, or add the roles
|
|
|
|
|
# if the membership already exists.
|
2014-10-23 21:46:40 +00:00
|
|
|
# * project_ids : one or more project ids
|
|
|
|
|
# * role_ids : ids of the roles to give to each membership
|
|
|
|
|
#
|
|
|
|
|
# Example:
|
|
|
|
|
# Member.create_principal_memberships(user, :project_ids => [2, 5], :role_ids => [1, 3]
|
|
|
|
|
def self.create_principal_memberships(principal, attributes)
|
|
|
|
|
members = []
|
|
|
|
|
if attributes
|
|
|
|
|
project_ids = Array.wrap(attributes[:project_ids] || attributes[:project_id])
|
2017-03-08 21:37:13 +00:00
|
|
|
role_ids = Array.wrap(attributes[:role_ids])
|
2014-10-23 21:46:40 +00:00
|
|
|
project_ids.each do |project_id|
|
2017-03-08 20:35:11 +00:00
|
|
|
member = Member.find_or_new(project_id, principal)
|
|
|
|
|
member.role_ids |= role_ids
|
|
|
|
|
member.save
|
|
|
|
|
members << member
|
2014-10-23 21:46:40 +00:00
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
members
|
2010-03-18 15:49:11 +00:00
|
|
|
end
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2017-04-02 03:34:44 +00:00
|
|
|
# Finds or initializes a Member for the given project and principal
|
2013-02-02 12:50:45 +00:00
|
|
|
def self.find_or_new(project, principal)
|
|
|
|
|
project_id = project.is_a?(Project) ? project.id : project
|
|
|
|
|
principal_id = principal.is_a?(Principal) ? principal.id : principal
|
|
|
|
|
|
|
|
|
|
member = Member.find_by_project_id_and_user_id(project_id, principal_id)
|
|
|
|
|
member ||= Member.new(:project_id => project_id, :user_id => principal_id)
|
|
|
|
|
member
|
|
|
|
|
end
|
|
|
|
|
|
2009-05-10 10:54:31 +00:00
|
|
|
protected
|
2011-08-21 01:54:24 +00:00
|
|
|
|
2011-12-03 14:33:49 +00:00
|
|
|
def validate_role
|
2016-07-16 11:52:20 +00:00
|
|
|
errors.add(:role, :empty) if member_roles.empty? && roles.empty?
|
2009-05-10 10:54:31 +00:00
|
|
|
end
|
2006-06-28 18:11:03 +00:00
|
|
|
end
|