2039c8280d
This whitelists all existing offenses for the various CodeReuse cops, of which most are triggered by the CodeReuse/ActiveRecord cop.
165 lines
4.7 KiB
Ruby
165 lines
4.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Issues
|
|
class MoveService < Issues::BaseService
|
|
MoveError = Class.new(StandardError)
|
|
|
|
def execute(issue, new_project)
|
|
@old_issue = issue
|
|
@old_project = @project
|
|
@new_project = new_project
|
|
|
|
unless issue.can_move?(current_user, new_project)
|
|
raise MoveError, 'Cannot move issue due to insufficient permissions!'
|
|
end
|
|
|
|
if @project == new_project
|
|
raise MoveError, 'Cannot move issue to project it originates from!'
|
|
end
|
|
|
|
# Using transaction because of a high resources footprint
|
|
# on rewriting notes (unfolding references)
|
|
#
|
|
ActiveRecord::Base.transaction do
|
|
@new_issue = create_new_issue
|
|
|
|
update_new_issue
|
|
update_old_issue
|
|
end
|
|
|
|
notify_participants
|
|
|
|
@new_issue
|
|
end
|
|
|
|
private
|
|
|
|
def update_new_issue
|
|
rewrite_notes
|
|
copy_resource_label_events
|
|
rewrite_issue_award_emoji
|
|
add_note_moved_from
|
|
end
|
|
|
|
def update_old_issue
|
|
add_note_moved_to
|
|
close_issue
|
|
mark_as_moved
|
|
end
|
|
|
|
def create_new_issue
|
|
new_params = { id: nil, iid: nil, label_ids: cloneable_label_ids,
|
|
milestone_id: cloneable_milestone_id,
|
|
project: @new_project, author: @old_issue.author,
|
|
description: rewrite_content(@old_issue.description),
|
|
assignee_ids: @old_issue.assignee_ids }
|
|
|
|
new_params = @old_issue.serializable_hash.symbolize_keys.merge(new_params)
|
|
CreateService.new(@new_project, @current_user, new_params).execute
|
|
end
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def cloneable_label_ids
|
|
params = {
|
|
project_id: @new_project.id,
|
|
title: @old_issue.labels.pluck(:title),
|
|
include_ancestor_groups: true
|
|
}
|
|
|
|
LabelsFinder.new(current_user, params).execute.pluck(:id)
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
def cloneable_milestone_id
|
|
title = @old_issue.milestone&.title
|
|
return unless title
|
|
|
|
if @new_project.group && can?(current_user, :read_group, @new_project.group)
|
|
group_id = @new_project.group.id
|
|
end
|
|
|
|
params =
|
|
{ title: title, project_ids: @new_project.id, group_ids: group_id }
|
|
|
|
milestones = MilestonesFinder.new(params).execute
|
|
milestones.first&.id
|
|
end
|
|
|
|
def rewrite_notes
|
|
@old_issue.notes_with_associations.find_each do |note|
|
|
new_note = note.dup
|
|
new_params = { project: @new_project, noteable: @new_issue,
|
|
note: rewrite_content(new_note.note),
|
|
created_at: note.created_at,
|
|
updated_at: note.updated_at }
|
|
|
|
new_note.update(new_params)
|
|
|
|
rewrite_award_emoji(note, new_note)
|
|
end
|
|
end
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def copy_resource_label_events
|
|
@old_issue.resource_label_events.find_in_batches do |batch|
|
|
events = batch.map do |event|
|
|
event.attributes
|
|
.except('id', 'reference', 'reference_html')
|
|
.merge('issue_id' => @new_issue.id, 'action' => ResourceLabelEvent.actions[event.action])
|
|
end
|
|
|
|
Gitlab::Database.bulk_insert(ResourceLabelEvent.table_name, events)
|
|
end
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
def rewrite_issue_award_emoji
|
|
rewrite_award_emoji(@old_issue, @new_issue)
|
|
end
|
|
|
|
def rewrite_award_emoji(old_awardable, new_awardable)
|
|
old_awardable.award_emoji.each do |award|
|
|
new_award = award.dup
|
|
new_award.awardable = new_awardable
|
|
new_award.save
|
|
end
|
|
end
|
|
|
|
def rewrite_content(content)
|
|
return unless content
|
|
|
|
rewriters = [Gitlab::Gfm::ReferenceRewriter,
|
|
Gitlab::Gfm::UploadsRewriter]
|
|
|
|
rewriters.inject(content) do |text, klass|
|
|
rewriter = klass.new(text, @old_project, @current_user)
|
|
rewriter.rewrite(@new_project)
|
|
end
|
|
end
|
|
|
|
def close_issue
|
|
close_service = CloseService.new(@old_project, @current_user)
|
|
close_service.execute(@old_issue, notifications: false, system_note: false)
|
|
end
|
|
|
|
def add_note_moved_from
|
|
SystemNoteService.noteable_moved(@new_issue, @new_project,
|
|
@old_issue, @current_user,
|
|
direction: :from)
|
|
end
|
|
|
|
def add_note_moved_to
|
|
SystemNoteService.noteable_moved(@old_issue, @old_project,
|
|
@new_issue, @current_user,
|
|
direction: :to)
|
|
end
|
|
|
|
def mark_as_moved
|
|
@old_issue.update(moved_to: @new_issue)
|
|
end
|
|
|
|
def notify_participants
|
|
notification_service.async.issue_moved(@old_issue, @new_issue, @current_user)
|
|
end
|
|
end
|
|
end
|