summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--app/models/ci/stage.rb47
-rw-r--r--spec/models/ci/stage_spec.rb22
2 files changed, 67 insertions, 2 deletions
diff --git a/app/models/ci/stage.rb b/app/models/ci/stage.rb
index c163d047127..066903ddc5b 100644
--- a/app/models/ci/stage.rb
+++ b/app/models/ci/stage.rb
@@ -3,6 +3,7 @@ module Ci
extend Ci::Model
include Importable
include HasStatus
+ include Gitlab::OptimisticLocking
enumerate_status!
@@ -15,5 +16,51 @@ module Ci
validates :project, presence: true, unless: :importing?
validates :pipeline, presence: true, unless: :importing?
validates :name, presence: true, unless: :importing?
+
+ state_machine :status, initial: :created do
+ event :enqueue do
+ transition created: :pending
+ transition [:success, :failed, :canceled, :skipped] => :running
+ end
+
+ event :run do
+ transition any - [:running] => :running
+ end
+
+ event :skip do
+ transition any - [:skipped] => :skipped
+ end
+
+ event :drop do
+ transition any - [:failed] => :failed
+ end
+
+ event :succeed do
+ transition any - [:success] => :success
+ end
+
+ event :cancel do
+ transition any - [:canceled] => :canceled
+ end
+
+ event :block do
+ transition any - [:manual] => :manual
+ end
+ end
+
+ def update!
+ retry_optimistic_lock(self) do
+ case commit_statuses.latest.status
+ when 'pending' then enqueue
+ when 'running' then run
+ when 'success' then succeed
+ when 'failed' then drop
+ when 'canceled' then cancel
+ when 'manual' then block
+ when 'skipped' then skip
+ else skip
+ end
+ end
+ end
end
end
diff --git a/spec/models/ci/stage_spec.rb b/spec/models/ci/stage_spec.rb
index 49573175266..e829ccb048e 100644
--- a/spec/models/ci/stage_spec.rb
+++ b/spec/models/ci/stage_spec.rb
@@ -1,9 +1,9 @@
require 'spec_helper'
describe Ci::Stage, :models do
- describe 'associations' do
- let(:stage) { create(:ci_stage_entity) }
+ let(:stage) { create(:ci_stage_entity) }
+ describe 'associations' do
before do
create(:ci_build, stage_id: stage.id)
create(:commit_status, stage_id: stage.id)
@@ -39,4 +39,22 @@ describe Ci::Stage, :models do
end
end
end
+
+ describe 'update!' do
+ context 'when stage objects needs to be updated' do
+ before do
+ create(:ci_build, :success, stage_id: stage.id)
+ create(:ci_build, :running, stage_id: stage.id)
+ end
+
+ it 'updates stage status correctly' do
+ expect { stage.update! }
+ .to change { stage.reload.status }
+ .to 'running'
+ end
+ end
+
+ context 'when stage object is locked' do
+ end
+ end
end