issuable_base_service.rb 6.7 KB
Newer Older
1 2 3 4
class IssuableBaseService < BaseService
  private

  def create_assignee_note(issuable)
5
    SystemNoteService.change_assignee(
6 7 8 9
      issuable, issuable.project, current_user, issuable.assignee)
  end

  def create_milestone_note(issuable)
10
    SystemNoteService.change_milestone(
11 12
      issuable, issuable.project, current_user, issuable.milestone)
  end
N
Nikita Verkhovin 已提交
13

14 15 16 17
  def create_labels_note(issuable, old_labels)
    added_labels = issuable.labels - old_labels
    removed_labels = old_labels - issuable.labels

18
    SystemNoteService.change_label(
N
Nikita Verkhovin 已提交
19 20
      issuable, issuable.project, current_user, added_labels, removed_labels)
  end
21 22 23 24 25

  def create_title_change_note(issuable, old_title)
    SystemNoteService.change_title(
      issuable, issuable.project, current_user, old_title)
  end
26 27 28 29 30 31

  def create_branch_change_note(issuable, branch_type, old_branch, new_branch)
    SystemNoteService.change_branch(
      issuable, issuable.project, current_user, branch_type,
      old_branch, new_branch)
  end
32

33 34 35 36 37 38
  def create_task_status_note(issuable)
    issuable.updated_tasks.each do |task|
      SystemNoteService.change_task_status(issuable, issuable.project, current_user, task)
    end
  end

39
  def filter_params(issuable_ability_name = :issue)
40 41 42
    filter_assignee
    filter_milestone
    filter_labels
43

44 45 46
    ability = :"admin_#{issuable_ability_name}"

    unless can?(current_user, ability, project)
47
      params.delete(:milestone_id)
48
      params.delete(:labels)
49 50
      params.delete(:add_label_ids)
      params.delete(:remove_label_ids)
51 52 53 54
      params.delete(:label_ids)
      params.delete(:assignee_id)
    end
  end
55

56 57 58 59 60 61 62
  def filter_assignee
    if params[:assignee_id] == IssuableFinder::NONE
      params[:assignee_id] = ''
    end
  end

  def filter_milestone
63 64
    milestone_id = params[:milestone_id]
    return unless milestone_id
65

66 67
    if milestone_id == IssuableFinder::NONE ||
        project.milestones.find_by(id: milestone_id).nil?
68 69 70 71 72
      params[:milestone_id] = ''
    end
  end

  def filter_labels
73 74 75
    filter_labels_in_param(:add_label_ids)
    filter_labels_in_param(:remove_label_ids)
    filter_labels_in_param(:label_ids)
76
    find_or_create_label_ids
77 78
  end

79 80
  def filter_labels_in_param(key)
    return if params[key].to_a.empty?
81

82
    params[key] = project.labels.where(id: params[key]).pluck(:id)
83 84
  end

85 86 87 88 89 90 91 92 93 94 95
  def find_or_create_label_ids
    labels = params.delete(:labels)
    return unless labels

    params[:label_ids] = labels.split(",").map do |label_name|
      project.labels.create_with(color: Label::DEFAULT_COLOR)
                    .find_or_create_by(title: label_name.strip)
                    .id
    end
  end

96
  def process_label_ids(attributes, existing_label_ids: nil)
97 98 99
    label_ids = attributes.delete(:label_ids)
    add_label_ids = attributes.delete(:add_label_ids)
    remove_label_ids = attributes.delete(:remove_label_ids)
100

101
    new_label_ids = existing_label_ids || label_ids || []
102

103 104 105 106 107 108
    if add_label_ids.blank? && remove_label_ids.blank?
      new_label_ids = label_ids if label_ids
    else
      new_label_ids |= add_label_ids if add_label_ids
      new_label_ids -= remove_label_ids if remove_label_ids
    end
109 110 111 112

    new_label_ids
  end

113
  def merge_slash_commands_into_params!(issuable)
D
Douwe Maan 已提交
114 115
    description, command_params =
      SlashCommands::InterpretService.new(project, current_user).
