74 lines
2.1 KiB
Ruby
74 lines
2.1 KiB
Ruby
|
# frozen_string_literal: true
|
||
|
|
||
|
module Suggestions
|
||
|
class ApplyService < ::BaseService
|
||
|
def initialize(current_user)
|
||
|
@current_user = current_user
|
||
|
end
|
||
|
|
||
|
def execute(suggestion)
|
||
|
unless suggestion.appliable?
|
||
|
return error('Suggestion is not appliable')
|
||
|
end
|
||
|
|
||
|
unless latest_source_head?(suggestion)
|
||
|
return error('The file has been changed')
|
||
|
end
|
||
|
|
||
|
params = file_update_params(suggestion)
|
||
|
result = ::Files::UpdateService.new(suggestion.project, @current_user, params).execute
|
||
|
|
||
|
if result[:status] == :success
|
||
|
suggestion.update(commit_id: result[:result], applied: true)
|
||
|
end
|
||
|
|
||
|
result
|
||
|
rescue Files::UpdateService::FileChangedError
|
||
|
error('The file has been changed')
|
||
|
end
|
||
|
|
||
|
private
|
||
|
|
||
|
# Checks whether the latest source branch HEAD matches with
|
||
|
# the position HEAD we're using to update the file content. Since
|
||
|
# the persisted HEAD is updated async (for MergeRequest),
|
||
|
# it's more consistent to fetch this data directly from the
|
||
|
# repository.
|
||
|
def latest_source_head?(suggestion)
|
||
|
suggestion.position.head_sha == suggestion.noteable.source_branch_sha
|
||
|
end
|
||
|
|
||
|
def file_update_params(suggestion)
|
||
|
blob = suggestion.diff_file.new_blob
|
||
|
file_path = suggestion.file_path
|
||
|
branch_name = suggestion.branch
|
||
|
file_content = new_file_content(suggestion, blob)
|
||
|
commit_message = "Apply suggestion to #{file_path}"
|
||
|
|
||
|
file_last_commit =
|
||
|
Gitlab::Git::Commit.last_for_path(suggestion.project.repository,
|
||
|
blob.commit_id,
|
||
|
blob.path)
|
||
|
|
||
|
{
|
||
|
file_path: file_path,
|
||
|
branch_name: branch_name,
|
||
|
start_branch: branch_name,
|
||
|
commit_message: commit_message,
|
||
|
file_content: file_content,
|
||
|
last_commit_sha: file_last_commit&.id
|
||
|
}
|
||
|
end
|
||
|
|
||
|
def new_file_content(suggestion, blob)
|
||
|
range = suggestion.from_line_index..suggestion.to_line_index
|
||
|
|
||
|
blob.load_all_data!
|
||
|
content = blob.data.lines
|
||
|
content[range] = suggestion.to_content
|
||
|
|
||
|
content.join
|
||
|
end
|
||
|
end
|
||
|
end
|