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