2014-02-18 05:41:21 -05:00
|
|
|
require 'mime/types'
|
|
|
|
|
|
|
|
module API
|
|
|
|
class Commits < Grape::API
|
2016-11-21 14:15:46 -05:00
|
|
|
include PaginationParams
|
|
|
|
|
2014-02-18 05:41:21 -05:00
|
|
|
before { authenticate! }
|
|
|
|
before { authorize! :download_code, user_project }
|
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
params do
|
|
|
|
requires :id, type: String, desc: 'The ID of a project'
|
|
|
|
end
|
2014-02-18 05:41:21 -05:00
|
|
|
resource :projects do
|
2016-10-15 06:09:02 -04:00
|
|
|
desc 'Get a project repository commits' do
|
|
|
|
success Entities::RepoCommit
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
optional :ref_name, type: String, desc: 'The name of a repository branch or tag, if not given the default branch is used'
|
2017-02-20 09:54:37 -05:00
|
|
|
optional :since, type: DateTime, desc: 'Only commits after or on this date will be returned'
|
|
|
|
optional :until, type: DateTime, desc: 'Only commits before or on this date will be returned'
|
2016-10-15 06:09:02 -04:00
|
|
|
optional :page, type: Integer, default: 0, desc: 'The page for pagination'
|
|
|
|
optional :per_page, type: Integer, default: 20, desc: 'The number of results per page'
|
2016-06-20 21:37:40 -04:00
|
|
|
optional :path, type: String, desc: 'The file path'
|
2016-10-15 06:09:02 -04:00
|
|
|
end
|
2014-02-18 05:41:21 -05:00
|
|
|
get ":id/repository/commits" do
|
|
|
|
ref = params[:ref_name] || user_project.try(:default_branch) || 'master'
|
2016-10-15 06:09:02 -04:00
|
|
|
offset = params[:page] * params[:per_page]
|
|
|
|
|
|
|
|
commits = user_project.repository.commits(ref,
|
2016-06-20 21:37:40 -04:00
|
|
|
path: params[:path],
|
2016-10-15 06:09:02 -04:00
|
|
|
limit: params[:per_page],
|
|
|
|
offset: offset,
|
|
|
|
after: params[:since],
|
|
|
|
before: params[:until])
|
2014-02-18 05:41:21 -05:00
|
|
|
|
|
|
|
present commits, with: Entities::RepoCommit
|
|
|
|
end
|
|
|
|
|
2016-08-29 19:58:32 -04:00
|
|
|
desc 'Commit multiple file changes as one commit' do
|
2016-10-15 06:09:02 -04:00
|
|
|
success Entities::RepoCommitDetail
|
2016-08-29 19:58:32 -04:00
|
|
|
detail 'This feature was introduced in GitLab 8.13'
|
|
|
|
end
|
|
|
|
params do
|
2017-02-02 09:24:30 -05:00
|
|
|
requires :branch, type: String, desc: 'The name of branch'
|
2016-08-29 19:58:32 -04:00
|
|
|
requires :commit_message, type: String, desc: 'Commit message'
|
2016-11-24 07:30:53 -05:00
|
|
|
requires :actions, type: Array[Hash], desc: 'Actions to perform in commit'
|
2016-08-29 19:58:32 -04:00
|
|
|
optional :author_email, type: String, desc: 'Author email for commit'
|
|
|
|
optional :author_name, type: String, desc: 'Author name for commit'
|
|
|
|
end
|
|
|
|
post ":id/repository/commits" do
|
|
|
|
authorize! :push_code, user_project
|
|
|
|
|
2017-02-02 09:24:30 -05:00
|
|
|
attrs = declared_params.merge(start_branch: declared_params[:branch], target_branch: declared_params[:branch])
|
|
|
|
|
2016-08-29 19:58:32 -04:00
|
|
|
attrs[:actions].map! do |action|
|
|
|
|
action[:action] = action[:action].to_sym
|
|
|
|
action[:file_path].slice!(0) if action[:file_path] && action[:file_path].start_with?('/')
|
|
|
|
action[:previous_path].slice!(0) if action[:previous_path] && action[:previous_path].start_with?('/')
|
|
|
|
action
|
|
|
|
end
|
|
|
|
|
|
|
|
result = ::Files::MultiService.new(user_project, current_user, attrs).execute
|
|
|
|
|
|
|
|
if result[:status] == :success
|
|
|
|
commit_detail = user_project.repository.commits(result[:result], limit: 1).first
|
|
|
|
present commit_detail, with: Entities::RepoCommitDetail
|
|
|
|
else
|
|
|
|
render_api_error!(result[:message], 400)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
desc 'Get a specific commit of a project' do
|
|
|
|
success Entities::RepoCommitDetail
|
|
|
|
failure [[404, 'Not Found']]
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :sha, type: String, desc: 'A commit sha, or the name of a branch or tag'
|
|
|
|
end
|
2014-02-18 05:41:21 -05:00
|
|
|
get ":id/repository/commits/:sha" do
|
2016-10-15 06:09:02 -04:00
|
|
|
commit = user_project.commit(params[:sha])
|
|
|
|
|
2014-02-18 05:41:21 -05:00
|
|
|
not_found! "Commit" unless commit
|
2016-10-15 06:09:02 -04:00
|
|
|
|
2014-02-18 05:41:21 -05:00
|
|
|
present commit, with: Entities::RepoCommitDetail
|
|
|
|
end
|
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
desc 'Get the diff for a specific commit of a project' do
|
|
|
|
failure [[404, 'Not Found']]
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :sha, type: String, desc: 'A commit sha, or the name of a branch or tag'
|
|
|
|
end
|
2014-02-18 05:41:21 -05:00
|
|
|
get ":id/repository/commits/:sha/diff" do
|
2016-10-15 06:09:02 -04:00
|
|
|
commit = user_project.commit(params[:sha])
|
|
|
|
|
2014-02-18 05:41:21 -05:00
|
|
|
not_found! "Commit" unless commit
|
2016-10-15 06:09:02 -04:00
|
|
|
|
2016-07-27 13:00:34 -04:00
|
|
|
commit.raw_diffs.to_a
|
2014-02-18 05:41:21 -05:00
|
|
|
end
|
2014-06-27 10:48:30 -04:00
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
desc "Get a commit's comments" do
|
|
|
|
success Entities::CommitNote
|
|
|
|
failure [[404, 'Not Found']]
|
|
|
|
end
|
|
|
|
params do
|
2016-11-21 14:15:46 -05:00
|
|
|
use :pagination
|
2016-10-15 06:09:02 -04:00
|
|
|
requires :sha, type: String, desc: 'A commit sha, or the name of a branch or tag'
|
|
|
|
end
|
2014-06-27 10:48:30 -04:00
|
|
|
get ':id/repository/commits/:sha/comments' do
|
2016-10-15 06:09:02 -04:00
|
|
|
commit = user_project.commit(params[:sha])
|
|
|
|
|
2014-06-27 10:48:30 -04:00
|
|
|
not_found! 'Commit' unless commit
|
2017-02-15 08:24:18 -05:00
|
|
|
notes = user_project.notes.where(commit_id: commit.id).order(:created_at)
|
2016-10-15 06:09:02 -04:00
|
|
|
|
2014-06-27 10:48:30 -04:00
|
|
|
present paginate(notes), with: Entities::CommitNote
|
|
|
|
end
|
|
|
|
|
2016-12-12 07:04:13 -05:00
|
|
|
desc 'Cherry pick commit into a branch' do
|
|
|
|
detail 'This feature was introduced in GitLab 8.15'
|
|
|
|
success Entities::RepoCommit
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :sha, type: String, desc: 'A commit sha to be cherry picked'
|
|
|
|
requires :branch, type: String, desc: 'The name of the branch'
|
|
|
|
end
|
|
|
|
post ':id/repository/commits/:sha/cherry_pick' do
|
|
|
|
authorize! :push_code, user_project
|
|
|
|
|
|
|
|
commit = user_project.commit(params[:sha])
|
|
|
|
not_found!('Commit') unless commit
|
|
|
|
|
|
|
|
branch = user_project.repository.find_branch(params[:branch])
|
|
|
|
not_found!('Branch') unless branch
|
|
|
|
|
|
|
|
commit_params = {
|
|
|
|
commit: commit,
|
|
|
|
create_merge_request: false,
|
|
|
|
target_branch: params[:branch]
|
|
|
|
}
|
|
|
|
|
|
|
|
result = ::Commits::CherryPickService.new(user_project, current_user, commit_params).execute
|
|
|
|
|
|
|
|
if result[:status] == :success
|
|
|
|
branch = user_project.repository.find_branch(params[:branch])
|
|
|
|
present user_project.repository.commit(branch.dereferenced_target), with: Entities::RepoCommit
|
|
|
|
else
|
|
|
|
render_api_error!(result[:message], 400)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
desc 'Post comment to commit' do
|
|
|
|
success Entities::CommitNote
|
|
|
|
end
|
|
|
|
params do
|
|
|
|
requires :sha, type: String, regexp: /\A\h{6,40}\z/, desc: "The commit's SHA"
|
|
|
|
requires :note, type: String, desc: 'The text of the comment'
|
|
|
|
optional :path, type: String, desc: 'The file path'
|
|
|
|
given :path do
|
|
|
|
requires :line, type: Integer, desc: 'The line number'
|
|
|
|
requires :line_type, type: String, values: ['new', 'old'], default: 'new', desc: 'The type of the line'
|
|
|
|
end
|
|
|
|
end
|
2014-06-27 10:48:30 -04:00
|
|
|
post ':id/repository/commits/:sha/comments' do
|
2016-10-15 06:09:02 -04:00
|
|
|
commit = user_project.commit(params[:sha])
|
2014-06-27 10:48:30 -04:00
|
|
|
not_found! 'Commit' unless commit
|
2016-10-15 06:09:02 -04:00
|
|
|
|
2014-06-27 10:48:30 -04:00
|
|
|
opts = {
|
|
|
|
note: params[:note],
|
|
|
|
noteable_type: 'Commit',
|
|
|
|
commit_id: commit.id
|
|
|
|
}
|
|
|
|
|
2016-10-15 06:09:02 -04:00
|
|
|
if params[:path]
|
2016-07-27 13:00:34 -04:00
|
|
|
commit.raw_diffs(all_diffs: true).each do |diff|
|
2014-06-27 10:48:30 -04:00
|
|
|
next unless diff.new_path == params[:path]
|
2016-03-03 12:38:44 -05:00
|
|
|
lines = Gitlab::Diff::Parser.new.parse(diff.diff.each_line)
|
2014-06-27 10:48:30 -04:00
|
|
|
|
|
|
|
lines.each do |line|
|
2016-10-15 06:09:02 -04:00
|
|
|
next unless line.new_pos == params[:line] && line.type == params[:line_type]
|
2014-06-27 10:48:30 -04:00
|
|
|
break opts[:line_code] = Gitlab::Diff::LineCode.generate(diff.new_path, line.new_pos, line.old_pos)
|
|
|
|
end
|
|
|
|
|
|
|
|
break if opts[:line_code]
|
|
|
|
end
|
2016-05-10 18:41:46 -04:00
|
|
|
|
|
|
|
opts[:type] = LegacyDiffNote.name if opts[:line_code]
|
2014-06-27 10:48:30 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
note = ::Notes::CreateService.new(user_project, current_user, opts).execute
|
|
|
|
|
|
|
|
if note.save
|
|
|
|
present note, with: Entities::CommitNote
|
|
|
|
else
|
2015-01-07 04:46:00 -05:00
|
|
|
render_api_error!("Failed to save note #{note.errors.messages}", 400)
|
2014-06-27 10:48:30 -04:00
|
|
|
end
|
|
|
|
end
|
2014-02-18 05:41:21 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|