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

569 lines
18 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
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
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
2017-08-17 22:00:37 +05:30
cache_markdown_field :description, issuable_state_filter_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
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
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,
:email,
:public_email,
to: :author,
allow_nil: true,
prefix: true
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 }
# we validate the description against DESCRIPTION_LENGTH_MAX only for Issuables being created
# to avoid breaking the existing Issuables which may have their descriptions longer
validates :description, length: { maximum: DESCRIPTION_LENGTH_MAX }, allow_blank: true, on: :create
validate :description_max_length_for_new_records_is_valid, on: :update
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
# The `to_ability_name` method is not an user input.
scope :assigned, -> do
where("EXISTS (SELECT TRUE FROM #{to_ability_name}_assignees WHERE #{to_ability_name}_id = #{to_ability_name}s.id)")
end
scope :unassigned, -> do
where("NOT EXISTS (SELECT TRUE FROM #{to_ability_name}_assignees WHERE #{to_ability_name}_id = #{to_ability_name}s.id)")
end
2021-09-04 01:27:46 +05:30
scope :assigned_to, ->(users) do
assignees_class = self.reflect_on_association("#{to_ability_name}_assignees").klass
2019-07-31 22:56:46 +05:30
2021-09-04 01:27:46 +05:30
condition = assignees_class.where(user_id: users).where(Arel.sql("#{to_ability_name}_id = #{to_ability_name}s.id"))
where(condition.arel.exists)
end
2020-05-24 23:13:21 +05:30
scope :not_assigned_to, ->(users) do
2021-09-04 01:27:46 +05:30
assignees_class = self.reflect_on_association("#{to_ability_name}_assignees").klass
condition = assignees_class.where(user_id: users).where(Arel.sql("#{to_ability_name}_id = #{to_ability_name}s.id"))
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
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
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
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
2019-03-13 22:55:13 +05:30
private
2019-12-21 20:55:43 +05:30
def description_max_length_for_new_records_is_valid
if new_record? && description.length > Issuable::DESCRIPTION_LENGTH_MAX
errors.add(:description, :too_long, count: Issuable::DESCRIPTION_LENGTH_MAX)
end
end
def truncate_description_on_import!
self.description = description&.slice(0, Issuable::DESCRIPTION_LENGTH_MAX) if importing?
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
[:assignees, :author, { notes: [:author, :award_emoji] }]
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.
#
milestones_due_date = 'MIN(milestones.due_date)'
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])
2019-02-15 15:39:39 +05:30
.reorder(Gitlab::Database.nulls_last_order(milestones_due_date, direction),
Gitlab::Database.nulls_last_order('highest_priority', direction))
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
select(issuable_columns)
.select(extra_select_columns)
2021-09-30 23:02:18 +05:30
.from("#{table_name}")
.joins("JOIN LATERAL(#{highest_priority}) as highest_priorities ON TRUE")
.group(group_columns)
2019-02-15 15:39:39 +05:30
.reorder(Gitlab::Database.nulls_last_order('highest_priority', direction))
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
# 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]
2021-03-08 18:12:59 +05:30
elsif %w(merged_at_desc merged_at_asc).include?(sort)
grouping_columns << MergeRequest::Metrics.arel_table[:merged_at]
2021-10-27 15:23:28 +05:30
elsif %w(closed_at_desc closed_at_asc).include?(sort)
grouping_columns << MergeRequest::Metrics.arel_table[: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
# preventing errors in postgres when using CTE search optimisation
#
# Returns an array of arel columns
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
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)
author_id == user.id || assignees.exists?(user.id)
end
2014-09-02 18:07:02 +05:30
def today?
Date.today == created_at.to_date
end
2020-11-24 15:15:51 +05:30
def created_hours_ago
(Time.now.utc.to_i - created_at.utc.to_i) / 3600
end
2014-09-02 18:07:02 +05:30
def new?
2020-11-24 15:15:51 +05:30
created_hours_ago < 24
2014-09-02 18:07:02 +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
2018-03-17 18:26:18 +05:30
def to_hook_data(user, old_associations: {})
changes = previous_changes
2019-03-02 22:35:43 +05:30
if old_associations
2020-07-28 23:09:34 +05:30
old_labels = old_associations.fetch(:labels, labels)
old_assignees = old_associations.fetch(:assignees, assignees)
2021-09-30 23:02:18 +05:30
old_severity = old_associations.fetch(:severity, severity)
2018-03-17 18:26:18 +05:30
2019-03-02 22:35:43 +05:30
if old_labels != labels
changes[:labels] = [old_labels.map(&:hook_attrs), labels.map(&:hook_attrs)]
2018-03-17 18:26:18 +05:30
end
2016-04-02 18:10:28 +05:30
2019-03-02 22:35:43 +05:30
if old_assignees != assignees
2019-07-31 22:56:46 +05:30
changes[:assignees] = [old_assignees.map(&:hook_attrs), assignees.map(&:hook_attrs)]
2019-03-02 22:35:43 +05:30
end
2018-03-17 18:26:18 +05:30
2021-09-30 23:02:18 +05:30
if supports_severity? && old_severity != severity
changes[:severity] = [old_severity, severity]
end
2019-03-02 22:35:43 +05:30
if self.respond_to?(:total_time_spent)
2020-07-28 23:09:34 +05:30
old_total_time_spent = old_associations.fetch(:total_time_spent, total_time_spent)
2021-09-04 01:27:46 +05:30
old_time_change = old_associations.fetch(:time_change, time_change)
2019-03-02 22:35:43 +05:30
if old_total_time_spent != total_time_spent
changes[:total_time_spent] = [old_total_time_spent, total_time_spent]
2021-09-04 01:27:46 +05:30
changes[:time_change] = [old_time_change, time_change]
2019-03-02 22:35:43 +05:30
end
2018-03-17 18:26:18 +05:30
end
end
Gitlab::HookData::IssuableBuilder.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
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
{
'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?
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
Taskable.get_updated_tasks(old_content: previous_changes['description'].first,
new_content: description)
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
def ensure_metrics
self.metrics || create_metrics
end
##
2018-12-13 13:39:08 +05:30
# Overridden in MergeRequest
2018-03-17 18:26:18 +05:30
#
def wipless_title_changed(old_title)
old_title != title
2016-09-29 09:46:39 +05:30
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')