gitlab-org--gitlab-foss/app/models/milestone.rb

163 lines
3.9 KiB
Ruby
Raw Normal View History

2012-04-08 21:28:58 +00:00
class Milestone < ActiveRecord::Base
# Represents a "No Milestone" state used for filtering Issues and Merge
# Requests that have no milestone assigned.
MilestoneStruct = Struct.new(:title, :name, :id)
None = MilestoneStruct.new('No Milestone', 'No Milestone', 0)
Any = MilestoneStruct.new('Any Milestone', '', -1)
2016-03-11 17:46:14 +00:00
Upcoming = MilestoneStruct.new('Upcoming', '#upcoming', -2)
include InternalId
2015-02-06 00:49:41 +00:00
include Sortable
2015-12-24 13:43:07 +00:00
include Referable
include StripAttribute
2016-03-07 04:07:19 +00:00
include Milestoneish
2012-04-08 21:28:58 +00:00
belongs_to :project
has_many :issues
has_many :labels, -> { distinct.reorder('labels.title') }, through: :issues
has_many :merge_requests
2016-03-07 04:07:19 +00:00
has_many :participants, -> { distinct.reorder('users.name') }, through: :issues, source: :assignee
2012-04-08 21:28:58 +00:00
2013-02-18 09:10:09 +00:00
scope :active, -> { with_state(:active) }
scope :closed, -> { with_state(:closed) }
scope :of_projects, ->(ids) { where(project_id: ids) }
2012-12-14 05:34:05 +00:00
validates :title, presence: true, uniqueness: { scope: :project_id }
2012-10-09 00:10:04 +00:00
validates :project, presence: true
2013-02-18 09:10:09 +00:00
strip_attributes :title
2013-02-18 13:22:18 +00:00
state_machine :state, initial: :active do
2013-02-18 09:10:09 +00:00
event :close do
2013-02-18 13:22:18 +00:00
transition active: :closed
2013-02-18 09:10:09 +00:00
end
event :activate do
2013-02-18 13:22:18 +00:00
transition closed: :active
2013-02-18 09:10:09 +00:00
end
state :closed
state :active
end
2012-04-08 21:28:58 +00:00
alias_attribute :name, :title
2015-08-13 14:48:21 +00:00
class << self
# Searches for milestones matching the given query.
#
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
#
# query - The search query as a String
#
# Returns an ActiveRecord::Relation.
2015-08-13 14:48:21 +00:00
def search(query)
t = arel_table
pattern = "%#{query}%"
where(t[:title].matches(pattern).or(t[:description].matches(pattern)))
2015-08-13 14:48:21 +00:00
end
end
2015-12-24 13:43:07 +00:00
def self.reference_pattern
nil
end
def self.link_reference_pattern
@link_reference_pattern ||= super("milestones", /(?<milestone>\d+)/)
2015-12-24 13:43:07 +00:00
end
def self.upcoming_ids_by_projects(projects)
rel = unscoped.of_projects(projects).active.where('due_date > ?', Time.now)
if Gitlab::Database.postgresql?
rel.order(:project_id, :due_date).pluck('DISTINCT ON (project_id) id')
else
rel.
group(:project_id).
having('due_date = MIN(due_date)').
pluck(:id, :project_id, :due_date).
map(&:first)
end
2016-03-11 17:46:14 +00:00
end
2015-12-24 13:43:07 +00:00
def to_reference(from_project = nil)
2016-01-07 11:26:05 +00:00
escaped_title = self.title.gsub("]", "\\]")
h = Gitlab::Routing.url_helpers
2016-01-07 11:26:05 +00:00
url = h.namespace_project_milestone_url(self.project.namespace, self.project, self)
"[#{escaped_title}](#{url})"
2015-12-24 13:43:07 +00:00
end
def reference_link_text(from_project = nil)
2016-01-05 15:45:53 +00:00
self.title
2015-12-24 13:43:07 +00:00
end
2012-12-14 05:34:05 +00:00
def expired?
if due_date
due_date.past?
2012-12-14 05:34:05 +00:00
else
false
end
2012-04-08 21:28:58 +00:00
end
2012-04-08 21:28:58 +00:00
def expires_at
if due_date
if due_date.past?
"expired on #{due_date.to_s(:medium)}"
else
"expires on #{due_date.to_s(:medium)}"
end
2013-03-20 21:46:30 +00:00
end
2012-04-08 21:28:58 +00:00
end
2012-12-14 05:34:05 +00:00
def can_be_closed?
2013-02-18 09:10:09 +00:00
active? && issues.opened.count.zero?
2012-12-19 03:14:05 +00:00
end
def is_empty?(user = nil)
total_items_count(user).zero?
2012-12-14 05:34:05 +00:00
end
def author_id
nil
end
def title=(value)
write_attribute(:title, Sanitize.clean(value.to_s)) if value.present?
2016-05-04 21:21:57 +00:00
end
# Sorts the issues for the given IDs.
#
# This method runs a single SQL query using a CASE statement to update the
# position of all issues in the current milestone (scoped to the list of IDs).
#
# Given the ids [10, 20, 30] this method produces a SQL query something like
# the following:
#
# UPDATE issues
# SET position = CASE
# WHEN id = 10 THEN 1
# WHEN id = 20 THEN 2
# WHEN id = 30 THEN 3
# ELSE position
# END
# WHERE id IN (10, 20, 30);
#
# This method expects that the IDs given in `ids` are already Fixnums.
def sort_issues(ids)
pairs = []
ids.each_with_index do |id, index|
pairs << id
pairs << index + 1
end
conditions = 'WHEN id = ? THEN ? ' * ids.length
issues.where(id: ids).
update_all(["position = CASE #{conditions} ELSE position END", *pairs])
end
2012-04-08 21:28:58 +00:00
end