2018-08-03 13:22:24 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
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
|
2018-09-20 22:17:37 -04:00
|
|
|
BLOCKED_STATUS = %w[manual scheduled].freeze
|
2019-02-26 21:13:06 -05:00
|
|
|
AVAILABLE_STATUSES = %w[created preparing pending running success failed canceled skipped manual scheduled].freeze
|
2018-09-20 22:17:37 -04:00
|
|
|
STARTED_STATUSES = %w[running success failed skipped manual scheduled].freeze
|
2019-02-26 21:13:06 -05:00
|
|
|
ACTIVE_STATUSES = %w[preparing pending running].freeze
|
2017-02-21 18:32:18 -05:00
|
|
|
COMPLETED_STATUSES = %w[success failed canceled skipped].freeze
|
2019-02-26 21:13:06 -05:00
|
|
|
ORDERED_STATUSES = %w[failed preparing pending running manual scheduled canceled success skipped created].freeze
|
2017-07-04 09:31:15 -04:00
|
|
|
STATUSES_ENUM = { created: 0, pending: 1, running: 2, success: 3,
|
2018-09-20 22:17:37 -04:00
|
|
|
failed: 4, canceled: 5, skipped: 6, manual: 7,
|
2019-02-26 21:13:06 -05:00
|
|
|
scheduled: 8, preparing: 9 }.freeze
|
2016-04-13 09:40:12 -04:00
|
|
|
|
2018-06-04 05:29:39 -04:00
|
|
|
UnknownStatusError = Class.new(StandardError)
|
|
|
|
|
2016-04-13 09:40:12 -04:00
|
|
|
class_methods do
|
2016-04-12 13:59:44 -04:00
|
|
|
def status_sql
|
2017-06-15 08:43:47 -04:00
|
|
|
scope_relevant = respond_to?(:exclude_ignored) ? exclude_ignored : all
|
|
|
|
scope_warnings = respond_to?(:failed_but_allowed) ? failed_but_allowed : none
|
2017-03-02 07:45:01 -05:00
|
|
|
|
2017-06-15 08:43:47 -04:00
|
|
|
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
|
2018-09-20 22:17:37 -04:00
|
|
|
scheduled = scope_relevant.scheduled.select('count(*)').to_sql
|
2019-02-26 21:13:06 -05:00
|
|
|
preparing = scope_relevant.preparing.select('count(*)').to_sql
|
2017-06-15 08:43:47 -04:00
|
|
|
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
|
|
|
|
2016-10-03 06:22:51 -04:00
|
|
|
"(CASE
|
2017-06-15 08:43:47 -04:00
|
|
|
WHEN (#{builds})=(#{skipped}) AND (#{warnings}) THEN 'success'
|
2016-12-05 08:48:37 -05:00
|
|
|
WHEN (#{builds})=(#{skipped}) THEN 'skipped'
|
2016-12-05 11:52:50 -05:00
|
|
|
WHEN (#{builds})=(#{success}) THEN 'success'
|
2016-10-03 07:35:53 -04:00
|
|
|
WHEN (#{builds})=(#{created}) THEN 'created'
|
2019-02-26 21:13:06 -05:00
|
|
|
WHEN (#{builds})=(#{preparing}) THEN 'preparing'
|
2016-12-07 06:47:06 -05:00
|
|
|
WHEN (#{builds})=(#{success})+(#{skipped}) THEN 'success'
|
2016-10-03 06:22:51 -04:00
|
|
|
WHEN (#{builds})=(#{success})+(#{skipped})+(#{canceled}) THEN 'canceled'
|
|
|
|
WHEN (#{builds})=(#{created})+(#{skipped})+(#{pending}) THEN 'pending'
|
2017-03-03 06:51:23 -05:00
|
|
|
WHEN (#{running})+(#{pending})>0 THEN 'running'
|
2017-03-03 08:35:19 -05:00
|
|
|
WHEN (#{manual})>0 THEN 'manual'
|
2018-10-01 00:38:37 -04:00
|
|
|
WHEN (#{scheduled})>0 THEN 'scheduled'
|
2019-02-26 21:13:06 -05:00
|
|
|
WHEN (#{preparing})>0 THEN 'preparing'
|
2017-03-20 14:16:48 -04:00
|
|
|
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
|
2017-05-08 08:26:08 -04:00
|
|
|
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
|
2016-10-04 12:10:23 -04:00
|
|
|
|
|
|
|
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
|
|
|
|
2016-08-11 09:22:35 -04:00
|
|
|
state_machine :status, initial: :created do
|
|
|
|
state :created, value: 'created'
|
2019-02-26 21:13:06 -05:00
|
|
|
state :preparing, value: 'preparing'
|
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'
|
2017-03-03 08:35:19 -05:00
|
|
|
state :manual, value: 'manual'
|
2018-09-21 02:24:19 -04:00
|
|
|
state :scheduled, value: 'scheduled'
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
|
|
|
|
2016-08-11 09:22:35 -04:00
|
|
|
scope :created, -> { where(status: 'created') }
|
2019-02-26 21:13:06 -05:00
|
|
|
scope :preparing, -> { where(status: 'preparing') }
|
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') }
|
2019-01-16 07:09:29 -05:00
|
|
|
scope :failed, -> { where(status: 'failed') }
|
|
|
|
scope :canceled, -> { where(status: 'canceled') }
|
|
|
|
scope :skipped, -> { where(status: 'skipped') }
|
|
|
|
scope :manual, -> { where(status: 'manual') }
|
|
|
|
scope :scheduled, -> { where(status: 'scheduled') }
|
2019-02-26 21:13:06 -05:00
|
|
|
scope :alive, -> { where(status: [:created, :preparing, :pending, :running]) }
|
2017-02-19 17:17:24 -05:00
|
|
|
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]) }
|
2016-11-19 08:04:11 -05:00
|
|
|
scope :failed_or_canceled, -> { where(status: [:failed, :canceled]) }
|
2016-11-16 11:57:50 -05:00
|
|
|
|
|
|
|
scope :cancelable, -> do
|
2019-02-26 21:13:06 -05:00
|
|
|
where(status: [:running, :preparing, :pending, :created, :scheduled])
|
2016-11-16 11:57:50 -05:00
|
|
|
end
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def started?
|
2016-08-11 09:22:35 -04:00
|
|
|
STARTED_STATUSES.include?(status) && started_at
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def active?
|
2016-08-11 09:22:35 -04:00
|
|
|
ACTIVE_STATUSES.include?(status)
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def complete?
|
2016-08-11 09:22:35 -04:00
|
|
|
COMPLETED_STATUSES.include?(status)
|
2016-03-31 13:51:28 -04:00
|
|
|
end
|
2016-08-10 10:45:30 -04:00
|
|
|
|
2017-03-05 03:24:57 -05:00
|
|
|
def blocked?
|
2018-09-24 00:12:11 -04:00
|
|
|
BLOCKED_STATUS.include?(status)
|
2017-03-05 03:24:57 -05:00
|
|
|
end
|
|
|
|
|
2016-08-10 10:45:30 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def calculate_duration
|
|
|
|
if started_at && finished_at
|
|
|
|
finished_at - started_at
|
|
|
|
elsif started_at
|
2016-08-19 04:57:25 -04:00
|
|
|
Time.now - started_at
|
2016-08-10 10:45:30 -04:00
|
|
|
end
|
|
|
|
end
|
2016-04-12 04:23:31 -04:00
|
|
|
end
|