debian-mirror-gitlab/lib/api/repositories.rb

346 lines
13 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'mime/types'
module API
2021-01-03 14:25:43 +05:30
class Repositories < ::API::Base
2017-08-17 22:00:37 +05:30
include PaginationParams
2021-01-29 00:20:46 +05:30
content_type :txt, 'text/plain'
2020-06-23 00:09:42 +05:30
helpers ::API::Helpers::HeadersHelpers
2022-01-26 12:08:38 +05:30
helpers do
params :release_params do
requires :version,
type: String,
regexp: Gitlab::Regex.unbounded_semver_regex,
2023-01-13 00:05:48 +05:30
desc: 'The version of the release, using the semantic versioning format',
documentation: { example: '1.0.0' }
2022-01-26 12:08:38 +05:30
optional :from,
type: String,
2023-01-13 00:05:48 +05:30
desc: 'The first commit in the range of commits to use for the changelog',
documentation: { example: 'ed899a2f4b50b4370feeea94676502b42383c746' }
2022-01-26 12:08:38 +05:30
optional :to,
type: String,
2023-01-13 00:05:48 +05:30
desc: 'The last commit in the range of commits to use for the changelog',
documentation: { example: '6104942438c14ec7bd21c6cd5bd995272b3faff6' }
2022-01-26 12:08:38 +05:30
optional :date,
type: DateTime,
2023-01-13 00:05:48 +05:30
desc: 'The date and time of the release',
documentation: { type: 'dateTime', example: '2021-09-20T11:50:22.001+00:00' }
2022-01-26 12:08:38 +05:30
optional :trailer,
type: String,
desc: 'The Git trailer to use for determining if commits are to be included in the changelog',
2023-01-13 00:05:48 +05:30
default: ::Repositories::ChangelogService::DEFAULT_TRAILER,
documentation: { example: 'Changelog' }
2022-01-26 12:08:38 +05:30
end
end
2023-06-20 00:43:36 +05:30
before { authorize_read_code! }
2014-09-02 18:07:02 +05:30
2021-01-29 00:20:46 +05:30
feature_category :source_code_management
2017-08-17 22:00:37 +05:30
params do
2023-01-13 00:05:48 +05:30
requires :id, types: [String, Integer],
desc: 'The ID or URL-encoded path of the project',
documentation: { example: 1 }
2017-08-17 22:00:37 +05:30
end
2019-02-15 15:39:39 +05:30
resource :projects, requirements: API::NAMESPACE_OR_PROJECT_REQUIREMENTS do
2014-09-02 18:07:02 +05:30
helpers do
2022-01-26 12:08:38 +05:30
include Gitlab::RepositoryArchiveRateLimiter
2020-04-08 14:13:33 +05:30
2014-09-02 18:07:02 +05:30
def handle_project_member_errors(errors)
if errors[:project_access].any?
error!(errors[:project_access], 422)
end
2018-03-17 18:26:18 +05:30
2014-09-02 18:07:02 +05:30
not_found!
end
2017-08-17 22:00:37 +05:30
2021-10-27 15:23:28 +05:30
def assign_blob_vars!(limit:)
2023-06-20 00:43:36 +05:30
authorize_read_code!
2017-08-17 22:00:37 +05:30
@repo = user_project.repository
begin
2021-10-27 15:23:28 +05:30
@blob = Gitlab::Git::Blob.raw(@repo, params[:sha], limit: limit)
2021-06-08 01:23:25 +05:30
rescue StandardError
2017-08-17 22:00:37 +05:30
not_found! 'Blob'
end
not_found! 'Blob' unless @blob
end
2021-12-11 22:18:48 +05:30
def fetch_target_project(current_user, user_project, params)
return user_project unless params[:from_project_id].present?
MergeRequestTargetProjectFinder
.new(current_user: current_user, source_project: user_project, project_feature: :repository)
.execute(include_routes: true).find_by_id(params[:from_project_id])
end
def compare_cache_key(current_user, user_project, target_project, params)
[
user_project,
target_project,
current_user,
:repository_compare,
target_project.repository.commit(params[:from]),
user_project.repository.commit(params[:to]),
params
]
end
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Get a project repository tree' do
2018-03-17 18:26:18 +05:30
success Entities::TreeObject
2017-08-17 22:00:37 +05:30
end
params do
2023-01-13 00:05:48 +05:30
optional :ref, type: String,
desc: 'The name of a repository branch or tag, if not given the default branch is used',
documentation: { example: 'main' }
optional :path, type: String, desc: 'The path of the tree', documentation: { example: 'files/html' }
2017-08-17 22:00:37 +05:30
optional :recursive, type: Boolean, default: false, desc: 'Used to get a recursive tree'
2021-11-11 11:23:49 +05:30
2017-08-17 22:00:37 +05:30
use :pagination
2022-08-27 11:52:29 +05:30
optional :pagination, type: String, values: %w(legacy keyset none), default: 'legacy', desc: 'Specify the pagination method ("none" is only valid if "recursive" is true)'
2021-11-11 11:23:49 +05:30
2022-08-27 11:52:29 +05:30
given pagination: ->(value) { value == 'keyset' } do
2023-01-13 00:05:48 +05:30
optional :page_token, type: String,
desc: 'Record from which to start the keyset pagination',
documentation: { example: 'a1e8f8d745cc87e3a9248358d9352bb7f9a0aeba' }
2021-11-11 11:23:49 +05:30
end
2022-08-27 11:52:29 +05:30
given pagination: ->(value) { value == 'none' } do
given recursive: ->(value) { value == false } do
validates([:pagination], except_values: { value: 'none', message: 'cannot be "none" unless "recursive" is true' })
end
end
2017-08-17 22:00:37 +05:30
end
2021-12-11 22:18:48 +05:30
get ':id/repository/tree', urgency: :low do
2021-11-11 11:23:49 +05:30
tree_finder = ::Repositories::TreeFinder.new(user_project, declared_params(include_missing: false))
not_found!("Tree") unless tree_finder.commit_exists?
2014-09-02 18:07:02 +05:30
2021-11-11 11:23:49 +05:30
tree = Gitlab::Pagination::GitalyKeysetPager.new(self, user_project).paginate(tree_finder)
2015-04-26 12:48:37 +05:30
2021-11-11 11:23:49 +05:30
present tree, with: Entities::TreeObject
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Get raw blob contents from the repository'
params do
2023-01-13 00:05:48 +05:30
requires :sha, type: String,
desc: 'The commit hash', documentation: { example: '7d70e02340bac451f281cecf0a980907974bd8be' }
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
get ':id/repository/blobs/:sha/raw' do
2021-10-27 15:23:28 +05:30
# Load metadata enough to ask Workhorse to load the whole blob
assign_blob_vars!(limit: 0)
2014-09-02 18:07:02 +05:30
2020-06-23 00:09:42 +05:30
no_cache_headers
2017-08-17 22:00:37 +05:30
send_git_blob @repo, @blob
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Get a blob from the repository'
params do
2023-01-13 00:05:48 +05:30
requires :sha, type: String,
desc: 'The commit hash', documentation: { example: '7d70e02340bac451f281cecf0a980907974bd8be' }
2017-08-17 22:00:37 +05:30
end
get ':id/repository/blobs/:sha' do
2021-10-27 15:23:28 +05:30
assign_blob_vars!(limit: -1)
2017-08-17 22:00:37 +05:30
{
size: @blob.size,
encoding: "base64",
content: Base64.strict_encode64(@blob.data),
sha: @blob.id
}
end
2015-04-26 12:48:37 +05:30
2017-08-17 22:00:37 +05:30
desc 'Get an archive of the repository'
params do
2023-01-13 00:05:48 +05:30
optional :sha, type: String,
desc: 'The commit sha of the archive to be downloaded',
documentation: { example: '7d70e02340bac451f281cecf0a980907974bd8be' }
optional :format, type: String, desc: 'The archive format', documentation: { example: 'tar.gz' }
optional :path, type: String,
desc: 'Subfolder of the repository to be downloaded', documentation: { example: 'files/archives' }
2017-08-17 22:00:37 +05:30
end
2017-09-10 17:25:29 +05:30
get ':id/repository/archive', requirements: { format: Gitlab::PathRegex.archive_formats_regex } do
2022-01-26 12:08:38 +05:30
check_archive_rate_limit!(current_user, user_project) do
render_api_error!({ error: _('This archive has been requested too many times. Try again later.') }, 429)
2020-04-08 14:13:33 +05:30
end
2020-03-28 13:19:24 +05:30
not_acceptable! if Gitlab::HotlinkingDetector.intercept_hotlinking?(request)
2021-11-18 22:05:49 +05:30
send_git_archive user_project.repository, ref: params[:sha], format: params[:format], append_sha: true, path: params[:path]
2021-06-08 01:23:25 +05:30
rescue StandardError
2019-07-07 11:18:12 +05:30
not_found!('File')
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Compare two branches, tags, or commits' do
success Entities::Compare
end
params do
2023-01-13 00:05:48 +05:30
requires :from, type: String,
desc: 'The commit, branch name, or tag name to start comparison',
documentation: { example: 'main' }
requires :to, type: String,
desc: 'The commit, branch name, or tag name to stop comparison',
documentation: { example: 'feature' }
optional :from_project_id, type: Integer, desc: 'The project to compare from', documentation: { example: 1 }
2018-11-08 19:23:39 +05:30
optional :straight, type: Boolean, desc: 'Comparison method, `true` for direct comparison between `from` and `to` (`from`..`to`), `false` to compare using merge base (`from`...`to`)', default: false
2017-08-17 22:00:37 +05:30
end
2021-12-11 22:18:48 +05:30
get ':id/repository/compare', urgency: :low do
target_project = fetch_target_project(current_user, user_project, params)
2021-09-30 23:02:18 +05:30
2021-12-11 22:18:48 +05:30
if target_project.blank?
render_api_error!("Target project id:#{params[:from_project_id]} is not a fork of project id:#{params[:id]}", 400)
end
2023-05-08 21:46:49 +05:30
unless can?(current_user, :read_code, target_project)
forbidden!("You don't have access to this fork's parent project")
end
2021-09-30 23:02:18 +05:30
2021-12-11 22:18:48 +05:30
cache_key = compare_cache_key(current_user, user_project, target_project, declared_params)
2021-04-29 21:17:54 +05:30
2022-04-04 11:22:00 +05:30
cache_action(cache_key, expires_in: 1.minute) do
2021-09-30 23:02:18 +05:30
compare = CompareService.new(user_project, params[:to]).execute(target_project, params[:from], straight: params[:straight])
2020-03-13 15:44:24 +05:30
2021-09-30 23:02:18 +05:30
if compare
2022-09-01 20:07:04 +05:30
present compare, with: Entities::Compare, current_user: current_user
2021-09-30 23:02:18 +05:30
else
not_found!("Ref")
end
2020-03-13 15:44:24 +05:30
end
2014-09-02 18:07:02 +05:30
end
2017-08-17 22:00:37 +05:30
desc 'Get repository contributors' do
success Entities::Contributor
end
params do
use :pagination
2018-05-09 12:01:36 +05:30
optional :order_by, type: String, values: %w[email name commits], default: 'commits', desc: 'Return contributors ordered by `name` or `email` or `commits`'
optional :sort, type: String, values: %w[asc desc], default: 'asc', desc: 'Sort by asc (ascending) or desc (descending)'
2017-08-17 22:00:37 +05:30
end
2014-09-02 18:07:02 +05:30
get ':id/repository/contributors' do
2019-07-07 11:18:12 +05:30
contributors = ::Kaminari.paginate_array(user_project.repository.contributors(order_by: params[:order_by], sort: params[:sort]))
present paginate(contributors), with: Entities::Contributor
2021-06-08 01:23:25 +05:30
rescue StandardError
2019-07-07 11:18:12 +05:30
not_found!
2014-09-02 18:07:02 +05:30
end
2018-11-20 20:47:30 +05:30
desc 'Get the common ancestor between commits' do
success Entities::Commit
end
params do
2023-01-13 00:05:48 +05:30
requires :refs, type: Array[String],
coerce_with: ::API::Validations::Types::CommaSeparatedToArray.coerce,
desc: 'The refs to find the common ancestor of, multiple refs can be passed',
documentation: { example: 'main' }
2018-11-20 20:47:30 +05:30
end
get ':id/repository/merge_base' do
refs = params[:refs]
2018-12-13 13:39:08 +05:30
if refs.size < 2
render_api_error!('Provide at least 2 refs', 400)
2018-11-20 20:47:30 +05:30
end
merge_base = Gitlab::Git::MergeBase.new(user_project.repository, refs)
if merge_base.unknown_refs.any?
ref_noun = 'ref'.pluralize(merge_base.unknown_refs.size)
message = "Could not find #{ref_noun}: #{merge_base.unknown_refs.join(', ')}"
render_api_error!(message, 400)
end
if merge_base.commit
present merge_base.commit, with: Entities::Commit
else
not_found!("Merge Base")
end
end
2021-03-11 19:13:27 +05:30
2022-01-26 12:08:38 +05:30
desc 'Generates a changelog section for a release and returns it' do
detail 'This feature was introduced in GitLab 14.6'
2023-01-13 00:05:48 +05:30
success Entities::Changelog
2021-03-11 19:13:27 +05:30
end
params do
2022-01-26 12:08:38 +05:30
use :release_params
2022-08-13 15:12:31 +05:30
optional :config_file,
type: String,
2023-01-13 00:05:48 +05:30
documentation: { example: '.gitlab/changelog_config.yml' },
2022-08-13 15:12:31 +05:30
desc: "The file path to the configuration file as stored in the project's Git repository. Defaults to '.gitlab/changelog_config.yml'"
2022-01-26 12:08:38 +05:30
end
get ':id/repository/changelog' do
service = ::Repositories::ChangelogService.new(
user_project,
current_user,
**declared_params(include_missing: false)
)
changelog = service.execute(commit_to_changelog: false)
2021-03-11 19:13:27 +05:30
2022-01-26 12:08:38 +05:30
present changelog, with: Entities::Changelog
2022-05-07 20:08:51 +05:30
rescue Gitlab::Changelog::Error => ex
render_api_error!("Failed to generate the changelog: #{ex.message}", 422)
2022-01-26 12:08:38 +05:30
end
2021-03-11 19:13:27 +05:30
2022-01-26 12:08:38 +05:30
desc 'Generates a changelog section for a release and commits it in a changelog file' do
detail 'This feature was introduced in GitLab 13.9'
2023-01-13 00:05:48 +05:30
success code: 200
2022-01-26 12:08:38 +05:30
end
params do
use :release_params
2021-03-11 19:13:27 +05:30
optional :branch,
type: String,
2023-01-13 00:05:48 +05:30
desc: 'The branch to commit the changelog changes to',
documentation: { example: 'main' }
2021-03-11 19:13:27 +05:30
2022-08-13 15:12:31 +05:30
optional :config_file,
type: String,
2023-01-13 00:05:48 +05:30
documentation: { example: '.gitlab/changelog_config.yml' },
2022-08-13 15:12:31 +05:30
desc: "The file path to the configuration file as stored in the project's Git repository. Defaults to '.gitlab/changelog_config.yml'"
2021-03-11 19:13:27 +05:30
optional :file,
type: String,
desc: 'The file to commit the changelog changes to',
2023-01-13 00:05:48 +05:30
default: ::Repositories::ChangelogService::DEFAULT_FILE,
documentation: { example: 'CHANGELOG.md' }
2021-03-11 19:13:27 +05:30
optional :message,
type: String,
2023-01-13 00:05:48 +05:30
desc: 'The commit message to use when committing the changelog',
documentation: { example: 'Initial commit' }
2021-03-11 19:13:27 +05:30
end
post ':id/repository/changelog' do
2021-06-08 01:23:25 +05:30
branch = params[:branch] || user_project.default_branch_or_main
2021-03-11 19:13:27 +05:30
access = Gitlab::UserAccess.new(current_user, container: user_project)
unless access.can_push_to_branch?(branch)
forbidden!("You are not allowed to commit a changelog on this branch")
end
service = ::Repositories::ChangelogService.new(
user_project,
current_user,
**declared_params(include_missing: false)
)
2022-01-26 12:08:38 +05:30
service.execute(commit_to_changelog: true)
2021-03-11 19:13:27 +05:30
status(200)
rescue Gitlab::Changelog::Error => ex
render_api_error!("Failed to generate the changelog: #{ex.message}", 422)
end
2014-09-02 18:07:02 +05:30
end
end
end
2022-01-26 12:08:38 +05:30
API::Repositories.prepend_mod