gitlab-org--gitlab-foss/app/models/ability.rb

612 lines
16 KiB
Ruby
Raw Normal View History

2011-10-08 21:36:38 +00:00
class Ability
2012-10-09 00:10:04 +00:00
class << self
# rubocop: disable Metrics/CyclomaticComplexity
2013-01-25 09:30:49 +00:00
def allowed(user, subject)
return anonymous_abilities(user, subject) if user.nil?
2015-11-17 15:24:02 +00:00
return [] unless user.is_a?(User)
return [] if user.blocked?
2013-01-25 09:30:49 +00:00
2016-02-04 13:43:52 +00:00
case subject
when CommitStatus then commit_status_abilities(user, subject)
when Project then project_abilities(user, subject)
when Issue then issue_abilities(user, subject)
when Note then note_abilities(user, subject)
when ProjectSnippet then project_snippet_abilities(user, subject)
when PersonalSnippet then personal_snippet_abilities(user, subject)
when MergeRequest then merge_request_abilities(user, subject)
when Group then group_abilities(user, subject)
when Namespace then namespace_abilities(user, subject)
when GroupMember then group_member_abilities(user, subject)
when ProjectMember then project_member_abilities(user, subject)
2016-03-30 20:14:21 +00:00
when User then user_abilities
2016-06-15 10:12:26 +00:00
when ExternalIssue, Deployment, Environment then project_abilities(user, subject.project)
when Ci::Runner then runner_abilities(user, subject)
2015-11-12 08:43:30 +00:00
else []
2013-01-25 09:30:49 +00:00
end.concat(global_abilities(user))
end
# Given a list of users and a project this method returns the users that can
# read the given project.
def users_that_can_read_project(users, project)
if project.public?
users
else
users.select do |user|
if user.admin?
true
elsif project.internal? && !user.external?
true
elsif project.owner == user
true
elsif project.team.members.include?(user)
true
else
false
end
end
end
end
# Returns an Array of Issues that can be read by the given user.
#
# issues - The issues to reduce down to those readable by the user.
# user - The User for which to check the issues
def issues_readable_by_user(issues, user = nil)
return issues if user && user.admin?
issues.select { |issue| issue.visible_to_user?(user) }
end
# List of possible abilities for anonymous user
def anonymous_abilities(user, subject)
if subject.is_a?(PersonalSnippet)
anonymous_personal_snippet_abilities(subject)
elsif subject.is_a?(ProjectSnippet)
anonymous_project_snippet_abilities(subject)
elsif subject.is_a?(CommitStatus)
2016-02-04 10:09:42 +00:00
anonymous_commit_status_abilities(subject)
elsif subject.is_a?(Project) || subject.respond_to?(:project)
anonymous_project_abilities(subject)
elsif subject.is_a?(Group) || subject.respond_to?(:group)
anonymous_group_abilities(subject)
elsif subject.is_a?(User)
2016-03-30 20:14:21 +00:00
anonymous_user_abilities
2015-11-17 15:24:02 +00:00
else
[]
end
end
def anonymous_project_abilities(subject)
2015-11-17 15:24:02 +00:00
project = if subject.is_a?(Project)
subject
else
subject.project
end
if project && project.public?
rules = [
:read_project,
:read_wiki,
:read_label,
:read_milestone,
:read_project_snippet,
:read_project_member,
:read_merge_request,
:read_note,
:read_pipeline,
:read_commit_status,
2016-05-14 23:23:31 +00:00
:read_container_image,
:download_code
]
2016-02-04 10:09:42 +00:00
# Allow to read builds by anonymous user if guests are allowed
rules << :read_build if project.public_builds?
2016-03-17 19:38:51 +00:00
# Allow to read issues by anonymous user if issue is not confidential
rules << :read_issue unless subject.is_a?(Issue) && subject.confidential?
rules - project_disabled_features_rules(project)
else
[]
end
end
2016-02-04 10:09:42 +00:00
def anonymous_commit_status_abilities(subject)
rules = anonymous_project_abilities(subject.project)
# If subject is Ci::Build which inherits from CommitStatus filter the abilities
rules = filter_build_abilities(rules) if subject.is_a?(Ci::Build)
rules
end
def anonymous_group_abilities(subject)
2016-03-29 15:24:42 +00:00
rules = []
2015-11-17 15:24:02 +00:00
group = if subject.is_a?(Group)
subject
else
subject.group
end
2016-04-07 19:36:26 +00:00
rules << :read_group if group.public?
2016-03-29 15:24:42 +00:00
rules
end
def anonymous_personal_snippet_abilities(snippet)
if snippet.public?
[:read_personal_snippet]
else
[]
end
end
def anonymous_project_snippet_abilities(snippet)
if snippet.public?
[:read_project_snippet]
else
[]
end
end
2016-03-30 20:14:21 +00:00
def anonymous_user_abilities
[:read_user] unless restricted_public_level?
2016-03-29 15:24:42 +00:00
end
2013-01-25 09:30:49 +00:00
def global_abilities(user)
rules = []
rules << :create_group if user.can_create_group
rules << :read_users_list
2013-01-25 09:30:49 +00:00
rules
2011-10-08 21:36:38 +00:00
end
2012-10-09 00:10:04 +00:00
def project_abilities(user, project)
rules = []
2014-06-14 10:05:25 +00:00
key = "/user/#{user.id}/project/#{project.id}"
2014-06-14 10:05:25 +00:00
RequestStore.store[key] ||= begin
# Push abilities on the users team role
rules.push(*project_team_rules(project.team, user))
2011-10-08 21:36:38 +00:00
owner = user.admin? ||
project.owner == user ||
(project.group && project.group.has_owner?(user))
2016-03-20 20:03:53 +00:00
if owner
2016-03-20 20:03:53 +00:00
rules.push(*project_owner_rules)
end
if project.public? || (project.internal? && !user.external?)
2014-10-09 07:47:47 +00:00
rules.push(*public_project_rules)
2016-02-04 13:43:52 +00:00
# Allow to read builds for internal projects
rules << :read_build if project.public_builds?
unless owner || project.team.member?(user) || project_group_member?(project, user)
rules << :request_access if project.request_access_enabled
end
2014-06-14 10:05:25 +00:00
end
2014-06-14 10:05:25 +00:00
if project.archived?
rules -= project_archived_rules
end
rules - project_disabled_features_rules(project)
end
end
def project_team_rules(team, user)
# Rules based on role in project
if team.master?(user)
project_master_rules
elsif team.developer?(user)
project_dev_rules
elsif team.reporter?(user)
project_report_rules
elsif team.guest?(user)
project_guest_rules
else
[]
end
end
def public_project_rules
@public_project_rules ||= project_guest_rules + [
:download_code,
:fork_project,
2016-06-21 08:11:07 +00:00
:read_commit_status,
:read_pipeline,
:read_container_image
]
end
def project_guest_rules
@project_guest_rules ||= [
2012-10-09 00:10:04 +00:00
:read_project,
:read_wiki,
:read_issue,
:read_label,
2012-10-09 00:10:04 +00:00
:read_milestone,
2013-03-25 07:20:14 +00:00
:read_project_snippet,
:read_project_member,
2012-10-09 00:10:04 +00:00
:read_merge_request,
:read_note,
:create_project,
:create_issue,
2016-03-21 23:09:20 +00:00
:create_note,
:upload_file
]
end
2012-02-20 18:16:55 +00:00
def project_report_rules
@project_report_rules ||= project_guest_rules + [
2012-10-09 00:10:04 +00:00
:download_code,
2013-06-04 15:50:42 +00:00
:fork_project,
:create_project_snippet,
:update_issue,
:admin_issue,
:admin_label,
:read_commit_status,
:read_build,
2016-05-14 23:23:31 +00:00
:read_container_image,
2016-04-12 14:16:39 +00:00
:read_pipeline,
2016-06-10 21:36:54 +00:00
:read_environment,
:read_deployment
]
end
2012-02-20 18:16:55 +00:00
def project_dev_rules
@project_dev_rules ||= project_report_rules + [
:admin_merge_request,
:update_merge_request,
:create_commit_status,
:update_commit_status,
:create_build,
:update_build,
2016-04-12 14:16:39 +00:00
:create_pipeline,
:update_pipeline,
:create_merge_request,
:create_wiki,
:push_code,
2016-05-14 23:23:31 +00:00
:create_container_image,
:update_container_image,
2016-06-10 21:36:54 +00:00
:create_environment,
:create_deployment
]
end
def project_archived_rules
@project_archived_rules ||= [
:create_merge_request,
:push_code,
:push_code_to_protected_branches,
:update_merge_request,
:admin_merge_request
]
end
def project_master_rules
@project_master_rules ||= project_dev_rules + [
:push_code_to_protected_branches,
:update_project_snippet,
2016-06-14 16:34:48 +00:00
:update_environment,
:update_deployment,
2012-10-09 00:10:04 +00:00
:admin_milestone,
2013-03-25 07:20:14 +00:00
:admin_project_snippet,
:admin_project_member,
2012-10-09 00:10:04 +00:00
:admin_merge_request,
:admin_note,
:admin_wiki,
:admin_project,
:admin_commit_status,
2016-04-12 14:16:39 +00:00
:admin_build,
2016-05-14 23:23:31 +00:00
:admin_container_image,
2016-06-10 21:36:54 +00:00
:admin_pipeline,
:admin_environment,
:admin_deployment
]
end
2011-10-08 21:36:38 +00:00
2016-03-20 20:03:53 +00:00
def project_owner_rules
@project_owner_rules ||= project_master_rules + [
:change_namespace,
:change_visibility_level,
:rename_project,
:remove_project,
:archive_project,
2016-03-14 20:46:44 +00:00
:remove_fork_project,
2016-03-21 13:12:52 +00:00
:destroy_merge_request,
:destroy_issue
]
2012-10-09 00:10:04 +00:00
end
2011-10-17 10:39:03 +00:00
def project_disabled_features_rules(project)
rules = []
unless project.issues_enabled
rules += named_abilities('issue')
end
unless project.merge_requests_enabled
rules += named_abilities('merge_request')
end
unless project.issues_enabled or project.merge_requests_enabled
rules += named_abilities('label')
rules += named_abilities('milestone')
end
unless project.snippets_enabled
rules += named_abilities('project_snippet')
end
unless project.wiki_enabled
rules += named_abilities('wiki')
end
unless project.builds_enabled
rules += named_abilities('build')
2016-04-12 14:16:39 +00:00
rules += named_abilities('pipeline')
2016-06-10 21:36:54 +00:00
rules += named_abilities('environment')
rules += named_abilities('deployment')
end
unless project.container_registry_enabled
2016-05-14 23:23:31 +00:00
rules += named_abilities('container_image')
end
rules
end
def group_abilities(user, group)
rules = []
2016-03-02 22:13:50 +00:00
rules << :read_group if can_read_group?(user, group)
owner = user.admin? || group.has_owner?(user)
master = owner || group.has_master?(user)
2016-03-20 20:03:53 +00:00
# Only group masters and group owners can create new projects
if master
rules += [
:create_projects,
2016-03-02 22:13:50 +00:00
:admin_milestones
]
end
# Only group owner and administrators can admin group
if owner
2015-11-17 14:49:37 +00:00
rules += [
:admin_group,
:admin_namespace,
2016-03-02 22:13:50 +00:00
:admin_group_member,
:change_visibility_level
2015-11-17 14:49:37 +00:00
]
end
if group.public? || (group.internal? && !user.external?)
rules << :request_access if group.request_access_enabled && group.users.exclude?(user)
end
rules.flatten
end
2016-03-02 22:13:50 +00:00
def can_read_group?(user, group)
2016-03-21 23:09:20 +00:00
return true if user.admin?
return true if group.public?
return true if group.internal? && !user.external?
return true if group.users.include?(user)
GroupProjectsFinder.new(group).execute(user).any?
2016-03-02 22:13:50 +00:00
end
def can_edit_note?(user, note)
2016-07-25 13:21:55 +00:00
return false if !note.editable? || !user.present?
return true if note.author == user || user.admin?
if note.project
max_access_level = note.project.team.max_member_access(user.id)
max_access_level >= Gitlab::Access::MASTER
else
false
end
end
def namespace_abilities(user, namespace)
2013-06-21 19:44:40 +00:00
rules = []
# Only namespace owner and administrators can admin it
2013-06-21 19:44:40 +00:00
if namespace.owner == user || user.admin?
2015-11-17 14:49:37 +00:00
rules += [
:create_projects,
:admin_namespace
]
2013-06-21 19:44:40 +00:00
end
rules.flatten
end
[:issue, :merge_request].each do |name|
2011-10-17 10:39:03 +00:00
define_method "#{name}_abilities" do |user, subject|
rules = []
if subject.author == user || (subject.respond_to?(:assignee) && subject.assignee == user)
rules += [
2011-10-17 10:39:03 +00:00
:"read_#{name}",
:"update_#{name}",
2011-10-17 10:39:03 +00:00
]
end
rules += project_abilities(user, subject.project)
2016-03-17 19:38:51 +00:00
rules = filter_confidential_issues_abilities(user, subject, rules) if subject.is_a?(Issue)
rules
end
end
def note_abilities(user, note)
rules = []
if note.author == user
rules += [
:read_note,
:update_note,
:admin_note
]
end
if note.respond_to?(:project) && note.project
rules += project_abilities(user, note.project)
2011-10-17 10:39:03 +00:00
end
rules
2011-10-17 10:39:03 +00:00
end
2014-02-07 16:59:55 +00:00
def personal_snippet_abilities(user, snippet)
rules = []
if snippet.author == user
rules += [
:read_personal_snippet,
:update_personal_snippet,
:admin_personal_snippet
]
end
if snippet.public? || (snippet.internal? && !user.external?)
rules << :read_personal_snippet
end
rules
end
def project_snippet_abilities(user, snippet)
rules = []
if snippet.author == user || user.admin?
rules += [
:read_project_snippet,
:update_project_snippet,
:admin_project_snippet
]
end
if snippet.public? || (snippet.internal? && !user.external?) || (snippet.private? && snippet.project.team.member?(user))
rules << :read_project_snippet
end
rules
end
def group_member_abilities(user, subject)
2014-02-07 16:59:55 +00:00
rules = []
target_user = subject.user
group = subject.group
2015-11-17 14:49:37 +00:00
unless group.last_owner?(target_user)
can_manage = group_abilities(user, group).include?(:admin_group_member)
if can_manage
2015-11-17 14:49:37 +00:00
rules << :update_group_member
rules << :destroy_group_member
elsif user == target_user
2015-11-17 14:49:37 +00:00
rules << :destroy_group_member
end
2014-02-07 16:59:55 +00:00
end
2014-02-07 16:59:55 +00:00
rules
end
def project_member_abilities(user, subject)
rules = []
target_user = subject.user
project = subject.project
unless target_user == project.owner
2015-11-17 14:49:37 +00:00
can_manage = project_abilities(user, project).include?(:admin_project_member)
if can_manage
2015-11-17 14:49:37 +00:00
rules << :update_project_member
rules << :destroy_project_member
elsif user == target_user
2015-11-17 14:49:37 +00:00
rules << :destroy_project_member
end
end
2015-11-17 14:49:37 +00:00
rules
end
2016-02-04 10:09:42 +00:00
def commit_status_abilities(user, subject)
rules = project_abilities(user, subject.project)
# If subject is Ci::Build which inherits from CommitStatus filter the abilities
rules = filter_build_abilities(rules) if subject.is_a?(Ci::Build)
rules
end
def filter_build_abilities(rules)
# If we can't read build we should also not have that
# ability when looking at this in context of commit_status
2016-02-04 13:43:52 +00:00
%w(read create update admin).each do |rule|
rules.delete(:"#{rule}_commit_status") unless rules.include?(:"#{rule}_build")
end
rules
end
def runner_abilities(user, runner)
if user.is_admin?
[:assign_runner]
elsif runner.is_shared? || runner.locked?
[]
elsif user.ci_authorized_runners.include?(runner)
[:assign_runner]
else
[]
end
end
2016-03-30 20:14:21 +00:00
def user_abilities
2016-03-29 15:24:42 +00:00
[:read_user]
end
def abilities
@abilities ||= begin
abilities = Six.new
abilities << self
abilities
end
end
private
2016-03-30 20:14:21 +00:00
def restricted_public_level?
2016-04-06 21:09:24 +00:00
current_application_settings.restricted_visibility_levels.include?(Gitlab::VisibilityLevel::PUBLIC)
2016-03-30 20:14:21 +00:00
end
def named_abilities(name)
[
:"read_#{name}",
:"create_#{name}",
:"update_#{name}",
:"admin_#{name}"
]
end
2016-03-17 19:38:51 +00:00
def filter_confidential_issues_abilities(user, issue, rules)
return rules if user.admin? || !issue.confidential?
unless issue.author == user || issue.assignee == user || issue.project.team.member?(user, Gitlab::Access::REPORTER)
2016-03-17 19:38:51 +00:00
rules.delete(:admin_issue)
rules.delete(:read_issue)
rules.delete(:update_issue)
end
rules
end
def project_group_member?(project, user)
project.group &&
(
project.group.members.exists?(user_id: user.id) ||
project.group.requesters.exists?(user_id: user.id)
)
end
2011-10-17 10:39:03 +00:00
end
2011-10-08 21:36:38 +00:00
end