116
      execute(params[:description], issuable)
117

D
Douwe Maan 已提交
118 119 120
    params[:description] = description

    params.merge!(command_params)
121 122
  end

123
  def create_issuable(issuable, attributes, label_ids:)
124
    issuable.with_transaction_returning_status do
125 126 127 128 129
      if issuable.save
        issuable.update_attributes(label_ids: label_ids)
      end
    end
  end
130

131
  def create(issuable)
132
    merge_slash_commands_into_params!(issuable)
133
    filter_params
134

135 136 137 138 139 140 141 142 143 144
    params.delete(:state_event)
    params[:author] ||= current_user
    label_ids = process_label_ids(params)

    issuable.assign_attributes(params)

    before_create(issuable)

    if params.present? && create_issuable(issuable, params, label_ids: label_ids)
      after_create(issuable)
145 146 147 148 149 150
      issuable.create_cross_references!(current_user)
      execute_hooks(issuable)
    end

    issuable
  end
151

152 153 154 155 156 157 158
  def before_create(issuable)
    # To be overridden by subclasses
  end

  def after_create(issuable)
    # To be overridden by subclasses
  end
159

160 161 162
  def update_issuable(issuable, attributes)
    issuable.with_transaction_returning_status do
      issuable.update(attributes.merge(updated_by: current_user))
163
    end
164 165
  end

166 167
  def update(issuable)
    change_state(issuable)
168
    change_subscription(issuable)
169
    change_todo(issuable)
170 171
    filter_params
    old_labels = issuable.labels.to_a
172
    old_mentioned_users = issuable.mentioned_users.to_a
173

174 175
    params[:label_ids] = process_label_ids(params, existing_label_ids: issuable.label_ids)

176
    if params.present? && update_issuable(issuable, params)
177
      issuable.reset_events_cache
178 179 180 181 182 183

      # We do not touch as it will affect a update on updated_at field
      ActiveRecord::Base.no_touching do
        handle_common_system_notes(issuable, old_labels: old_labels)
      end

184
      handle_changes(issuable, old_labels: old_labels, old_mentioned_users: old_mentioned_users)
185 186 187 188 189 190 191 192 193 194 195 196 197 198 199
      issuable.create_new_cross_references!(current_user)
      execute_hooks(issuable, 'update')
    end

    issuable
  end

  def change_state(issuable)
    case params.delete(:state_event)
    when 'reopen'
      reopen_service.new(project, current_user, {}).execute(issuable)
    when 'close'
      close_service.new(project, current_user, {}).execute(issuable)
    end
  end
200

201 202 203 204 205 206 207 208 209
  def change_subscription(issuable)
    case params.delete(:subscription_event)
    when 'subscribe'
      issuable.subscribe(current_user)
    when 'unsubscribe'
      issuable.unsubscribe(current_user)
    end
  end

210 211
  def change_todo(issuable)
    case params.delete(:todo_event)
212
    when 'add'
213 214 215
      todo_service.mark_todo(issuable, current_user)
    when 'done'
      todo = TodosFinder.new(current_user).execute.find_by(target: issuable)
D
Douwe Maan 已提交
216
      todo_service.mark_todos_as_done([todo], current_user) if todo
217 218 219
    end
  end

220
  def has_changes?(issuable, old_labels: [])
221 222 223 224 225 226
    valid_attrs = [:title, :description, :assignee_id, :milestone_id, :target_branch]

    attrs_changed = valid_attrs.any? do |attr|
      issuable.previous_changes.include?(attr.to_s)
    end

227
    labels_changed = issuable.labels != old_labels
228 229 230 231

    attrs_changed || labels_changed
  end

232
  def handle_common_system_notes(issuable, old_labels: [])
233 234 235 236 237 238 239
    if issuable.previous_changes.include?('title')
      create_title_change_note(issuable, issuable.previous_changes['title'].first)
    end

    if issuable.previous_changes.include?('description') && issuable.tasks?
      create_task_status_note(issuable)
    end
240

241
    create_labels_note(issuable, old_labels) if issuable.labels != old_labels
242
  end
243
end