221 lines
7.1 KiB
Ruby
221 lines
7.1 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module QuickActions
|
|
class InterpretService < BaseService
|
|
include Gitlab::Utils::StrongMemoize
|
|
include Gitlab::QuickActions::Dsl
|
|
include Gitlab::QuickActions::IssueActions
|
|
include Gitlab::QuickActions::IssuableActions
|
|
include Gitlab::QuickActions::IssueAndMergeRequestActions
|
|
include Gitlab::QuickActions::MergeRequestActions
|
|
include Gitlab::QuickActions::CommitActions
|
|
include Gitlab::QuickActions::CommonActions
|
|
include Gitlab::QuickActions::RelateActions
|
|
include Gitlab::QuickActions::WorkItemActions
|
|
|
|
attr_reader :quick_action_target
|
|
|
|
# Counts how many commands have been executed.
|
|
# Used to display relevant feedback on UI when a note
|
|
# with only commands has been processed.
|
|
attr_accessor :commands_executed_count
|
|
|
|
# Takes an quick_action_target and returns an array of all the available commands
|
|
# represented with .to_h
|
|
def available_commands(quick_action_target)
|
|
@quick_action_target = quick_action_target
|
|
|
|
self.class.command_definitions.map do |definition|
|
|
next unless definition.available?(self)
|
|
|
|
definition.to_h(self)
|
|
end.compact
|
|
end
|
|
|
|
# Takes a text and interprets the commands that are extracted from it.
|
|
# Returns the content without commands, a hash of changes to be applied to a record
|
|
# and a string containing the execution_message to show to the user.
|
|
def execute(content, quick_action_target, only: nil)
|
|
return [content, {}, ''] unless current_user.can?(:use_quick_actions)
|
|
|
|
@quick_action_target = quick_action_target
|
|
@updates = {}
|
|
@execution_message = {}
|
|
|
|
content, commands = extractor.extract_commands(content, only: only)
|
|
extract_updates(commands)
|
|
|
|
[content, @updates, execution_messages_for(commands), command_names(commands)]
|
|
end
|
|
|
|
# Takes a text and interprets the commands that are extracted from it.
|
|
# Returns the content without commands, and array of changes explained.
|
|
# `keep_actions: true` will keep the quick actions in the content.
|
|
def explain(content, quick_action_target, keep_actions: false)
|
|
return [content, []] unless current_user.can?(:use_quick_actions)
|
|
|
|
@quick_action_target = quick_action_target
|
|
|
|
content, commands = extractor(keep_actions).extract_commands(content)
|
|
commands = explain_commands(commands)
|
|
[content, commands]
|
|
end
|
|
|
|
private
|
|
|
|
def failed_parse(message)
|
|
raise Gitlab::QuickActions::CommandDefinition::ParseError, message
|
|
end
|
|
|
|
def extractor(keep_actions = false)
|
|
Gitlab::QuickActions::Extractor.new(self.class.command_definitions, keep_actions: keep_actions)
|
|
end
|
|
|
|
# Find users for commands like /assign
|
|
#
|
|
# eg. /assign me and @jane and jack
|
|
def extract_users(params)
|
|
Gitlab::QuickActions::UsersExtractor
|
|
.new(current_user, project: project, group: group, target: quick_action_target, text: params)
|
|
.execute
|
|
|
|
rescue Gitlab::QuickActions::UsersExtractor::Error => err
|
|
extract_users_failed(err)
|
|
end
|
|
|
|
def extract_users_failed(err)
|
|
case err
|
|
when Gitlab::QuickActions::UsersExtractor::MissingError
|
|
failed_parse(format(_("Failed to find users for %{missing}"), missing: err.message))
|
|
when Gitlab::QuickActions::UsersExtractor::TooManyRefsError
|
|
failed_parse(format(_('Too many references. Quick actions are limited to at most %{max_count} user references'),
|
|
max_count: err.limit))
|
|
when Gitlab::QuickActions::UsersExtractor::TooManyFoundError
|
|
failed_parse(format(_("Too many users found. Quick actions are limited to at most %{max_count} users"),
|
|
max_count: err.limit))
|
|
else
|
|
Gitlab::ErrorTracking.track_and_raise_for_dev_exception(err)
|
|
failed_parse(_('Something went wrong'))
|
|
end
|
|
end
|
|
|
|
def find_milestones(project, params = {})
|
|
group_ids = project.group.self_and_ancestors.select(:id) if project.group
|
|
|
|
MilestonesFinder.new(params.merge(project_ids: [project.id], group_ids: group_ids)).execute
|
|
end
|
|
|
|
def parent
|
|
project || group
|
|
end
|
|
|
|
def group
|
|
strong_memoize(:group) do
|
|
quick_action_target.group if quick_action_target.respond_to?(:group)
|
|
end
|
|
end
|
|
|
|
def find_labels(labels_params = nil)
|
|
extract_references(labels_params, :label) | find_labels_by_name_no_tilde(labels_params)
|
|
end
|
|
|
|
def find_labels_by_name_no_tilde(labels_params)
|
|
return Label.none if label_with_tilde?(labels_params)
|
|
|
|
finder_params = { include_ancestor_groups: true }
|
|
finder_params[:project_id] = project.id if project
|
|
finder_params[:group_id] = group.id if group
|
|
finder_params[:name] = extract_label_names(labels_params) if labels_params
|
|
|
|
LabelsFinder.new(current_user, finder_params).execute
|
|
end
|
|
|
|
def label_with_tilde?(labels_params)
|
|
labels_params&.include?('~')
|
|
end
|
|
|
|
def extract_label_names(labels_params)
|
|
# '"A" "A B C" A B' => ["A", "A B C", "A", "B"]
|
|
labels_params.scan(/"([^"]+)"|([^ ]+)/).flatten.compact
|
|
end
|
|
|
|
def find_label_references(labels_param, format = :id)
|
|
labels_to_reference(find_labels(labels_param), format)
|
|
end
|
|
|
|
def labels_to_reference(labels, format = :id)
|
|
labels.map { |l| l.to_reference(format: format) }
|
|
end
|
|
|
|
def find_label_ids(labels_param)
|
|
find_labels(labels_param).map(&:id)
|
|
end
|
|
|
|
def explain_commands(commands)
|
|
map_commands(commands, :explain)
|
|
end
|
|
|
|
def execution_messages_for(commands)
|
|
map_commands(commands, :execute_message).join(' ')
|
|
end
|
|
|
|
def map_commands(commands, method)
|
|
commands.map do |name_or_alias, arg|
|
|
definition = self.class.definition_by_name(name_or_alias)
|
|
next unless definition
|
|
|
|
case method
|
|
when :explain
|
|
definition.explain(self, arg)
|
|
when :execute_message
|
|
@execution_message[definition.name.to_sym] || definition.execute_message(self, arg)
|
|
end
|
|
end.compact
|
|
end
|
|
|
|
def command_names(commands)
|
|
commands.flatten.map do |name|
|
|
definition = self.class.definition_by_name(name)
|
|
next unless definition
|
|
|
|
name
|
|
end.compact
|
|
end
|
|
|
|
def extract_updates(commands)
|
|
commands.each do |name, arg|
|
|
definition = self.class.definition_by_name(name)
|
|
next unless definition
|
|
|
|
definition.execute(self, arg)
|
|
usage_ping_tracking(definition.name, arg)
|
|
end
|
|
end
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def extract_references(arg, type)
|
|
return [] unless arg
|
|
|
|
ext = Gitlab::ReferenceExtractor.new(project, current_user)
|
|
|
|
ext.analyze(arg, author: current_user, group: group)
|
|
|
|
ext.references(type)
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
def usage_ping_tracking(quick_action_name, arg)
|
|
Gitlab::UsageDataCounters::QuickActionActivityUniqueCounter.track_unique_action(
|
|
quick_action_name.to_s,
|
|
args: arg&.strip,
|
|
user: current_user
|
|
)
|
|
end
|
|
|
|
def can?(ability, object)
|
|
Ability.allowed?(current_user, ability, object)
|
|
end
|
|
end
|
|
end
|
|
|
|
QuickActions::InterpretService.prepend_mod_with('QuickActions::InterpretService')
|