347 lines
13 KiB
Ruby
347 lines
13 KiB
Ruby
# frozen_string_literal: true
|
||
|
||
module Gitlab
|
||
module QuickActions
|
||
module IssuableActions
|
||
extend ActiveSupport::Concern
|
||
include Gitlab::QuickActions::Dsl
|
||
|
||
SHRUG = '¯\\_(ツ)_/¯'
|
||
TABLEFLIP = '(╯°□°)╯︵ ┻━┻'
|
||
|
||
included do
|
||
# Issue, MergeRequest, Epic: quick actions definitions
|
||
desc do
|
||
_('Close this %{quick_action_target}') % { quick_action_target: target_issuable_name }
|
||
end
|
||
explanation do
|
||
_('Closes this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
execution_message do
|
||
_('Closed this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
quick_action_target.open? &&
|
||
current_user.can?(:"update_#{quick_action_target.to_ability_name}", quick_action_target)
|
||
end
|
||
command :close do
|
||
@updates[:state_event] = 'close'
|
||
end
|
||
|
||
desc do
|
||
_('Reopen this %{quick_action_target}') %
|
||
{ quick_action_target: target_issuable_name }
|
||
end
|
||
explanation do
|
||
_('Reopens this %{quick_action_target}.') %
|
||
{ quick_action_target: target_issuable_name }
|
||
end
|
||
execution_message do
|
||
_('Reopened this %{quick_action_target}.') %
|
||
{ quick_action_target: target_issuable_name }
|
||
end
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
quick_action_target.closed? &&
|
||
current_user.can?(:"update_#{quick_action_target.to_ability_name}", quick_action_target)
|
||
end
|
||
command :reopen do
|
||
@updates[:state_event] = 'reopen'
|
||
end
|
||
|
||
desc { _('Change title') }
|
||
explanation do |title_param|
|
||
_('Changes the title to "%{title_param}".') % { title_param: title_param }
|
||
end
|
||
execution_message do |title_param|
|
||
_('Changed the title to "%{title_param}".') % { title_param: title_param }
|
||
end
|
||
params '<New title>'
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
current_user.can?(:"update_#{quick_action_target.to_ability_name}", quick_action_target)
|
||
end
|
||
command :title do |title_param|
|
||
@updates[:title] = title_param
|
||
end
|
||
|
||
desc { _('Add label(s)') }
|
||
explanation do |labels_param|
|
||
labels = find_label_references(labels_param)
|
||
|
||
if labels.any?
|
||
_("Adds %{labels} %{label_text}.") %
|
||
{ labels: labels.join(' '), label_text: 'label'.pluralize(labels.count) }
|
||
end
|
||
end
|
||
params '~label1 ~"label 2"'
|
||
types ::Issuable
|
||
condition do
|
||
current_user.can?(:"set_#{quick_action_target.to_ability_name}_metadata", quick_action_target) &&
|
||
find_labels.any?
|
||
end
|
||
command :label, :labels do |labels_param|
|
||
run_label_command(labels: find_labels(labels_param), command: :label, updates_key: :add_label_ids)
|
||
end
|
||
|
||
desc { _('Remove all or specific label(s)') }
|
||
explanation do |labels_param = nil|
|
||
label_references = labels_param.present? ? find_label_references(labels_param) : []
|
||
if label_references.any?
|
||
_("Removes %{label_references} %{label_text}.") %
|
||
{ label_references: label_references.join(' '), label_text: 'label'.pluralize(label_references.count) }
|
||
else
|
||
_('Removes all labels.')
|
||
end
|
||
end
|
||
params '~label1 ~"label 2"'
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
quick_action_target.labels.any? &&
|
||
current_user.can?(:"set_#{quick_action_target.to_ability_name}_metadata", quick_action_target)
|
||
end
|
||
command :unlabel, :remove_label do |labels_param = nil|
|
||
if labels_param.present?
|
||
labels = find_labels(labels_param)
|
||
label_ids = labels.map(&:id)
|
||
label_references = labels_to_reference(labels, :name)
|
||
|
||
if label_ids.any?
|
||
@updates[:remove_label_ids] ||= []
|
||
@updates[:remove_label_ids] += label_ids
|
||
|
||
@updates[:remove_label_ids].uniq!
|
||
end
|
||
else
|
||
@updates[:label_ids] = []
|
||
label_references = []
|
||
end
|
||
|
||
@execution_message[:unlabel] = remove_label_message(label_references)
|
||
end
|
||
|
||
desc { _('Replace all label(s)') }
|
||
explanation do |labels_param|
|
||
labels = find_label_references(labels_param)
|
||
"Replaces all labels with #{labels.join(' ')} #{'label'.pluralize(labels.count)}." if labels.any?
|
||
end
|
||
params '~label1 ~"label 2"'
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
quick_action_target.labels.any? &&
|
||
current_user.can?(:"set_#{quick_action_target.to_ability_name}_metadata", quick_action_target)
|
||
end
|
||
command :relabel do |labels_param|
|
||
run_label_command(labels: find_labels(labels_param), command: :relabel, updates_key: :label_ids)
|
||
end
|
||
|
||
desc { _('Add a to do') }
|
||
explanation { _('Adds a to do.') }
|
||
execution_message { _('Added a to do.') }
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
!TodoService.new.todo_exist?(quick_action_target, current_user)
|
||
end
|
||
command :todo do
|
||
@updates[:todo_event] = 'add'
|
||
end
|
||
|
||
desc { _('Mark to do as done') }
|
||
explanation { _('Marks to do as done.') }
|
||
execution_message { _('Marked to do as done.') }
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
TodoService.new.todo_exist?(quick_action_target, current_user)
|
||
end
|
||
command :done do
|
||
@updates[:todo_event] = 'done'
|
||
end
|
||
|
||
desc { _('Subscribe') }
|
||
explanation do
|
||
_('Subscribes to this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
execution_message do
|
||
_('Subscribed to this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
!quick_action_target.subscribed?(current_user, project)
|
||
end
|
||
command :subscribe do
|
||
@updates[:subscription_event] = 'subscribe'
|
||
end
|
||
|
||
desc { _('Unsubscribe') }
|
||
explanation do
|
||
_('Unsubscribes from this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
execution_message do
|
||
_('Unsubscribed from this %{quick_action_target}.') % { quick_action_target: target_issuable_name }
|
||
end
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted? &&
|
||
quick_action_target.subscribed?(current_user, project)
|
||
end
|
||
command :unsubscribe do
|
||
@updates[:subscription_event] = 'unsubscribe'
|
||
end
|
||
|
||
desc { _('Toggle emoji award') }
|
||
explanation do |name|
|
||
_("Toggles :%{name}: emoji award.") % { name: name } if name
|
||
end
|
||
execution_message do |name|
|
||
_("Toggled :%{name}: emoji award.") % { name: name } if name
|
||
end
|
||
params ':emoji:'
|
||
types ::Issuable
|
||
condition do
|
||
quick_action_target.persisted?
|
||
end
|
||
parse_params do |emoji_param|
|
||
match = emoji_param.match(Banzai::Filter::EmojiFilter.emoji_pattern)
|
||
match[1] if match
|
||
end
|
||
command :award do |name|
|
||
if name && quick_action_target.user_can_award?(current_user)
|
||
@updates[:emoji_award] = name
|
||
end
|
||
end
|
||
|
||
desc { _("Append the comment with %{shrug}") % { shrug: SHRUG } }
|
||
params '<Comment>'
|
||
types ::Issuable
|
||
substitution :shrug do |comment|
|
||
"#{comment} #{SHRUG}"
|
||
end
|
||
|
||
desc { _("Append the comment with %{tableflip}") % { tableflip: TABLEFLIP } }
|
||
params '<Comment>'
|
||
types ::Issuable
|
||
substitution :tableflip do |comment|
|
||
"#{comment} #{TABLEFLIP}"
|
||
end
|
||
|
||
desc { _('Set severity') }
|
||
explanation { _('Sets the severity') }
|
||
params '1 / S1 / Critical'
|
||
types Issue
|
||
condition do
|
||
!quick_action_target.persisted? || quick_action_target.supports_severity?
|
||
end
|
||
parse_params do |severity|
|
||
find_severity(severity)
|
||
end
|
||
command :severity do |severity|
|
||
next unless quick_action_target.supports_severity?
|
||
|
||
if severity
|
||
if quick_action_target.persisted?
|
||
::Issues::UpdateService.new(container: quick_action_target.project, current_user: current_user, params: { severity: severity }).execute(quick_action_target)
|
||
else
|
||
quick_action_target.build_issuable_severity(severity: severity)
|
||
end
|
||
|
||
@execution_message[:severity] = _("Severity updated to %{severity}.") % { severity: severity.capitalize }
|
||
else
|
||
@execution_message[:severity] = _('No severity matches the provided parameter')
|
||
end
|
||
end
|
||
|
||
desc { _("Make %{type} confidential") % { type: target_issuable_name } }
|
||
explanation { _("Makes this %{type} confidential.") % { type: target_issuable_name } }
|
||
types ::Issuable
|
||
condition { quick_action_target.supports_confidentiality? && can_make_confidential? }
|
||
command :confidential do
|
||
@updates[:confidential] = true
|
||
|
||
@execution_message[:confidential] = confidential_execution_message
|
||
end
|
||
|
||
private
|
||
|
||
def find_severity(severity_param)
|
||
return unless severity_param
|
||
|
||
severity_param = severity_param.downcase
|
||
severities = IssuableSeverity::SEVERITY_QUICK_ACTION_PARAMS.values.map { |vals| vals.map(&:downcase) }
|
||
|
||
matched_severity = severities.find do |severity_values|
|
||
severity_values.include?(severity_param)
|
||
end
|
||
|
||
return unless matched_severity
|
||
|
||
matched_severity[0]
|
||
end
|
||
|
||
def run_label_command(labels:, command:, updates_key:)
|
||
return if labels.empty?
|
||
|
||
@updates[updates_key] ||= []
|
||
@updates[updates_key] += labels.map(&:id)
|
||
@updates[updates_key].uniq!
|
||
|
||
label_references = labels_to_reference(labels, :name)
|
||
@execution_message[command] = case command
|
||
when :relabel
|
||
_('Replaced all labels with %{label_references} %{label_text}.') %
|
||
{
|
||
label_references: label_references.join(' '),
|
||
label_text: 'label'.pluralize(label_references.count)
|
||
}
|
||
when :label
|
||
_('Added %{label_references} %{label_text}.') %
|
||
{
|
||
label_references: label_references.join(' '),
|
||
label_text: 'label'.pluralize(labels.count)
|
||
}
|
||
end
|
||
end
|
||
|
||
def remove_label_message(label_references)
|
||
if label_references.any?
|
||
_("Removed %{label_references} %{label_text}.") %
|
||
{ label_references: label_references.join(' '), label_text: 'label'.pluralize(label_references.count) }
|
||
else
|
||
_('Removed all labels.')
|
||
end
|
||
end
|
||
|
||
def target_issuable_name
|
||
quick_action_target.to_ability_name.humanize(capitalize: false)
|
||
end
|
||
|
||
def can_make_confidential?
|
||
confidentiality_not_supported = quick_action_target.respond_to?(:issue_type_supports?) &&
|
||
!quick_action_target.issue_type_supports?(:confidentiality)
|
||
|
||
return false if confidentiality_not_supported
|
||
|
||
!quick_action_target.confidential? && current_user.can?(:set_confidentiality, quick_action_target)
|
||
end
|
||
|
||
def confidential_execution_message
|
||
confidential_error_message.presence || _("Made this %{type} confidential.") % { type: target_issuable_name }
|
||
end
|
||
|
||
def confidential_error_message
|
||
return unless quick_action_target.respond_to?(:confidentiality_errors)
|
||
|
||
quick_action_target.confidentiality_errors.join("\n")
|
||
end
|
||
end
|
||
end
|
||
end
|
||
end
|