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! save_pull_attached_issues! 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 save_pull_attached_issues! if attached_issue_ids.size > 1 raise "最多只能关联一个疑修。" else attached_issue_ids.each do |issue| PullAttachedIssue.create!(issue_id: issue, pull_request_id: pull_request&.id) end end end def attached_issue_ids Array(@params[:attached_issue_ids]) 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]}/#{@params[:merge_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