2021-09-04 01:27:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# This model represents the surface where a CI_JOB_TOKEN can be used.
|
|
|
|
# A Scope is initialized with the project that the job token belongs to,
|
|
|
|
# and indicates what are all the other projects that the token could access.
|
|
|
|
#
|
|
|
|
# By default a job token can only access its own project, which is the same
|
|
|
|
# project that defines the scope.
|
|
|
|
# By adding ScopeLinks to the scope we can allow other projects to be accessed
|
|
|
|
# by the job token. This works as an allowlist of projects for a job token.
|
|
|
|
#
|
|
|
|
# If a project is not included in the scope we should not allow the job user
|
|
|
|
# to access it since operations using CI_JOB_TOKEN should be considered untrusted.
|
|
|
|
|
|
|
|
module Ci
|
|
|
|
module JobToken
|
|
|
|
class Scope
|
|
|
|
attr_reader :source_project
|
|
|
|
|
|
|
|
def initialize(project)
|
|
|
|
@source_project = project
|
|
|
|
end
|
|
|
|
|
|
|
|
def includes?(target_project)
|
|
|
|
# if the setting is disabled any project is considered to be in scope.
|
|
|
|
return true unless source_project.ci_job_token_scope_enabled?
|
|
|
|
|
|
|
|
target_project.id == source_project.id ||
|
|
|
|
Ci::JobToken::ProjectScopeLink.from_project(source_project).to_project(target_project).exists?
|
|
|
|
end
|
|
|
|
|
|
|
|
def all_projects
|
2022-10-11 01:57:18 +05:30
|
|
|
Project.from_union(target_projects, remove_duplicates: false)
|
2021-09-04 01:27:46 +05:30
|
|
|
end
|
2021-11-18 22:05:49 +05:30
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def target_project_ids
|
|
|
|
Ci::JobToken::ProjectScopeLink.from_project(source_project).pluck(:target_project_id)
|
|
|
|
end
|
2022-10-11 01:57:18 +05:30
|
|
|
|
|
|
|
def target_projects
|
|
|
|
[
|
|
|
|
Project.id_in(source_project),
|
|
|
|
Project.id_in(target_project_ids)
|
|
|
|
]
|
|
|
|
end
|
2021-09-04 01:27:46 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|