|
- 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] = Addressable::URI.escape(base_pull_params[:head])
- base_pull_params[:base] = Addressable::URI.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 "merge_user_login参数错误" if is_original && @project.fork_users.joins(:owner).where(users: {login: @params[:merge_user_login]}).blank?
- 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, Addressable::URI.escape(@params[:base]), Addressable::URI.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
|