issue.rb 3.01 KB
Newer Older
1 2 3 4
# == Schema Information
#
# Table name: issues
#
Stan Hu's avatar
Stan Hu committed
5 6 7 8 9 10 11 12 13 14 15 16 17 18
#  id            :integer          not null, primary key
#  title         :string(255)
#  assignee_id   :integer
#  author_id     :integer
#  project_id    :integer
#  created_at    :datetime
#  updated_at    :datetime
#  position      :integer          default(0)
#  branch_name   :string(255)
#  description   :text
#  milestone_id  :integer
#  state         :string(255)
#  iid           :integer
#  updated_by_id :integer
19 20
#

21 22 23
require 'carrierwave/orm/activerecord'
require 'file_size_validator'

gitlabhq's avatar
gitlabhq committed
24
class Issue < ActiveRecord::Base
25
  include InternalId
26 27
  include Issuable
  include Referable
28
  include Sortable
29
  include Taskable
30

31 32
  ActsAsTaggableOn.strict_case_match = true

33 34 35
  belongs_to :project
  validates :project, presence: true

36 37 38
  scope :of_group,
    ->(group) { where(project_id: group.projects.select(:id).reorder(nil)) }

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
39
  scope :cared, ->(user) { where(assignee_id: user) }
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
40
  scope :open_for, ->(user) { opened.assigned_to(user) }
41
  scope :in_projects, ->(project_ids) { where(project_id: project_ids) }
42

Andrew8xx8's avatar
Andrew8xx8 committed
43
  state_machine :state, initial: :opened do
Andrew8xx8's avatar
Andrew8xx8 committed
44 45 46 47 48
    event :close do
      transition [:reopened, :opened] => :closed
    end

    event :reopen do
Andrew8xx8's avatar
Andrew8xx8 committed
49
      transition closed: :reopened
Andrew8xx8's avatar
Andrew8xx8 committed
50 51 52 53 54 55
    end

    state :opened
    state :reopened
    state :closed
  end
56

57 58 59 60
  def hook_attrs
    attributes
  end

61 62 63 64
  def self.reference_prefix
    '#'
  end

65 66 67 68 69
  # Pattern used to extract `#123` issue references from text
  #
  # This pattern supports cross-project references.
  def self.reference_pattern
    %r{
70 71
      (#{Project.reference_pattern})?
      #{Regexp.escape(reference_prefix)}(?<issue>\d+)
72
    }x
Kirill Zaitsev's avatar
Kirill Zaitsev committed
73 74
  end

75 76 77 78
  def self.link_reference_pattern
    super("issues", /(?<issue>\d+)/)
  end

79 80 81 82 83 84 85 86 87 88
  def to_reference(from_project = nil)
    reference = "#{self.class.reference_prefix}#{iid}"

    if cross_project_reference?(from_project)
      reference = project.to_reference + reference
    end

    reference
  end

89
  def referenced_merge_requests(current_user = nil)
90 91
    Gitlab::ReferenceExtractor.lazily do
      [self, *notes].flat_map do |note|
92
        note.all_references(current_user).merge_requests
93 94
      end
    end.sort_by(&:iid)
95 96
  end

Drew Blessing's avatar
Drew Blessing committed
97 98 99 100 101 102 103 104 105
  # Reset issue events cache
  #
  # Since we do cache @event we need to reset cache in special cases:
  # * when an issue is updated
  # Events cache stored like  events/23-20130109142513.
  # The cache key includes updated_at timestamp.
  # Thus it will automatically generate a new fragment
  # when the event is updated because the key changes.
  def reset_events_cache
106
    Event.reset_event_cache_for(self)
Drew Blessing's avatar
Drew Blessing committed
107
  end
108 109 110 111 112

  # To allow polymorphism with MergeRequest.
  def source_project
    project
  end
113 114 115

  # From all notes on this issue, we'll select the system notes about linked
  # merge requests. Of those, the MRs closing `self` are returned.
116 117 118
  def closed_by_merge_requests(current_user = nil)
    return [] unless open?

119
    notes.system.flat_map do |note|
120 121
      note.all_references(current_user).merge_requests
    end.uniq.select { |mr| mr.open? && mr.closes_issue?(self) }
122
  end
gitlabhq's avatar
gitlabhq committed
123
end