debian-mirror-gitlab/app/models/concerns/issuable.rb

670 lines
21 KiB
Ruby
Raw Normal View History

2018-11-20 20:47:30 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
# == Issuable concern
#
# Contains common functionality shared between Issues and MergeRequests
#
2019-12-21 20:55:43 +05:30
# Used by Issue, MergeRequest, Epic
2014-09-02 18:07:02 +05:30
#
module Issuable
extend ActiveSupport::Concern
2018-03-17 18:26:18 +05:30
include Gitlab::SQL::Pattern
2018-11-18 11:00:15 +05:30
include Redactable
2016-11-03 12:29:30 +05:30
include CacheMarkdownField
2015-09-11 14:41:01 +05:30
include Participable
2015-10-24 18:46:33 +05:30
include Mentionable
2020-03-13 15:44:24 +05:30
include Milestoneable
2016-06-02 11:05:42 +05:30
include Subscribable
2015-12-23 02:04:40 +05:30
include StripAttribute
include Awardable
2017-08-17 22:00:37 +05:30
include Taskable
include Importable
include Editable
2017-09-10 17:25:29 +05:30
include AfterCommitQueue
2018-03-17 18:26:18 +05:30
include Sortable
include CreatedAtFilterable
2018-03-27 19:54:05 +05:30
include UpdatedAtFilterable
2019-07-07 11:18:12 +05:30
include ClosedAtFilterable
2019-12-21 20:55:43 +05:30
include VersionedDescription
2021-11-11 11:23:49 +05:30
include SortableTitle
2023-07-07 10:43:13 +05:30
include Exportable
2019-12-21 20:55:43 +05:30
TITLE_LENGTH_MAX = 255
TITLE_HTML_LENGTH_MAX = 800
DESCRIPTION_LENGTH_MAX = 1.megabyte
DESCRIPTION_HTML_LENGTH_MAX = 5.megabytes
2021-09-30 23:02:18 +05:30
SEARCHABLE_FIELDS = %w(title description).freeze
2022-11-25 23:54:43 +05:30
MAX_NUMBER_OF_ASSIGNEES_OR_REVIEWERS = 200
2019-12-21 20:55:43 +05:30
STATE_ID_MAP = {
opened: 1,
closed: 2,
merged: 3,
locked: 4
}.with_indifferent_access.freeze
2017-08-17 22:00:37 +05:30
2014-09-02 18:07:02 +05:30
included do
2016-11-03 12:29:30 +05:30
cache_markdown_field :title, pipeline: :single_line
2022-01-26 12:08:38 +05:30
cache_markdown_field :description, issuable_reference_expansion_enabled: true
2016-11-03 12:29:30 +05:30
2018-11-18 11:00:15 +05:30
redact_field :description
2019-07-07 11:18:12 +05:30
belongs_to :author, class_name: 'User'
belongs_to :updated_by, class_name: 'User'
2017-08-17 22:00:37 +05:30
belongs_to :last_edited_by, class_name: 'User'
2017-09-10 17:25:29 +05:30
has_many :notes, as: :noteable, inverse_of: :noteable, dependent: :destroy do # rubocop:disable Cop/ActiveRecordDependent
def authors_loaded?
2016-08-24 12:49:21 +05:30
# We check first if we're loaded to not load unnecessarily.
loaded? && to_a.all? { |note| note.association(:author).loaded? }
end
2016-08-24 12:49:21 +05:30
def award_emojis_loaded?
# We check first if we're loaded to not load unnecessarily.
loaded? && to_a.all? { |note| note.association(:award_emoji).loaded? }
end
2022-01-26 12:08:38 +05:30
def projects_loaded?
# We check first if we're loaded to not load unnecessarily.
loaded? && to_a.all? { |note| note.association(:project).loaded? }
end
def system_note_metadata_loaded?
# We check first if we're loaded to not load unnecessarily.
loaded? && to_a.all? { |note| note.association(:system_note_metadata).loaded? }
end
end
2016-09-29 09:46:39 +05:30
2020-10-24 23:57:45 +05:30
has_many :note_authors, -> { distinct }, through: :notes, source: :author
2022-05-07 20:08:51 +05:30
has_many :user_note_authors, -> { distinct.where("notes.system = false") }, through: :notes, source: :author
2020-10-24 23:57:45 +05:30
2021-06-08 01:23:25 +05:30
has_many :label_links, as: :target, inverse_of: :target
2014-09-02 18:07:02 +05:30
has_many :labels, through: :label_links
2021-04-29 21:17:54 +05:30
has_many :todos, as: :target
2014-09-02 18:07:02 +05:30
2020-10-24 23:57:45 +05:30
has_one :metrics, inverse_of: model_name.singular.to_sym, autosave: true
2016-09-29 09:46:39 +05:30
2017-08-17 22:00:37 +05:30
delegate :name,
2023-06-20 00:43:36 +05:30
:email,
:public_email,
to: :author,
allow_nil: true,
prefix: true
2017-08-17 22:00:37 +05:30
2014-09-02 18:07:02 +05:30
validates :author, presence: true
2019-12-21 20:55:43 +05:30
validates :title, presence: true, length: { maximum: TITLE_LENGTH_MAX }
2023-02-26 17:17:37 +05:30
# we validate the description against DESCRIPTION_LENGTH_MAX only for Issuables being created and on updates if
# the description changes to avoid breaking the existing Issuables which may have their descriptions longer
validates :description, bytesize: { maximum: -> { DESCRIPTION_LENGTH_MAX } }, if: :validate_description_length?
2022-11-25 23:54:43 +05:30
validate :validate_assignee_size_length, unless: :importing?
2014-09-02 18:07:02 +05:30
2019-12-21 20:55:43 +05:30
before_validation :truncate_description_on_import!
2014-09-02 18:07:02 +05:30
scope :authored, ->(user) { where(author_id: user) }
2021-04-17 20:07:23 +05:30
scope :not_authored, ->(user) { where.not(author_id: user) }
2015-11-26 14:37:03 +05:30
scope :recent, -> { reorder(id: :desc) }
2014-09-02 18:07:02 +05:30
scope :of_projects, ->(ids) { where(project_id: ids) }
2017-09-10 17:25:29 +05:30
scope :opened, -> { with_state(:opened) }
2014-09-02 18:07:02 +05:30
scope :closed, -> { with_state(:closed) }
2019-07-31 22:56:46 +05:30
# rubocop:disable GitlabSecurity/SqlInjection
2022-07-23 23:45:48 +05:30
# The `assignee_association_name` method is not an user input.
2019-07-31 22:56:46 +05:30
scope :assigned, -> do
2022-07-23 23:45:48 +05:30
where("EXISTS (SELECT TRUE FROM #{assignee_association_name}_assignees WHERE #{assignee_association_name}_id = #{assignee_association_name}s.id)")
2019-07-31 22:56:46 +05:30
end
scope :unassigned, -> do
2022-07-23 23:45:48 +05:30
where("NOT EXISTS (SELECT TRUE FROM #{assignee_association_name}_assignees WHERE #{assignee_association_name}_id = #{assignee_association_name}s.id)")
2019-07-31 22:56:46 +05:30
end
2021-09-04 01:27:46 +05:30
scope :assigned_to, ->(users) do
2022-07-23 23:45:48 +05:30
assignees_class = self.reflect_on_association("#{assignee_association_name}_assignees").klass
2019-07-31 22:56:46 +05:30
2022-07-23 23:45:48 +05:30
condition = assignees_class.where(user_id: users).where(Arel.sql("#{assignee_association_name}_id = #{assignee_association_name}s.id"))
2021-09-04 01:27:46 +05:30
where(condition.arel.exists)
end
2020-05-24 23:13:21 +05:30
scope :not_assigned_to, ->(users) do
2022-07-23 23:45:48 +05:30
assignees_class = self.reflect_on_association("#{assignee_association_name}_assignees").klass
2021-09-04 01:27:46 +05:30
2022-07-23 23:45:48 +05:30
condition = assignees_class.where(user_id: users).where(Arel.sql("#{assignee_association_name}_id = #{assignee_association_name}s.id"))
2021-09-04 01:27:46 +05:30
where(condition.arel.exists.not)
2020-05-24 23:13:21 +05:30
end
2021-09-04 01:27:46 +05:30
# rubocop:enable GitlabSecurity/SqlInjection
2020-05-24 23:13:21 +05:30
scope :without_label, -> { joins("LEFT OUTER JOIN label_links ON label_links.target_type = '#{name}' AND label_links.target_id = #{table_name}.id").where(label_links: { id: nil }) }
2020-04-22 19:07:51 +05:30
scope :with_label_ids, ->(label_ids) { joins(:label_links).where(label_links: { label_id: label_ids }) }
2015-12-23 02:04:40 +05:30
scope :join_project, -> { joins(:project) }
2017-08-17 22:00:37 +05:30
scope :inc_notes_with_associations, -> { includes(notes: [:project, :author, :award_emoji]) }
2015-12-23 02:04:40 +05:30
scope :references_project, -> { references(:project) }
2016-06-02 11:05:42 +05:30
scope :non_archived, -> { join_project.where(projects: { archived: false }) }
2021-04-29 21:17:54 +05:30
scope :includes_for_bulk_update, -> do
associations = %i[author assignees epic group labels metrics project source_project target_project].select do |association|
reflect_on_association(association)
end
includes(*associations)
end
2015-12-23 02:04:40 +05:30
attr_mentionable :title, pipeline: :single_line
attr_mentionable :description
participant :author
participant :notes_with_associations
2019-07-31 22:56:46 +05:30
participant :assignees
2021-10-27 15:23:28 +05:30
strip_attributes! :title
2016-06-02 11:05:42 +05:30
2020-04-22 19:07:51 +05:30
class << self
def labels_hash
issue_labels = Hash.new { |h, k| h[k] = [] }
relation = unscoped.where(id: self.select(:id)).eager_load(:labels)
relation.pluck(:id, 'labels.title').each do |issue_id, label|
issue_labels[issue_id] << label if label.present?
end
issue_labels
end
def locking_enabled?
false
end
2022-11-25 23:54:43 +05:30
def max_number_of_assignees_or_reviewers_message
# Assignees will be included in https://gitlab.com/gitlab-org/gitlab/-/issues/368936
format(_("total must be less than or equal to %{size}"), size: MAX_NUMBER_OF_ASSIGNEES_OR_REVIEWERS)
end
2020-03-13 15:44:24 +05:30
end
2016-09-29 09:46:39 +05:30
# We want to use optimistic lock for cases when only title or description are involved
# http://api.rubyonrails.org/classes/ActiveRecord/Locking/Optimistic.html
def locking_enabled?
2019-07-31 22:56:46 +05:30
will_save_change_to_title? || will_save_change_to_description?
2016-09-29 09:46:39 +05:30
end
2017-09-10 17:25:29 +05:30
def allows_multiple_assignees?
false
end
def has_multiple_assignees?
assignees.count > 1
end
2019-03-13 22:55:13 +05:30
2020-11-24 15:15:51 +05:30
def allows_reviewers?
2020-06-23 00:09:42 +05:30
false
end
2020-11-24 15:15:51 +05:30
def supports_time_tracking?
2021-01-03 14:25:43 +05:30
is_a?(TimeTrackable)
2020-11-24 15:15:51 +05:30
end
def supports_severity?
incident?
end
2022-01-26 12:08:38 +05:30
def supports_escalation?
incident?
end
2020-11-24 15:15:51 +05:30
def incident?
is_a?(Issue) && super
end
def supports_issue_type?
is_a?(Issue)
end
2021-03-08 18:12:59 +05:30
def supports_assignee?
false
end
2023-01-13 00:05:48 +05:30
def supports_confidentiality?
false
end
2020-11-24 15:15:51 +05:30
def severity
2021-02-22 17:27:13 +05:30
return IssuableSeverity::DEFAULT unless supports_severity?
2020-11-24 15:15:51 +05:30
issuable_severity&.severity || IssuableSeverity::DEFAULT
end
2023-07-07 10:43:13 +05:30
def exportable_restricted_associations
super + [:notes]
end
2019-03-13 22:55:13 +05:30
private
2023-02-26 17:17:37 +05:30
def validate_description_length?
return false unless description_changed?
previous_description = changes['description'].first
# previous_description will be nil for new records
return true if previous_description.blank?
previous_description.bytesize <= DESCRIPTION_LENGTH_MAX
2019-12-21 20:55:43 +05:30
end
def truncate_description_on_import!
self.description = description&.slice(0, Issuable::DESCRIPTION_LENGTH_MAX) if importing?
end
2022-11-25 23:54:43 +05:30
def validate_assignee_size_length
return true unless assignees.size > MAX_NUMBER_OF_ASSIGNEES_OR_REVIEWERS
errors.add :assignees,
-> (_object, _data) { self.class.max_number_of_assignees_or_reviewers_message }
end
2014-09-02 18:07:02 +05:30
end
2018-11-20 20:47:30 +05:30
class_methods do
2021-03-08 18:12:59 +05:30
def participant_includes
2022-11-25 23:54:43 +05:30
[:author, :award_emoji, { notes: [:author, :award_emoji, :system_note_metadata] }]
2021-03-08 18:12:59 +05:30
end
2016-06-02 11:05:42 +05:30
# Searches for records with a matching title.
#
2020-06-23 00:09:42 +05:30
# This method uses ILIKE on PostgreSQL.
2016-06-02 11:05:42 +05:30
#
# query - The search query as a String
#
# Returns an ActiveRecord::Relation.
2014-09-02 18:07:02 +05:30
def search(query)
2018-03-17 18:26:18 +05:30
fuzzy_search(query, [:title])
2014-09-02 18:07:02 +05:30
end
2019-12-21 20:55:43 +05:30
def available_states
@available_states ||= STATE_ID_MAP.slice(*available_state_names)
end
# Available state names used to persist state_id column using state machine
2019-07-07 11:18:12 +05:30
#
# Override this on subclasses if different states are needed
#
2019-12-21 20:55:43 +05:30
# Check MergeRequest.available_states_names for example
def available_state_names
[:opened, :closed]
2019-07-07 11:18:12 +05:30
end
2016-06-02 11:05:42 +05:30
# Searches for records with a matching title or description.
#
2020-06-23 00:09:42 +05:30
# This method uses ILIKE on PostgreSQL.
2016-06-02 11:05:42 +05:30
#
# query - The search query as a String
2019-03-02 22:35:43 +05:30
# matched_columns - Modify the scope of the query. 'title', 'description' or joining them with a comma.
2016-06-02 11:05:42 +05:30
#
# Returns an ActiveRecord::Relation.
2021-09-30 23:02:18 +05:30
def full_search(query, matched_columns: nil, use_minimum_char_limit: true)
if matched_columns
matched_columns = matched_columns.to_s.split(',')
matched_columns &= SEARCHABLE_FIELDS
matched_columns.map!(&:to_sym)
end
2019-03-02 22:35:43 +05:30
2021-09-30 23:02:18 +05:30
search_columns = matched_columns.presence || [:title, :description]
2019-03-02 22:35:43 +05:30
2021-09-30 23:02:18 +05:30
fuzzy_search(query, search_columns, use_minimum_char_limit: use_minimum_char_limit)
2015-04-26 12:48:37 +05:30
end
2019-07-07 11:18:12 +05:30
def simple_sorts
super.except('name_asc', 'name_desc')
end
2018-05-09 12:01:36 +05:30
def sort_by_attribute(method, excluded_labels: [])
2018-03-17 18:26:18 +05:30
sorted =
case method.to_s
2019-12-04 20:38:33 +05:30
when 'downvotes_desc' then order_downvotes_desc
when 'label_priority', 'label_priority_asc' then order_labels_priority(excluded_labels: excluded_labels)
when 'label_priority_desc' then order_labels_priority('DESC', excluded_labels: excluded_labels)
when 'milestone', 'milestone_due_asc' then order_milestone_due_asc
when 'milestone_due_desc' then order_milestone_due_desc
when 'popularity_asc' then order_upvotes_asc
when 'popularity', 'popularity_desc', 'upvotes_desc' then order_upvotes_desc
when 'priority', 'priority_asc' then order_due_date_and_labels_priority(excluded_labels: excluded_labels)
when 'priority_desc' then order_due_date_and_labels_priority('DESC', excluded_labels: excluded_labels)
2021-11-11 11:23:49 +05:30
when 'title_asc' then order_title_asc.with_order_id_desc
when 'title_desc' then order_title_desc.with_order_id_desc
2018-03-17 18:26:18 +05:30
else order_by(method)
end
2016-08-24 12:49:21 +05:30
# Break ties with the ID column for pagination
2018-11-18 11:00:15 +05:30
sorted.with_order_id_desc
2014-09-02 18:07:02 +05:30
end
2016-04-02 18:10:28 +05:30
2019-02-15 15:39:39 +05:30
def order_due_date_and_labels_priority(direction = 'ASC', excluded_labels: [])
2017-08-17 22:00:37 +05:30
# The order_ methods also modify the query in other ways:
#
# - For milestones, we add a JOIN.
# - For label priority, we change the SELECT, and add a GROUP BY.#
#
# After doing those, we need to reorder to the order we want. The existing
# ORDER BYs won't work because:
#
# 1. We need milestone due date first.
# 2. We can't ORDER BY a column that isn't in the GROUP BY and doesn't
# have an aggregate function applied, so we do a useless MIN() instead.
#
2022-06-21 17:19:12 +05:30
milestones_due_date = Milestone.arel_table[:due_date].minimum
milestones_due_date_with_direction = direction == 'ASC' ? milestones_due_date.asc : milestones_due_date.desc
highest_priority_arel = Arel.sql('highest_priority')
highest_priority_arel_with_direction = direction == 'ASC' ? highest_priority_arel.asc : highest_priority_arel.desc
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
order_milestone_due_asc
.order_labels_priority(excluded_labels: excluded_labels, extra_select_columns: [milestones_due_date])
2023-06-20 00:43:36 +05:30
.reorder(milestones_due_date_with_direction.nulls_last, highest_priority_arel_with_direction.nulls_last)
2017-08-17 22:00:37 +05:30
end
2020-03-13 15:44:24 +05:30
def order_labels_priority(direction = 'ASC', excluded_labels: [], extra_select_columns: [], with_cte: false)
2021-02-22 17:27:13 +05:30
highest_priority = highest_label_priority(
2016-11-03 12:29:30 +05:30
target_type: name,
target_column: "#{table_name}.id",
project_column: "#{table_name}.#{project_foreign_key}",
excluded_labels: excluded_labels
2021-02-22 17:27:13 +05:30
).to_sql
2016-09-13 17:45:13 +05:30
2020-04-08 14:13:33 +05:30
# When using CTE make sure to select the same columns that are on the group_by clause.
# This prevents errors when ignored columns are present in the database.
issuable_columns = with_cte ? issue_grouping_columns(use_cte: with_cte) : "#{table_name}.*"
2021-09-30 23:02:18 +05:30
group_columns = issue_grouping_columns(use_cte: with_cte) + ["highest_priorities.label_priority"]
2017-08-17 22:00:37 +05:30
2021-09-30 23:02:18 +05:30
extra_select_columns.unshift("highest_priorities.label_priority as highest_priority")
2020-04-08 14:13:33 +05:30
2022-06-21 17:19:12 +05:30
highest_priority_arel = Arel.sql('highest_priority')
highest_priority_arel_with_direction = direction == 'ASC' ? highest_priority_arel.asc : highest_priority_arel.desc
2020-04-08 14:13:33 +05:30
select(issuable_columns)
.select(extra_select_columns)
2023-03-04 22:38:38 +05:30
.from(table_name.to_s)
2021-09-30 23:02:18 +05:30
.joins("JOIN LATERAL(#{highest_priority}) as highest_priorities ON TRUE")
.group(group_columns)
2022-06-21 17:19:12 +05:30
.reorder(highest_priority_arel_with_direction.nulls_last)
2016-04-02 18:10:28 +05:30
end
2020-05-24 23:13:21 +05:30
def with_label(title, sort = nil)
if title.is_a?(Array) && title.size > 1
joins(:labels).where(labels: { title: title }).group(*grouping_columns(sort)).having("COUNT(DISTINCT labels.title) = #{title.size}")
else
joins(:labels).where(labels: { title: title })
end
2016-04-02 18:10:28 +05:30
end
2020-06-23 00:09:42 +05:30
def any_label(sort = nil)
if sort
joins(:label_links).group(*grouping_columns(sort))
else
joins(:label_links).distinct
end
end
# Includes table keys in group by clause when sorting
2022-04-04 11:22:00 +05:30
# preventing errors in Postgres
#
# Returns an array of Arel columns
#
def grouping_columns(sort)
2021-03-08 18:12:59 +05:30
sort = sort.to_s
grouping_columns = [arel_table[:id]]
2018-03-17 18:26:18 +05:30
if %w(milestone_due_desc milestone_due_asc milestone).include?(sort)
milestone_table = Milestone.arel_table
grouping_columns << milestone_table[:id]
grouping_columns << milestone_table[:due_date]
2022-11-25 23:54:43 +05:30
elsif %w(merged_at_desc merged_at_asc merged_at).include?(sort)
grouping_columns << MergeRequest::Metrics.arel_table[:id]
2021-03-08 18:12:59 +05:30
grouping_columns << MergeRequest::Metrics.arel_table[:merged_at]
2022-11-25 23:54:43 +05:30
elsif %w(closed_at_desc closed_at_asc closed_at).include?(sort)
grouping_columns << MergeRequest::Metrics.arel_table[:id]
grouping_columns << MergeRequest::Metrics.arel_table[:latest_closed_at]
end
2016-04-02 18:10:28 +05:30
grouping_columns
2016-04-02 18:10:28 +05:30
end
2016-11-24 13:41:30 +05:30
2020-03-13 15:44:24 +05:30
# Includes all table keys in group by clause when sorting
2022-04-04 11:22:00 +05:30
# preventing errors in Postgres when using CTE search optimization
#
# Returns an array of Arel columns
2020-03-13 15:44:24 +05:30
#
def issue_grouping_columns(use_cte: false)
if use_cte
2020-04-08 14:13:33 +05:30
attribute_names.map { |attr| arel_table[attr.to_sym] }
2020-03-13 15:44:24 +05:30
else
2021-09-30 23:02:18 +05:30
[arel_table[:id]]
2020-03-13 15:44:24 +05:30
end
end
2016-11-24 13:41:30 +05:30
def to_ability_name
model_name.singular
end
2018-03-27 19:54:05 +05:30
def parent_class
::Project
end
2022-07-23 23:45:48 +05:30
def assignee_association_name
to_ability_name
end
2014-09-02 18:07:02 +05:30
end
2019-12-21 20:55:43 +05:30
def state
self.class.available_states.key(state_id)
end
def state=(value)
self.state_id = self.class.available_states[value]
end
2019-12-04 20:38:33 +05:30
def resource_parent
project
end
2019-07-31 22:56:46 +05:30
def assignee_or_author?(user)
2022-11-25 23:54:43 +05:30
author_id == user.id || assignee?(user)
end
def assignee?(user)
# Necessary so we can preload the association and avoid N + 1 queries
if assignees.loaded?
assignees.to_a.include?(user)
else
assignees.exists?(user.id)
end
2019-07-31 22:56:46 +05:30
end
2015-10-24 18:46:33 +05:30
def open?
2017-09-10 17:25:29 +05:30
opened?
2015-10-24 18:46:33 +05:30
end
2018-11-18 11:00:15 +05:30
def overdue?
return false unless respond_to?(:due_date)
due_date.try(:past?) || false
end
2016-06-02 11:05:42 +05:30
def user_notes_count
if notes.loaded?
# Use the in-memory association to select and count to avoid hitting the db
notes.to_a.count { |note| !note.system? }
else
# do the count query
notes.user.count
end
2015-04-26 12:48:37 +05:30
end
2017-08-17 22:00:37 +05:30
def subscribed_without_subscriptions?(user, project)
2021-04-29 21:17:54 +05:30
participant?(user)
end
2020-06-23 00:09:42 +05:30
def can_assign_epic?(user)
false
end
2022-05-07 20:08:51 +05:30
def hook_association_changes(old_associations)
changes = {}
2018-03-17 18:26:18 +05:30
2022-05-07 20:08:51 +05:30
old_labels = old_associations.fetch(:labels, labels)
old_assignees = old_associations.fetch(:assignees, assignees)
old_severity = old_associations.fetch(:severity, severity)
2018-03-17 18:26:18 +05:30
2022-05-07 20:08:51 +05:30
if old_labels != labels
changes[:labels] = [old_labels.map(&:hook_attrs), labels.map(&:hook_attrs)]
end
2016-04-02 18:10:28 +05:30
2022-05-07 20:08:51 +05:30
if old_assignees != assignees
changes[:assignees] = [old_assignees.map(&:hook_attrs), assignees.map(&:hook_attrs)]
end
if supports_severity? && old_severity != severity
changes[:severity] = [old_severity, severity]
end
if supports_escalation? && escalation_status
current_escalation_status = escalation_status.status_name
old_escalation_status = old_associations.fetch(:escalation_status, current_escalation_status)
2018-03-17 18:26:18 +05:30
2022-05-07 20:08:51 +05:30
if old_escalation_status != current_escalation_status
changes[:escalation_status] = [old_escalation_status, current_escalation_status]
2021-09-30 23:02:18 +05:30
end
2022-05-07 20:08:51 +05:30
end
2021-09-30 23:02:18 +05:30
2022-05-07 20:08:51 +05:30
if self.respond_to?(:total_time_spent)
old_total_time_spent = old_associations.fetch(:total_time_spent, total_time_spent)
old_time_change = old_associations.fetch(:time_change, time_change)
2019-03-02 22:35:43 +05:30
2022-05-07 20:08:51 +05:30
if old_total_time_spent != total_time_spent
changes[:total_time_spent] = [old_total_time_spent, total_time_spent]
changes[:time_change] = [old_time_change, time_change]
2018-03-17 18:26:18 +05:30
end
end
2022-05-07 20:08:51 +05:30
changes
end
2022-08-27 11:52:29 +05:30
def hook_reviewer_changes(old_associations)
changes = {}
old_reviewers = old_associations.fetch(:reviewers, reviewers)
if old_reviewers != reviewers
changes[:reviewers] = [old_reviewers.map(&:hook_attrs), reviewers.map(&:hook_attrs)]
end
changes
end
2022-05-07 20:08:51 +05:30
def to_hook_data(user, old_associations: {})
changes = previous_changes
if old_associations.present?
changes.merge!(hook_association_changes(old_associations))
2022-08-27 11:52:29 +05:30
changes.merge!(hook_reviewer_changes(old_associations)) if allows_reviewers?
2022-05-07 20:08:51 +05:30
end
2022-07-16 23:28:13 +05:30
Gitlab::DataBuilder::Issuable.new(self).build(user: user, changes: changes)
2014-09-02 18:07:02 +05:30
end
def labels_array
labels.to_a
end
2014-09-02 18:07:02 +05:30
def label_names
labels.order('title ASC').pluck(:title)
end
2022-06-21 17:19:12 +05:30
def labels_hook_attrs
labels.map(&:hook_attrs)
end
2022-08-27 11:52:29 +05:30
def allows_scoped_labels?
false
end
2015-09-11 14:41:01 +05:30
# Convert this Issuable class name to a format usable by Ability definitions
#
# Examples:
#
# issuable.class # => MergeRequest
# issuable.to_ability_name # => "merge_request"
def to_ability_name
2016-11-24 13:41:30 +05:30
self.class.to_ability_name
2015-09-11 14:41:01 +05:30
end
# Returns a Hash of attributes to be used for Twitter card metadata
def card_attributes
{
2022-08-27 11:52:29 +05:30
'Author' => author.try(:name),
2019-07-31 22:56:46 +05:30
'Assignee' => assignee_list
}
end
2019-07-31 22:56:46 +05:30
def assignee_list
assignees.map(&:name).to_sentence
end
def assignee_username_list
assignees.map(&:username).to_sentence
end
2015-10-24 18:46:33 +05:30
def notes_with_associations
# If A has_many Bs, and B has_many Cs, and you do
# `A.includes(b: :c).each { |a| a.b.includes(:c) }`, sadly ActiveRecord
# will do the inclusion again. So, we check if all notes in the relation
# already have their authors loaded (possibly because the scope
# `inc_notes_with_associations` was used) and skip the inclusion if that's
# the case.
2016-08-24 12:49:21 +05:30
includes = []
includes << :author unless notes.authors_loaded?
includes << :award_emoji unless notes.award_emojis_loaded?
2022-01-26 12:08:38 +05:30
includes << :project unless notes.projects_loaded?
includes << :system_note_metadata unless notes.system_note_metadata_loaded?
2018-03-17 18:26:18 +05:30
2016-08-24 12:49:21 +05:30
if includes.any?
notes.includes(includes)
else
notes
end
2015-10-24 18:46:33 +05:30
end
2015-12-23 02:04:40 +05:30
def updated_tasks
2023-06-20 00:43:36 +05:30
Taskable.get_updated_tasks(
old_content: previous_changes['description'].first,
new_content: description
)
2015-12-23 02:04:40 +05:30
end
2016-06-02 11:05:42 +05:30
##
# Method that checks if issuable can be moved to another project.
#
# Should be overridden if issuable can be moved.
#
def can_move?(*)
false
end
2016-09-29 09:46:39 +05:30
2018-03-17 18:26:18 +05:30
##
# Override in issuable specialization
#
def first_contribution?
false
end
##
2018-12-13 13:39:08 +05:30
# Overridden in MergeRequest
2018-03-17 18:26:18 +05:30
#
2022-04-04 11:22:00 +05:30
def draftless_title_changed(old_title)
2018-03-17 18:26:18 +05:30
old_title != title
2016-09-29 09:46:39 +05:30
end
2022-11-25 23:54:43 +05:30
def read_ability_for(participable_source)
return super if participable_source == self
name = participable_source.try(:issuable_ability_name) || :read_issuable_participables
{ name: name, subject: self }
end
2023-05-27 22:25:52 +05:30
def supports_health_status?
false
end
2014-09-02 18:07:02 +05:30
end
2019-12-04 20:38:33 +05:30
2021-06-08 01:23:25 +05:30
Issuable.prepend_mod_with('Issuable')