forked from Gitlink/forgeplus
176 lines
5.4 KiB
Ruby
176 lines
5.4 KiB
Ruby
class PullRequests::CreateService < ApplicationService
|
|
|
|
attr_reader :current_user, :owner, :project, :params
|
|
attr_accessor :pull_issue, :pull_request
|
|
|
|
def initialize(current_user, owner, project, params)
|
|
@owner = owner
|
|
@project = project
|
|
@params = params
|
|
@current_user = current_user
|
|
end
|
|
|
|
def call
|
|
ActiveRecord::Base.transaction do
|
|
validate!
|
|
compare_head_base!
|
|
save_pull_issue!
|
|
save_pull_request!
|
|
save_issue_tags_relates!
|
|
save_tiding!
|
|
save_project_trend!
|
|
save_custom_journal_detail!
|
|
end
|
|
|
|
[pull_request, gitea_pull_request]
|
|
end
|
|
|
|
def pull_issue_params
|
|
{
|
|
user: @current_user,
|
|
project: @project,
|
|
subject: @params[:title],
|
|
description: @params[:body],
|
|
assigned_to_id: @params[:assigned_to_id],
|
|
fixed_version_id: @params[:fixed_version_id],
|
|
issue_tags_value: @params[:issue_tag_ids].present? ? @params[:issue_tag_ids].join(",") : "",
|
|
priority_id: @params[:priority_id],
|
|
issue_classify: "pull_request",
|
|
issue_type: @params[:issue_type] || "1",
|
|
tracker_id: 2,
|
|
status_id: 1,
|
|
}
|
|
end
|
|
|
|
def pull_issue
|
|
@pull_issue ||= Issue.new(pull_issue_params.compact)
|
|
end
|
|
|
|
def save_pull_issue!
|
|
pull_issue.save!
|
|
end
|
|
|
|
def pull_request
|
|
@pull_request ||= @project.pull_requests.new(pull_request_params.compact)
|
|
end
|
|
|
|
def save_pull_request!
|
|
pull_request.save!
|
|
end
|
|
|
|
def save_issue_tags_relates!
|
|
if issue_tag_ids.size > 1
|
|
raise "最多只能创建一个标记。"
|
|
else
|
|
issue_tag_ids.each do |tag|
|
|
IssueTagsRelate.create!(issue_id: pull_issue.id, issue_tag_id: tag)
|
|
end
|
|
end
|
|
end
|
|
|
|
def issue_tag_ids
|
|
Array(@params[:issue_tag_ids])
|
|
end
|
|
|
|
def save_tiding!
|
|
if @params[:assigned_to_id].present?
|
|
Tiding.create!(user_id: @params[:assigned_to_id],
|
|
trigger_user: @current_user,
|
|
container: pull_request,
|
|
parent_container: @project,
|
|
tiding_type: 'pull_request',
|
|
status: 0)
|
|
end
|
|
end
|
|
|
|
def save_project_trend!
|
|
project_trend.save!
|
|
end
|
|
|
|
def project_trend
|
|
@project_trend ||= pull_request.project_trends.new(
|
|
user: @current_user,
|
|
project: @project,
|
|
action_type: "create")
|
|
end
|
|
|
|
def pull_request_params
|
|
base_pull_params.merge({
|
|
user: @current_user,
|
|
issue: pull_issue,
|
|
fork_project_id: @params[:fork_project_id],
|
|
is_original: is_original,
|
|
files_count: @params[:files_count] || 0,
|
|
commits_count: @params[:commits_count] || 0
|
|
})
|
|
end
|
|
|
|
def save_custom_journal_detail!
|
|
if @params[:title].to_s.include?("WIP:")
|
|
pull_issue.custom_journal_detail("WIP", "", "这个合并请求被标记为尚未完成的工作。完成后请从标题中移除WIP:前缀。", @current_user&.id)
|
|
end
|
|
end
|
|
|
|
def gitea_pull_request
|
|
@gitea_pull_request ||= create_gitea_pull_request!
|
|
end
|
|
|
|
def create_gitea_pull_request!
|
|
@gitea_pull_request =
|
|
Gitea::PullRequest::CreateService.call(
|
|
@current_user&.gitea_token,
|
|
@owner.login,
|
|
@project&.identifier,
|
|
gitea_pull_request_params.compact)
|
|
end
|
|
|
|
def gitea_pull_request_params
|
|
assignee_login = User.find_by_id(params[:assigned_to_id])&.login
|
|
merge_original_pull_params.except(:milestone).merge(
|
|
# assignees: ["#{params[:assigned_login].to_s}"],
|
|
assignees: ["#{assignee_login.to_s}"])
|
|
end
|
|
|
|
def merge_original_pull_params
|
|
base_pull_params[:head] = CGI.escape(base_pull_params[:head])
|
|
base_pull_params[:base] = CGI.escape(base_pull_params[:base])
|
|
if pull_request.is_original && @params[:merge_user_login]
|
|
base_pull_params.merge(head: "#{@params[:merge_user_login]}:#{base_pull_params[:head]}")
|
|
else
|
|
base_pull_params
|
|
end
|
|
end
|
|
|
|
def base_pull_params
|
|
{
|
|
title: @params[:title], #标题
|
|
body: @params[:body], #内容
|
|
head: @params[:head], #源分支
|
|
base: @params[:base], #目标分支
|
|
milestone: 0, #里程碑,未与本地的里程碑关联
|
|
}
|
|
end
|
|
|
|
def validate!
|
|
raise "title参数不能为空" if @params[:title].blank?
|
|
raise "title不能超过255个字符" if @params[:title].length > 255
|
|
raise "head参数不能为空" if @params[:head].blank?
|
|
raise "base参数不能为空" if @params[:base].blank?
|
|
raise "fork_project_id参数错误" if is_original && !@project.forked_projects.pluck(:id).include?(@params[:fork_project_id].to_i)
|
|
raise "分支内容相同,无需创建合并请求" if @params[:head] === @params[:base] && !is_original
|
|
raise "合并请求已存在" if @project&.pull_requests.where(head: @params[:head], base: @params[:base], status: 0, is_original: is_original, fork_project_id: @params[:fork_project_id]).present?
|
|
raise @pull_issue.errors.full_messages.join(", ") unless pull_issue.valid?
|
|
raise @pull_request.errors.full_messages.join(", ") unless pull_request.valid?
|
|
end
|
|
|
|
def compare_head_base!
|
|
head = pull_request.is_original && @params[:merge_user_login] ? "#{@params[:merge_user_login]}/#{@project.identifier}:#{@params[:head]}" : @params[:head]
|
|
compare_result = Gitea::Repository::Commits::CompareService.call(@owner.login, @project.identifier, CGI.escape(@params[:base]), CGI.escape(head), @current_user.gitea_token)
|
|
raise '分支内容相同,无需创建合并请求' if compare_result["Commits"].blank? && compare_result["Diff"].blank?
|
|
end
|
|
|
|
def is_original
|
|
@params[:is_original] || false
|
|
end
|
|
end
|