|
- class Api::Pm::Issues::CreateService < ApplicationService
- include ActiveModel::Model
- include Api::V1::Issues::Concerns::Checkable
- include Api::V1::Issues::Concerns::Loadable
-
- attr_reader :project, :current_user
- attr_reader :status_id, :priority_id, :milestone_id, :branch_name, :start_date, :due_date, :subject, :description, :blockchain_token_num, :root_subject
- attr_reader :issue_tag_ids, :assigner_ids, :attachment_ids, :receivers_login, :enterprise_identifier, :pm_issue_type
- attr_accessor :created_issue
-
- validates :subject, presence: true
- validates :status_id, :priority_id, presence: true
- validates :project, :current_user, presence: true
- validates :blockchain_token_num, numericality: {greater_than: 0}, allow_blank: true
-
- def initialize(project, params, current_user = nil)
- @project = project
- @current_user = current_user
- @status_id = params[:status_id]
- @priority_id = params[:priority_id]
- @milestone_id = params[:milestone_id]
- @branch_name = params[:branch_name]
- @start_date = params[:start_date]
- @due_date = params[:due_date]
- @subject = params[:subject]
- @description = params[:description]
- @blockchain_token_num = params[:blockchain_token_num]
- @issue_tag_ids = params[:issue_tag_ids]
- @assigner_ids = params[:assigner_ids]
- @attachment_ids = params[:attachment_ids]
- @receivers_login = params[:receivers_login]
- @pm_project_id = params[:pm_project_id]
- @pm_sprint_id = params[:pm_sprint_id]
- @pm_issue_type = params[:pm_issue_type]
- @root_id = params[:root_id]
- @time_scale = params[:time_scale]
- @linkable_id = params[:link_able_id]
- @root_subject = params[:root_subject]
- @enterprise_identifier = params[:enterprise_identifier]
- end
-
- def call
- raise Error, errors.full_messages.join(', ') unless valid?
- ActiveRecord::Base.transaction do
- check_issue_status(status_id)
- check_issue_priority(priority_id)
- check_milestone(milestone_id) if milestone_id.present?
- check_issue_tags(issue_tag_ids) unless issue_tag_ids.blank?
- check_assigners(assigner_ids) unless assigner_ids.blank?
- check_attachments(attachment_ids) unless attachment_ids.blank?
- check_atme_receivers(receivers_login) unless receivers_login.blank?
- check_blockchain_token_num(current_user.id, project.id, blockchain_token_num) if blockchain_token_num.present?
- load_assigners(assigner_ids) unless assigner_ids.blank?
- load_attachments(attachment_ids) unless attachment_ids.blank?
- load_issue_tags(issue_tag_ids) unless issue_tag_ids.blank?
- load_atme_receivers(receivers_login) unless receivers_login.blank?
- try_lock("Api::Pm::Issues::CreateService:#{project.id}") # 开始写数据,加锁
- @created_issue = Issue.new(issue_attributes)
- @created_issue.pm_issue_type = @pm_issue_type
- if @root_subject.present? && [4, 5].include?(@pm_issue_type.to_i)
- @root_issue = Issue.find_by(subject: @root_subject, pm_issue_type: [4,5], pm_project_id: @pm_project_id,enterprise_identifier: @enterprise_identifier)
- unless @root_issue.present?
- @root_issue = Issue.create(subject: @root_subject, pm_issue_type: @pm_issue_type.to_i, pm_project_id: @pm_project_id, enterprise_identifier: @enterprise_identifier, status_id: 1, priority_id: 1, tracker_id: Tracker.first.id, project_id: @project.id, author_id: current_user.id)
- end
- @created_issue.root_id = @root_issue.id
- else
- @created_issue.root_id = @root_id
- end
- build_author_participants
- build_assigner_participants unless assigner_ids.blank?
- build_atme_participants if @atme_receivers.present?
- build_issue_journal_details
- build_issue_project_trends
- @created_issue.assigners = @assigners unless assigner_ids.blank?
- @created_issue.attachments = @attachments unless attachment_ids.blank?
- @created_issue.issue_tags = @issue_tags unless issue_tag_ids.blank?
- @created_issue.pm_project_id = @pm_project_id
- @created_issue.enterprise_identifier = @enterprise_identifier
- @created_issue.pm_sprint_id = @pm_sprint_id
- @created_issue.time_scale = @time_scale
- @created_issue.issue_tags_value = @issue_tags.order('id asc').pluck(:id).join(',') unless issue_tag_ids.blank?
- @created_issue.changer_id = @current_user.id
- @created_issue.save!
- if @created_issue.parent_issue.present?
- parent_issue = @created_issue.parent_issue
- if @created_issue.root_id.present? && parent_issue.present?
- journal = parent_issue.journals.create!({user_id: current_user.id})
- journal.journal_details.create!({property: @created_issue.pm_issue_type_string, prop_key: 'leaf_issue', value: @created_issue.id.to_s})
- end
- end
- if @linkable_id.present?
- PmLink.create(be_linkable_type: 'Issue', be_linkable_id: @created_issue.id, linkable_type: 'Issue', linkable_id: @linkable_id)
- another_issue = Issue.find_by_id(@linkable_id)
- if another_issue.present?
- journal = another_issue.journals.create!({user_id: @current_user.id})
- journal.journal_details.create!({property: 'link_issue', prop_key: "1", value: @created_issue.id.to_s})
- end
- end
- if Site.has_blockchain? && @project.use_blockchain
- if @created_issue.blockchain_token_num.present? && @created_issue.blockchain_token_num > 0
- Blockchain::CreateIssue.call({user_id: current_user.id, project_id: @created_issue.project_id, token_num: @created_issue.blockchain_token_num})
- end
-
- push_activity_2_blockchain('issue_create', @created_issue)
- end
-
- project.del_project_issue_cache_delete_count # 把缓存里存储项目删除issue的个数清除掉
- unless @project.id.zero?
- # 新增时向grimoirelab推送事件
- IssueWebhookJob.set(wait: 5.seconds).perform_later(@created_issue.id)
-
- # @信息发送
- AtmeService.call(current_user, @atme_receivers, @created_issue) unless receivers_login.blank?
-
- # 发消息
- if Site.has_notice_menu?
- SendTemplateMessageJob.perform_later('IssueAssigned', current_user.id, @created_issue&.id, assigner_ids) unless assigner_ids.blank?
- SendTemplateMessageJob.perform_later('ProjectIssue', current_user.id, @created_issue&.id)
- end
-
- # 触发webhook
- TouchWebhookJob.set(wait: 5.seconds).perform_later('IssueCreate', @created_issue&.id, current_user.id)
- TouchWebhookJob.set(wait: 5.seconds).perform_later('IssueLabel', @created_issue&.id, current_user.id, {issue_tag_ids: [[], issue_tag_ids]}) unless issue_tag_ids.blank?
- TouchWebhookJob.set(wait: 5.seconds).perform_later('IssueAssign', @created_issue&.id, current_user.id, {assigner_ids: [[], assigner_ids]}) unless assigner_ids.blank?
- end
- unlock("Api::Pm::Issues::CreateService:#{project.id}") # 结束写数据,解锁
- end
-
- return @created_issue
- end
-
- private
-
- def issue_attributes
- issue_attributes = {
- subject: subject,
- project_id: project.id,
- author_id: current_user.id,
- tracker_id: Tracker.first.id,
- status_id: status_id,
- priority_id: priority_id,
- project_issues_index: (project.get_last_project_issues_index + 1),
- issue_type: '1',
- issue_classify: 'issue'
- }
-
- issue_attributes.merge!({description: description}) if description.present?
- issue_attributes.merge!({fixed_version_id: milestone_id}) if milestone_id.present?
- issue_attributes.merge!({start_date: start_date}) if start_date.present?
- issue_attributes.merge!({due_date: due_date}) if due_date.present?
- issue_attributes.merge!({branch_name: branch_name}) if branch_name.present?
- issue_attributes.merge!({blockchain_token_num: blockchain_token_num}) if blockchain_token_num.present?
-
- issue_attributes
- end
-
- def build_author_participants
- @created_issue.issue_participants.new({participant_type: 'authored', participant_id: current_user.id})
- end
-
- def build_assigner_participants
- assigner_ids.each do |aid|
- @created_issue.issue_participants.new({participant_type: 'assigned', participant_id: aid})
- end
- end
-
- def build_atme_participants
- @atme_receivers.each do |receiver|
- @created_issue.issue_participants.new({participant_type: 'atme', participant_id: receiver.id})
- end
- end
-
- def build_issue_project_trends
- return if @project.id == 0
- @created_issue.project_trends.new({user_id: current_user.id, project_id: @project.id, action_type: 'create'})
- @created_issue.project_trends.new({user_id: current_user.id, project_id: @project.id, action_type: ProjectTrend::CLOSE}) if status_id.to_i == 5
- end
-
- def build_issue_journal_details
- journal = @created_issue.journals.new({user_id: current_user.id})
- journal.journal_details.new({property: @created_issue.pm_issue_type_string, prop_key: 1, old_value: '', value: ''})
- end
- end
|