debian-mirror-gitlab/app/finders/members_finder.rb

75 lines
2.3 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2017-08-17 22:00:37 +05:30
class MembersFinder
attr_reader :project, :current_user, :group
def initialize(project, current_user)
@project = project
@current_user = current_user
@group = project.group
end
2019-09-04 21:01:54 +05:30
def execute(include_descendants: false, include_invited_groups_members: false)
2017-08-17 22:00:37 +05:30
project_members = project.project_members
project_members = project_members.non_invite unless can?(current_user, :admin_project, project)
2019-09-04 21:01:54 +05:30
union_members = group_union_members(include_descendants, include_invited_groups_members)
2017-08-17 22:00:37 +05:30
2019-09-04 21:01:54 +05:30
if union_members.any?
distinct_union_of_members(union_members << project_members)
2018-03-17 18:26:18 +05:30
else
project_members
end
2017-08-17 22:00:37 +05:30
end
def can?(*args)
Ability.allowed?(*args)
end
2018-03-17 18:26:18 +05:30
private
2019-09-04 21:01:54 +05:30
def group_union_members(include_descendants, include_invited_groups_members)
[].tap do |members|
members << direct_group_members(include_descendants) if group
members << project_invited_groups_members if include_invited_groups_members
end
end
def direct_group_members(include_descendants)
GroupMembersFinder.new(group).execute(include_descendants: include_descendants).non_invite # rubocop: disable CodeReuse/Finder
end
def project_invited_groups_members
invited_groups_ids_including_ancestors = Gitlab::ObjectHierarchy
.new(project.invited_groups)
.base_and_ancestors
.public_or_visible_to_user(current_user)
.select(:id)
GroupMember.with_source_id(invited_groups_ids_including_ancestors)
end
def distinct_union_of_members(union_members)
union = Gitlab::SQL::Union.new(union_members, remove_duplicates: false) # rubocop: disable Gitlab/Union
sql = distinct_on(union)
Member.includes(:user).from([Arel.sql("(#{sql}) AS #{Member.table_name}")]) # rubocop: disable CodeReuse/ActiveRecord
end
2018-03-17 18:26:18 +05:30
def distinct_on(union)
# We're interested in a list of members without duplicates by user_id.
# We prefer project members over group members, project members should go first.
2019-10-12 21:52:04 +05:30
<<~SQL
SELECT DISTINCT ON (user_id, invite_email) member_union.*
FROM (#{union.to_sql}) AS member_union
ORDER BY user_id,
invite_email,
CASE
WHEN type = 'ProjectMember' THEN 1
WHEN type = 'GroupMember' THEN 2
ELSE 3
END
SQL
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
end