2013-01-03 02:06:07 -05:00
|
|
|
# == Issuable concern
|
2012-12-30 09:19:31 -05:00
|
|
|
#
|
2012-10-09 15:09:46 -04:00
|
|
|
# Contains common functionality shared between Issues and MergeRequests
|
2012-12-30 09:19:31 -05:00
|
|
|
#
|
|
|
|
# Used by Issue, MergeRequest
|
|
|
|
#
|
2013-01-03 02:06:07 -05:00
|
|
|
module Issuable
|
2012-08-08 21:40:57 -04:00
|
|
|
extend ActiveSupport::Concern
|
2016-10-06 17:17:11 -04:00
|
|
|
include CacheMarkdownField
|
2015-04-21 09:23:20 -04:00
|
|
|
include Participable
|
2015-10-14 10:20:11 -04:00
|
|
|
include Mentionable
|
2016-02-12 09:58:39 -05:00
|
|
|
include Subscribable
|
2015-11-26 10:16:50 -05:00
|
|
|
include StripAttribute
|
2016-04-16 15:09:08 -04:00
|
|
|
include Awardable
|
2016-09-26 17:36:11 -04:00
|
|
|
include Taskable
|
2016-12-23 00:44:02 -05:00
|
|
|
include TimeTrackable
|
2012-08-08 21:40:57 -04:00
|
|
|
|
|
|
|
included do
|
2016-10-06 17:17:11 -04:00
|
|
|
cache_markdown_field :title, pipeline: :single_line
|
|
|
|
cache_markdown_field :description
|
|
|
|
|
2012-08-10 18:07:50 -04:00
|
|
|
belongs_to :author, class_name: "User"
|
|
|
|
belongs_to :assignee, class_name: "User"
|
2015-07-30 08:45:54 -04:00
|
|
|
belongs_to :updated_by, class_name: "User"
|
2012-10-30 03:22:24 -04:00
|
|
|
belongs_to :milestone
|
2016-07-28 10:51:44 -04:00
|
|
|
has_many :notes, as: :noteable, inverse_of: :noteable, dependent: :destroy do
|
2016-06-06 16:19:39 -04:00
|
|
|
def authors_loaded?
|
2016-06-21 01:26:43 -04:00
|
|
|
# We check first if we're loaded to not load unnecessarily.
|
2016-06-06 16:19:39 -04:00
|
|
|
loaded? && to_a.all? { |note| note.association(:author).loaded? }
|
|
|
|
end
|
2016-06-21 01:26:43 -04:00
|
|
|
|
|
|
|
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
|
2016-06-06 16:19:39 -04:00
|
|
|
end
|
2016-09-14 10:10:31 -04:00
|
|
|
|
2014-07-29 12:19:26 -04:00
|
|
|
has_many :label_links, as: :target, dependent: :destroy
|
|
|
|
has_many :labels, through: :label_links
|
2016-03-23 22:14:02 -04:00
|
|
|
has_many :todos, as: :target, dependent: :destroy
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2016-09-20 08:43:11 -04:00
|
|
|
has_one :metrics
|
|
|
|
|
2012-10-08 20:10:04 -04:00
|
|
|
validates :author, presence: true
|
2016-12-02 07:54:57 -05:00
|
|
|
validates :title, presence: true, length: { maximum: 255 }
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2013-08-10 13:25:53 -04:00
|
|
|
scope :authored, ->(user) { where(author_id: user) }
|
2013-06-17 04:57:37 -04:00
|
|
|
scope :assigned_to, ->(u) { where(assignee_id: u.id)}
|
2015-11-11 09:17:12 -05:00
|
|
|
scope :recent, -> { reorder(id: :desc) }
|
2013-06-17 04:57:37 -04:00
|
|
|
scope :assigned, -> { where("assignee_id IS NOT NULL") }
|
|
|
|
scope :unassigned, -> { where("assignee_id IS NULL") }
|
2013-08-08 10:29:31 -04:00
|
|
|
scope :of_projects, ->(ids) { where(project_id: ids) }
|
2016-02-22 19:08:00 -05:00
|
|
|
scope :of_milestones, ->(ids) { where(milestone_id: ids) }
|
2016-05-19 18:30:38 -04:00
|
|
|
scope :with_milestone, ->(title) { left_joins_milestones.where(milestones: { title: title }) }
|
2014-02-28 15:43:16 -05:00
|
|
|
scope :opened, -> { with_state(:opened, :reopened) }
|
2014-06-04 11:10:53 -04:00
|
|
|
scope :only_opened, -> { with_state(:opened) }
|
|
|
|
scope :only_reopened, -> { with_state(:reopened) }
|
2014-02-28 15:43:16 -05:00
|
|
|
scope :closed, -> { with_state(:closed) }
|
2013-08-19 15:10:56 -04:00
|
|
|
|
2016-04-27 11:35:30 -04:00
|
|
|
scope :left_joins_milestones, -> { joins("LEFT OUTER JOIN milestones ON #{table_name}.milestone_id = milestones.id") }
|
2016-05-05 15:23:51 -04:00
|
|
|
scope :order_milestone_due_desc, -> { left_joins_milestones.reorder('milestones.due_date IS NULL, milestones.id IS NULL, milestones.due_date DESC') }
|
|
|
|
scope :order_milestone_due_asc, -> { left_joins_milestones.reorder('milestones.due_date IS NULL, milestones.id IS NULL, milestones.due_date ASC') }
|
2013-08-19 15:10:56 -04:00
|
|
|
|
2016-04-27 11:35:30 -04:00
|
|
|
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 }) }
|
2015-11-11 06:50:36 -05:00
|
|
|
scope :join_project, -> { joins(:project) }
|
2016-06-21 01:26:43 -04:00
|
|
|
scope :inc_notes_with_associations, -> { includes(notes: [ :project, :author, :award_emoji ]) }
|
2015-11-11 06:50:36 -05:00
|
|
|
scope :references_project, -> { references(:project) }
|
2016-04-02 08:36:41 -04:00
|
|
|
scope :non_archived, -> { join_project.where(projects: { archived: false }) }
|
2015-11-11 06:50:36 -05:00
|
|
|
|
2012-08-08 21:40:57 -04:00
|
|
|
delegate :name,
|
|
|
|
:email,
|
2012-08-10 18:07:50 -04:00
|
|
|
to: :author,
|
|
|
|
prefix: true
|
2012-08-08 21:40:57 -04:00
|
|
|
|
|
|
|
delegate :name,
|
|
|
|
:email,
|
2012-08-10 18:07:50 -04:00
|
|
|
to: :assignee,
|
|
|
|
allow_nil: true,
|
|
|
|
prefix: true
|
2012-08-08 21:40:57 -04:00
|
|
|
|
2015-10-22 09:40:02 -04:00
|
|
|
attr_mentionable :title, pipeline: :single_line
|
2016-05-26 07:38:28 -04:00
|
|
|
attr_mentionable :description
|
|
|
|
|
|
|
|
participant :author
|
|
|
|
participant :assignee
|
|
|
|
participant :notes_with_associations
|
|
|
|
|
2015-11-26 10:16:50 -05:00
|
|
|
strip_attributes :title
|
2016-03-14 16:46:44 -04:00
|
|
|
|
|
|
|
acts_as_paranoid
|
2016-06-01 19:27:21 -04:00
|
|
|
|
|
|
|
after_save :update_assignee_cache_counts, if: :assignee_id_changed?
|
2016-09-20 08:43:11 -04:00
|
|
|
after_save :record_metrics
|
2016-06-01 19:27:21 -04:00
|
|
|
|
|
|
|
def update_assignee_cache_counts
|
2016-12-14 14:57:14 -05:00
|
|
|
# make sure we flush the cache for both the old *and* new assignees(if they exist)
|
2016-12-14 16:39:53 -05:00
|
|
|
previous_assignee = User.find_by_id(assignee_id_was) if assignee_id_was
|
|
|
|
previous_assignee.update_cache_counts if previous_assignee
|
|
|
|
assignee.update_cache_counts if assignee
|
2016-06-01 19:27:21 -04:00
|
|
|
end
|
2016-08-01 11:34:17 -04:00
|
|
|
|
|
|
|
# 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?
|
|
|
|
title_changed? || description_changed?
|
|
|
|
end
|
2012-08-08 21:40:57 -04:00
|
|
|
end
|
|
|
|
|
2012-08-09 13:45:12 -04:00
|
|
|
module ClassMethods
|
2016-03-01 10:59:36 -05:00
|
|
|
# Searches for records with a matching title.
|
|
|
|
#
|
|
|
|
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
|
|
|
|
#
|
|
|
|
# query - The search query as a String
|
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation.
|
2012-08-09 13:45:12 -04:00
|
|
|
def search(query)
|
2016-03-01 10:59:36 -05:00
|
|
|
where(arel_table[:title].matches("%#{query}%"))
|
2012-08-09 13:45:12 -04:00
|
|
|
end
|
2014-01-14 13:49:32 -05:00
|
|
|
|
2016-03-01 10:59:36 -05:00
|
|
|
# Searches for records with a matching title or description.
|
|
|
|
#
|
|
|
|
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
|
|
|
|
#
|
|
|
|
# query - The search query as a String
|
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation.
|
2014-08-27 05:47:30 -04:00
|
|
|
def full_search(query)
|
2016-03-01 10:59:36 -05:00
|
|
|
t = arel_table
|
|
|
|
pattern = "%#{query}%"
|
|
|
|
|
|
|
|
where(t[:title].matches(pattern).or(t[:description].matches(pattern)))
|
2014-08-27 05:47:30 -04:00
|
|
|
end
|
|
|
|
|
2016-05-13 11:26:18 -04:00
|
|
|
def sort(method, excluded_labels: [])
|
2016-06-23 09:15:46 -04:00
|
|
|
sorted = case method.to_s
|
|
|
|
when 'milestone_due_asc' then order_milestone_due_asc
|
|
|
|
when 'milestone_due_desc' then order_milestone_due_desc
|
|
|
|
when 'downvotes_desc' then order_downvotes_desc
|
|
|
|
when 'upvotes_desc' then order_upvotes_desc
|
|
|
|
when 'priority' then order_labels_priority(excluded_labels: excluded_labels)
|
|
|
|
else
|
|
|
|
order_by(method)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Break ties with the ID column for pagination
|
|
|
|
sorted.order(id: :desc)
|
2014-01-14 13:49:32 -05:00
|
|
|
end
|
2016-05-12 03:23:21 -04:00
|
|
|
|
2016-05-13 11:26:18 -04:00
|
|
|
def order_labels_priority(excluded_labels: [])
|
2016-10-17 16:03:06 -04:00
|
|
|
params = {
|
|
|
|
target_type: name,
|
|
|
|
target_column: "#{table_name}.id",
|
2016-10-19 09:49:07 -04:00
|
|
|
project_column: "#{table_name}.#{project_foreign_key}",
|
2016-10-17 16:03:06 -04:00
|
|
|
excluded_labels: excluded_labels
|
|
|
|
}
|
|
|
|
|
|
|
|
highest_priority = highest_label_priority(params).to_sql
|
2016-07-26 17:21:20 -04:00
|
|
|
|
|
|
|
select("#{table_name}.*, (#{highest_priority}) AS highest_priority").
|
2016-05-13 11:26:18 -04:00
|
|
|
group(arel_table[:id]).
|
|
|
|
reorder(Gitlab::Database.nulls_last_order('highest_priority', 'ASC'))
|
|
|
|
end
|
|
|
|
|
2016-05-27 15:53:20 -04:00
|
|
|
def with_label(title, sort = nil)
|
2016-05-12 03:23:21 -04:00
|
|
|
if title.is_a?(Array) && title.size > 1
|
2016-05-31 18:33:46 -04:00
|
|
|
joins(:labels).where(labels: { title: title }).group(*grouping_columns(sort)).having("COUNT(DISTINCT labels.title) = #{title.size}")
|
2016-05-12 03:23:21 -04:00
|
|
|
else
|
|
|
|
joins(:labels).where(labels: { title: title })
|
|
|
|
end
|
|
|
|
end
|
2016-05-27 15:53:20 -04:00
|
|
|
|
|
|
|
# Includes table keys in group by clause when sorting
|
|
|
|
# preventing errors in postgres
|
|
|
|
#
|
|
|
|
# Returns an array of arel columns
|
2016-05-31 18:33:46 -04:00
|
|
|
def grouping_columns(sort)
|
|
|
|
grouping_columns = [arel_table[:id]]
|
2016-05-27 15:53:20 -04:00
|
|
|
|
|
|
|
if ["milestone_due_desc", "milestone_due_asc"].include?(sort)
|
|
|
|
milestone_table = Milestone.arel_table
|
2016-05-31 18:33:46 -04:00
|
|
|
grouping_columns << milestone_table[:id]
|
|
|
|
grouping_columns << milestone_table[:due_date]
|
2016-05-27 15:53:20 -04:00
|
|
|
end
|
|
|
|
|
2016-05-31 18:33:46 -04:00
|
|
|
grouping_columns
|
2016-05-27 15:53:20 -04:00
|
|
|
end
|
2016-10-26 13:34:06 -04:00
|
|
|
|
|
|
|
def to_ability_name
|
|
|
|
model_name.singular
|
|
|
|
end
|
2012-08-08 21:40:57 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def today?
|
|
|
|
Date.today == created_at.to_date
|
|
|
|
end
|
|
|
|
|
|
|
|
def new?
|
|
|
|
today? && created_at == updated_at
|
|
|
|
end
|
2012-10-09 18:25:29 -04:00
|
|
|
|
|
|
|
def is_being_reassigned?
|
|
|
|
assignee_id_changed?
|
|
|
|
end
|
|
|
|
|
2015-10-13 03:41:46 -04:00
|
|
|
def open?
|
|
|
|
opened? || reopened?
|
|
|
|
end
|
|
|
|
|
2016-06-01 05:23:09 -04:00
|
|
|
def user_notes_count
|
2016-06-06 16:19:39 -04:00
|
|
|
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
|
2016-06-01 05:23:09 -04:00
|
|
|
end
|
|
|
|
|
2016-10-31 15:19:14 -04:00
|
|
|
def subscribed_without_subscriptions?(user, project)
|
2016-03-01 11:33:13 -05:00
|
|
|
participants(user).include?(user)
|
|
|
|
end
|
|
|
|
|
2014-10-05 09:03:15 -04:00
|
|
|
def to_hook_data(user)
|
2015-10-17 18:11:36 -04:00
|
|
|
hook_data = {
|
2013-12-03 04:31:56 -05:00
|
|
|
object_kind: self.class.name.underscore,
|
2014-10-05 09:03:15 -04:00
|
|
|
user: user.hook_attrs,
|
Add new data to project in push, issue, merge-request and note webhooks data
- Add `avatar_url`, `description`, `git_ssh_url`, `git_http_url`,
`path_with_namespace` and `default_branch` in `project` in push, issue,
merge-request and note webhooks data
- Deprecate the `ssh_url` in favor of `git_ssh_url` and `http_url` in
favor of `git_http_url` in `project` for push, issue, merge-request and
note webhooks data
- Deprecate the `repository` key in push, issue, merge-request and
note webhooks data, use `project` instead
2016-02-06 09:20:21 -05:00
|
|
|
project: project.hook_attrs,
|
|
|
|
object_attributes: hook_attrs,
|
|
|
|
# DEPRECATED
|
|
|
|
repository: project.hook_attrs.slice(:name, :url, :description, :homepage)
|
2013-12-03 04:31:56 -05:00
|
|
|
}
|
2015-10-17 18:11:36 -04:00
|
|
|
hook_data.merge!(assignee: assignee.hook_attrs) if assignee
|
|
|
|
|
|
|
|
hook_data
|
2013-12-03 04:31:56 -05:00
|
|
|
end
|
2014-07-30 10:17:29 -04:00
|
|
|
|
2016-06-02 07:17:54 -04:00
|
|
|
def labels_array
|
|
|
|
labels.to_a
|
|
|
|
end
|
|
|
|
|
2014-07-30 10:17:29 -04:00
|
|
|
def label_names
|
|
|
|
labels.order('title ASC').pluck(:title)
|
|
|
|
end
|
|
|
|
|
2015-07-24 15:31:15 -04:00
|
|
|
# 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-10-26 13:34:06 -04:00
|
|
|
self.class.to_ability_name
|
2015-07-24 15:31:15 -04:00
|
|
|
end
|
|
|
|
|
2016-11-18 21:19:04 -05:00
|
|
|
# Convert this Issuable class name to a format usable by notifications.
|
|
|
|
#
|
|
|
|
# Examples:
|
|
|
|
#
|
|
|
|
# issuable.class # => MergeRequest
|
|
|
|
# issuable.human_class_name # => "merge request"
|
|
|
|
|
|
|
|
def human_class_name
|
|
|
|
@human_class_name ||= self.class.name.titleize.downcase
|
|
|
|
end
|
|
|
|
|
2015-12-24 17:03:54 -05:00
|
|
|
# Returns a Hash of attributes to be used for Twitter card metadata
|
|
|
|
def card_attributes
|
|
|
|
{
|
|
|
|
'Author' => author.try(:name),
|
|
|
|
'Assignee' => assignee.try(:name)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2015-10-13 05:49:01 -04:00
|
|
|
def notes_with_associations
|
2016-06-06 16:19:39 -04:00
|
|
|
# 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-06-21 01:26:43 -04:00
|
|
|
includes = []
|
|
|
|
includes << :author unless notes.authors_loaded?
|
|
|
|
includes << :award_emoji unless notes.award_emojis_loaded?
|
|
|
|
if includes.any?
|
|
|
|
notes.includes(includes)
|
|
|
|
else
|
|
|
|
notes
|
|
|
|
end
|
2015-10-13 05:49:01 -04:00
|
|
|
end
|
|
|
|
|
2015-10-22 11:18:59 -04:00
|
|
|
def updated_tasks
|
|
|
|
Taskable.get_updated_tasks(old_content: previous_changes['description'].first,
|
|
|
|
new_content: description)
|
2014-11-05 15:45:18 -05:00
|
|
|
end
|
2016-03-18 09:48:55 -04:00
|
|
|
|
|
|
|
##
|
|
|
|
# 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-20 08:43:11 -04:00
|
|
|
|
2016-10-12 08:01:34 -04:00
|
|
|
def assignee_or_author?(user)
|
|
|
|
# We're comparing IDs here so we don't need to load any associations.
|
|
|
|
author_id == user.id || assignee_id == user.id
|
|
|
|
end
|
|
|
|
|
2016-09-20 08:43:11 -04:00
|
|
|
def record_metrics
|
|
|
|
metrics = self.metrics || create_metrics
|
|
|
|
metrics.record!
|
|
|
|
end
|
2012-08-08 21:26:56 -04:00
|
|
|
end
|