gitlab-org--gitlab-foss/app/models/concerns/has_status.rb

118 lines
3.9 KiB
Ruby
Raw Normal View History

2016-08-24 22:55:32 -04:00
module HasStatus
2016-03-31 13:51:28 -04:00
extend ActiveSupport::Concern
2017-02-21 18:32:18 -05:00
DEFAULT_STATUS = 'created'.freeze
BLOCKED_STATUS = 'manual'.freeze
AVAILABLE_STATUSES = %w[created pending running success failed canceled skipped manual].freeze
STARTED_STATUSES = %w[running success failed skipped manual].freeze
ACTIVE_STATUSES = %w[pending running].freeze
2017-02-21 18:32:18 -05:00
COMPLETED_STATUSES = %w[success failed canceled skipped].freeze
ORDERED_STATUSES = %w[failed pending running manual canceled success skipped created].freeze
2016-04-13 09:40:12 -04:00
class_methods do
2016-04-12 13:59:44 -04:00
def status_sql
scope_relevant = respond_to?(:exclude_ignored) ? exclude_ignored : all
scope_warnings = respond_to?(:failed_but_allowed) ? failed_but_allowed : none
builds = scope_relevant.select('count(*)').to_sql
created = scope_relevant.created.select('count(*)').to_sql
success = scope_relevant.success.select('count(*)').to_sql
manual = scope_relevant.manual.select('count(*)').to_sql
pending = scope_relevant.pending.select('count(*)').to_sql
running = scope_relevant.running.select('count(*)').to_sql
skipped = scope_relevant.skipped.select('count(*)').to_sql
canceled = scope_relevant.canceled.select('count(*)').to_sql
warnings = scope_warnings.select('count(*) > 0').to_sql.presence || 'false'
2016-04-12 13:59:44 -04:00
"(CASE
WHEN (#{builds})=(#{skipped}) AND (#{warnings}) THEN 'success'
WHEN (#{builds})=(#{skipped}) THEN 'skipped'
WHEN (#{builds})=(#{success}) THEN 'success'
WHEN (#{builds})=(#{created}) THEN 'created'
2016-12-07 06:47:06 -05:00
WHEN (#{builds})=(#{success})+(#{skipped}) THEN 'success'
WHEN (#{builds})=(#{success})+(#{skipped})+(#{canceled}) THEN 'canceled'
WHEN (#{builds})=(#{created})+(#{skipped})+(#{pending}) THEN 'pending'
WHEN (#{running})+(#{pending})>0 THEN 'running'
WHEN (#{manual})>0 THEN 'manual'
WHEN (#{created})>0 THEN 'running'
2016-04-12 13:59:44 -04:00
ELSE 'failed'
END)"
end
2016-03-31 13:51:28 -04:00
def status
all.pluck(status_sql).first
2016-03-31 13:51:28 -04:00
end
2016-04-13 14:51:03 -04:00
def started_at
all.minimum(:started_at)
end
def finished_at
2016-04-16 15:46:26 -04:00
all.maximum(:finished_at)
2016-04-13 14:51:03 -04:00
end
def all_state_names
state_machines.values.flat_map(&:states).flat_map { |s| s.map(&:name) }
end
2016-03-31 13:51:28 -04:00
end
included do
2016-04-13 09:40:12 -04:00
validates :status, inclusion: { in: AVAILABLE_STATUSES }
2016-03-31 13:51:28 -04:00
state_machine :status, initial: :created do
state :created, value: 'created'
2016-03-31 13:51:28 -04:00
state :pending, value: 'pending'
state :running, value: 'running'
state :failed, value: 'failed'
state :success, value: 'success'
state :canceled, value: 'canceled'
2016-04-11 10:55:40 -04:00
state :skipped, value: 'skipped'
state :manual, value: 'manual'
2016-03-31 13:51:28 -04:00
end
scope :created, -> { where(status: 'created') }
2017-04-07 11:06:41 -04:00
scope :relevant, -> { where(status: AVAILABLE_STATUSES - ['created']) }
2016-03-31 13:51:28 -04:00
scope :running, -> { where(status: 'running') }
scope :pending, -> { where(status: 'pending') }
scope :success, -> { where(status: 'success') }
scope :failed, -> { where(status: 'failed') }
2016-04-12 13:57:54 -04:00
scope :canceled, -> { where(status: 'canceled') }
2016-04-13 11:26:22 -04:00
scope :skipped, -> { where(status: 'skipped') }
scope :manual, -> { where(status: 'manual') }
scope :created_or_pending, -> { where(status: [:created, :pending]) }
2016-03-31 13:51:28 -04:00
scope :running_or_pending, -> { where(status: [:running, :pending]) }
scope :finished, -> { where(status: [:success, :failed, :canceled]) }
scope :failed_or_canceled, -> { where(status: [:failed, :canceled]) }
scope :cancelable, -> do
where(status: [:running, :pending, :created])
end
2016-03-31 13:51:28 -04:00
end
def started?
STARTED_STATUSES.include?(status) && started_at
2016-03-31 13:51:28 -04:00
end
def active?
ACTIVE_STATUSES.include?(status)
2016-03-31 13:51:28 -04:00
end
def complete?
COMPLETED_STATUSES.include?(status)
2016-03-31 13:51:28 -04:00
end
def blocked?
BLOCKED_STATUS == status
end
private
def calculate_duration
if started_at && finished_at
finished_at - started_at
elsif started_at
Time.now - started_at
end
end
2016-04-12 04:23:31 -04:00
end