Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
80863e73
Commit
80863e73
authored
Jan 01, 2022
by
Fabio Pitino
Committed by
Stan Hu
Jan 01, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Introduce Gitlab::EventStore as simple pub-sub system
parent
6ef717fc
Changes
18
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
18 changed files
with
1063 additions
and
112 deletions
+1063
-112
app/events/ci/pipeline_created_event.rb
app/events/ci/pipeline_created_event.rb
+14
-0
app/services/ci/create_pipeline_service.rb
app/services/ci/create_pipeline_service.rb
+8
-1
app/workers/all_queues.yml
app/workers/all_queues.yml
+9
-0
app/workers/merge_requests/update_head_pipeline_worker.rb
app/workers/merge_requests/update_head_pipeline_worker.rb
+23
-0
app/workers/update_head_pipeline_for_merge_request_worker.rb
app/workers/update_head_pipeline_for_merge_request_worker.rb
+2
-0
config/feature_flags/development/ci_publish_pipeline_events.yml
.../feature_flags/development/ci_publish_pipeline_events.yml
+8
-0
config/sidekiq_queues.yml
config/sidekiq_queues.yml
+2
-0
doc/development/event_store.md
doc/development/event_store.md
+292
-0
doc/development/index.md
doc/development/index.md
+1
-0
lib/gitlab/event_store.rb
lib/gitlab/event_store.rb
+42
-0
lib/gitlab/event_store/event.rb
lib/gitlab/event_store/event.rb
+54
-0
lib/gitlab/event_store/store.rb
lib/gitlab/event_store/store.rb
+54
-0
lib/gitlab/event_store/subscriber.rb
lib/gitlab/event_store/subscriber.rb
+36
-0
lib/gitlab/event_store/subscription.rb
lib/gitlab/event_store/subscription.rb
+37
-0
spec/lib/gitlab/event_store/event_spec.rb
spec/lib/gitlab/event_store/event_spec.rb
+64
-0
spec/lib/gitlab/event_store/store_spec.rb
spec/lib/gitlab/event_store/store_spec.rb
+262
-0
spec/services/ci/create_pipeline_service_spec.rb
spec/services/ci/create_pipeline_service_spec.rb
+17
-111
spec/workers/merge_requests/update_head_pipeline_worker_spec.rb
...orkers/merge_requests/update_head_pipeline_worker_spec.rb
+138
-0
No files found.
app/events/ci/pipeline_created_event.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
module
Ci
class
PipelineCreatedEvent
<
::
Gitlab
::
EventStore
::
Event
def
schema
{
'type'
=>
'object'
,
'properties'
=>
{
'pipeline_id'
=>
{
'type'
=>
'integer'
}
}
}
end
end
end
app/services/ci/create_pipeline_service.rb
View file @
80863e73
...
@@ -95,7 +95,14 @@ module Ci
...
@@ -95,7 +95,14 @@ module Ci
.
build!
.
build!
if
pipeline
.
persisted?
if
pipeline
.
persisted?
if
Feature
.
enabled?
(
:ci_publish_pipeline_events
,
pipeline
.
project
,
default_enabled: :yaml
)
Gitlab
::
EventStore
.
publish
(
Ci
::
PipelineCreatedEvent
.
new
(
data:
{
pipeline_id:
pipeline
.
id
})
)
else
schedule_head_pipeline_update
schedule_head_pipeline_update
end
create_namespace_onboarding_action
create_namespace_onboarding_action
else
else
# If pipeline is not persisted, try to recover IID
# If pipeline is not persisted, try to recover IID
...
...
app/workers/all_queues.yml
View file @
80863e73
...
@@ -2420,6 +2420,15 @@
...
@@ -2420,6 +2420,15 @@
:weight:
1
:weight:
1
:idempotent:
true
:idempotent:
true
:tags: []
:tags: []
-
:name: merge_requests_update_head_pipeline
:worker_name: MergeRequests::UpdateHeadPipelineWorker
:feature_category: :code_review
:has_external_dependencies:
:urgency: :high
:resource_boundary: :cpu
:weight:
1
:idempotent:
true
:tags: []
-
:name: metrics_dashboard_prune_old_annotations
-
:name: metrics_dashboard_prune_old_annotations
:worker_name: Metrics::Dashboard::PruneOldAnnotationsWorker
:worker_name: Metrics::Dashboard::PruneOldAnnotationsWorker
:feature_category: :metrics
:feature_category: :metrics
...
...
app/workers/merge_requests/update_head_pipeline_worker.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
module
MergeRequests
class
UpdateHeadPipelineWorker
include
ApplicationWorker
include
Gitlab
::
EventStore
::
Subscriber
feature_category
:code_review
urgency
:high
worker_resource_boundary
:cpu
data_consistency
:always
idempotent!
def
handle_event
(
event
)
Ci
::
Pipeline
.
find_by_id
(
event
.
data
[
:pipeline_id
]).
try
do
|
pipeline
|
pipeline
.
all_merge_requests
.
opened
.
each
do
|
merge_request
|
UpdateHeadPipelineForMergeRequestWorker
.
perform_async
(
merge_request
.
id
)
end
end
end
end
end
app/workers/update_head_pipeline_for_merge_request_worker.rb
View file @
80863e73
...
@@ -8,8 +8,10 @@ class UpdateHeadPipelineForMergeRequestWorker
...
@@ -8,8 +8,10 @@ class UpdateHeadPipelineForMergeRequestWorker
sidekiq_options
retry:
3
sidekiq_options
retry:
3
include
PipelineQueue
include
PipelineQueue
# NOTE: this worker belongs to :code_review since there is no CI logic.
queue_namespace
:pipeline_processing
queue_namespace
:pipeline_processing
feature_category
:continuous_integration
feature_category
:continuous_integration
urgency
:high
urgency
:high
worker_resource_boundary
:cpu
worker_resource_boundary
:cpu
...
...
config/feature_flags/development/ci_publish_pipeline_events.yml
0 → 100644
View file @
80863e73
---
name
:
ci_publish_pipeline_events
introduced_by_url
:
https://gitlab.com/gitlab-org/gitlab/-/merge_requests/34042
rollout_issue_url
:
https://gitlab.com/gitlab-org/gitlab/-/issues/336752
milestone
:
'
14.3'
type
:
development
group
:
group::pipeline execution
default_enabled
:
false
config/sidekiq_queues.yml
View file @
80863e73
...
@@ -255,6 +255,8 @@
...
@@ -255,6 +255,8 @@
-
1
-
1
-
-
merge_requests_sync_code_owner_approval_rules
-
-
merge_requests_sync_code_owner_approval_rules
-
1
-
1
-
-
merge_requests_update_head_pipeline
-
1
-
-
metrics_dashboard_prune_old_annotations
-
-
metrics_dashboard_prune_old_annotations
-
1
-
1
-
-
metrics_dashboard_sync_dashboards
-
-
metrics_dashboard_sync_dashboards
...
...
doc/development/event_store.md
0 → 100644
View file @
80863e73
This diff is collapsed.
Click to expand it.
doc/development/index.md
View file @
80863e73
...
@@ -164,6 +164,7 @@ the [reviewer values](https://about.gitlab.com/handbook/engineering/workflow/rev
...
@@ -164,6 +164,7 @@ the [reviewer values](https://about.gitlab.com/handbook/engineering/workflow/rev
### General
### General
-
[
Directory structure
](
directory_structure.md
)
-
[
Directory structure
](
directory_structure.md
)
-
[
GitLab EventStore
](
event_store.md
)
to publish/subscribe to domain events
-
[
GitLab utilities
](
utilities.md
)
-
[
GitLab utilities
](
utilities.md
)
-
[
Newlines style guide
](
newlines_styleguide.md
)
-
[
Newlines style guide
](
newlines_styleguide.md
)
-
[
Logging
](
logging.md
)
-
[
Logging
](
logging.md
)
...
...
lib/gitlab/event_store.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
# Gitlab::EventStore is a simple pub-sub mechanism that lets you publish
# domain events and use Sidekiq workers as event handlers.
#
# It can be used to decouple domains from different bounded contexts
# by publishing domain events and let any interested parties subscribe
# to them.
#
module
Gitlab
module
EventStore
Error
=
Class
.
new
(
StandardError
)
InvalidEvent
=
Class
.
new
(
Error
)
InvalidSubscriber
=
Class
.
new
(
Error
)
def
self
.
publish
(
event
)
instance
.
publish
(
event
)
end
def
self
.
instance
@instance
||=
configure!
end
# Define all event subscriptions using:
#
# store.subscribe(DomainA::SomeWorker, to: DomainB::SomeEvent)
#
# It is possible to subscribe to a subset of events matching a condition:
#
# store.subscribe(DomainA::SomeWorker, to: DomainB::SomeEvent), if: ->(event) { event.data == :some_value }
#
def
self
.
configure!
Store
.
new
do
|
store
|
###
# Add subscriptions here:
store
.
subscribe
::
MergeRequests
::
UpdateHeadPipelineWorker
,
to:
::
Ci
::
PipelineCreatedEvent
end
end
private_class_method
:configure!
end
end
lib/gitlab/event_store/event.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
# An Event object represents a domain event that occurred in a bounded context.
# By publishing events we notify other bounded contexts about something
# that happened, so that they can react to it.
#
# Define new event classes under `app/events/<namespace>/` with a name
# representing something that happened in the past:
#
# class Projects::ProjectCreatedEvent < Gitlab::EventStore::Event
# def schema
# {
# 'type' => 'object',
# 'properties' => {
# 'project_id' => { 'type' => 'integer' }
# }
# }
# end
# end
#
# To publish it:
#
# Gitlab::EventStore.publish(
# Projects::ProjectCreatedEvent.new(data: { project_id: project.id })
# )
#
module
Gitlab
module
EventStore
class
Event
attr_reader
:data
def
initialize
(
data
:)
validate_schema!
(
data
)
@data
=
data
end
def
schema
raise
NotImplementedError
,
'must specify schema to validate the event'
end
private
def
validate_schema!
(
data
)
unless
data
.
is_a?
(
Hash
)
raise
Gitlab
::
EventStore
::
InvalidEvent
,
"Event data must be a Hash"
end
unless
JSONSchemer
.
schema
(
schema
).
valid?
(
data
.
deep_stringify_keys
)
raise
Gitlab
::
EventStore
::
InvalidEvent
,
"Data for event
#{
self
.
class
}
does not match the defined schema:
#{
schema
}
"
end
end
end
end
end
lib/gitlab/event_store/store.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
module
Gitlab
module
EventStore
class
Store
attr_reader
:subscriptions
def
initialize
@subscriptions
=
Hash
.
new
{
|
h
,
k
|
h
[
k
]
=
[]
}
yield
(
self
)
if
block_given?
# freeze the subscriptions as safety measure to avoid further
# subcriptions after initialization.
lock!
end
def
subscribe
(
worker
,
to
:,
if:
nil
)
condition
=
binding
.
local_variable_get
(
'if'
)
Array
(
to
).
each
do
|
event
|
validate_subscription!
(
worker
,
event
)
subscriptions
[
event
]
<<
Gitlab
::
EventStore
::
Subscription
.
new
(
worker
,
condition
)
end
end
def
publish
(
event
)
unless
event
.
is_a?
(
Event
)
raise
InvalidEvent
,
"Event being published is not an instance of Gitlab::EventStore::Event: got
#{
event
.
inspect
}
"
end
subscriptions
[
event
.
class
].
each
do
|
subscription
|
subscription
.
consume_event
(
event
)
end
end
private
def
lock!
@subscriptions
.
freeze
end
def
validate_subscription!
(
subscriber
,
event_class
)
unless
event_class
<
Event
raise
InvalidEvent
,
"Event being subscribed to is not a subclass of Gitlab::EventStore::Event: got
#{
event_class
}
"
end
unless
subscriber
.
respond_to?
(
:perform_async
)
raise
InvalidSubscriber
,
"Subscriber is not an ApplicationWorker: got
#{
subscriber
}
"
end
end
end
end
end
lib/gitlab/event_store/subscriber.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
# This module should be included in order to turn an ApplicationWorker
# into a Subscriber.
# This module overrides the `perform` method and provides a better and
# safer interface for handling events via `handle_event` method.
#
# @example:
# class SomeEventSubscriber
# include ApplicationWorker
# include Gitlab::EventStore::Subscriber
#
# def handle_event(event)
# # ...
# end
# end
module
Gitlab
module
EventStore
module
Subscriber
def
perform
(
event_type
,
data
)
raise
InvalidEvent
,
event_type
unless
self
.
class
.
const_defined?
(
event_type
)
event
=
event_type
.
constantize
.
new
(
data:
data
.
with_indifferent_access
)
handle_event
(
event
)
end
def
handle_event
(
event
)
raise
NotImplementedError
,
'you must implement this methods in order to handle events'
end
end
end
end
lib/gitlab/event_store/subscription.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
module
Gitlab
module
EventStore
class
Subscription
attr_reader
:worker
,
:condition
def
initialize
(
worker
,
condition
)
@worker
=
worker
@condition
=
condition
end
def
consume_event
(
event
)
return
unless
condition_met?
(
event
)
worker
.
perform_async
(
event
.
class
.
name
,
event
.
data
)
# TODO: Log dispatching of events to subscriber
# We rescue and track any exceptions here because we don't want to
# impact other subscribers if one is faulty.
# The method `condition_met?`, since it can run a block, it might encounter
# a bug. By raising an exception here we could interrupt the publishing
# process, preventing other subscribers from consuming the event.
rescue
StandardError
=>
e
Gitlab
::
ErrorTracking
.
track_and_raise_for_dev_exception
(
e
,
event_class:
event
.
class
.
name
,
event_data:
event
.
data
)
end
private
def
condition_met?
(
event
)
return
true
unless
condition
condition
.
call
(
event
)
end
end
end
end
spec/lib/gitlab/event_store/event_spec.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
Gitlab
::
EventStore
::
Event
do
let
(
:event_class
)
{
stub_const
(
'TestEvent'
,
Class
.
new
(
described_class
))
}
let
(
:event
)
{
event_class
.
new
(
data:
data
)
}
let
(
:data
)
{
{
project_id:
123
,
project_path:
'org/the-project'
}
}
context
'when schema is not defined'
do
it
'raises an error on initialization'
do
expect
{
event
}.
to
raise_error
(
NotImplementedError
)
end
end
context
'when schema is defined'
do
before
do
event_class
.
class_eval
do
def
schema
{
'required'
=>
[
'project_id'
],
'type'
=>
'object'
,
'properties'
=>
{
'project_id'
=>
{
'type'
=>
'integer'
},
'project_path'
=>
{
'type'
=>
'string'
}
}
}
end
end
end
describe
'schema validation'
do
context
'when data matches the schema'
do
it
'initializes the event correctly'
do
expect
(
event
.
data
).
to
eq
(
data
)
end
end
context
'when required properties are present as well as unknown properties'
do
let
(
:data
)
{
{
project_id:
123
,
unknown_key:
'unknown_value'
}
}
it
'initializes the event correctly'
do
expect
(
event
.
data
).
to
eq
(
data
)
end
end
context
'when some properties are missing'
do
let
(
:data
)
{
{
project_path:
'org/the-project'
}
}
it
'expects all properties to be present'
do
expect
{
event
}.
to
raise_error
(
Gitlab
::
EventStore
::
InvalidEvent
,
/does not match the defined schema/
)
end
end
context
'when data is not a Hash'
do
let
(
:data
)
{
123
}
it
'raises an error'
do
expect
{
event
}.
to
raise_error
(
Gitlab
::
EventStore
::
InvalidEvent
,
'Event data must be a Hash'
)
end
end
end
end
end
spec/lib/gitlab/event_store/store_spec.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
Gitlab
::
EventStore
::
Store
do
let
(
:event_klass
)
{
stub_const
(
'TestEvent'
,
Class
.
new
(
Gitlab
::
EventStore
::
Event
))
}
let
(
:event
)
{
event_klass
.
new
(
data:
data
)
}
let
(
:another_event_klass
)
{
stub_const
(
'TestAnotherEvent'
,
Class
.
new
(
Gitlab
::
EventStore
::
Event
))
}
let
(
:worker
)
do
stub_const
(
'EventSubscriber'
,
Class
.
new
).
tap
do
|
klass
|
klass
.
class_eval
do
include
ApplicationWorker
include
Gitlab
::
EventStore
::
Subscriber
def
handle_event
(
event
)
event
.
data
end
end
end
end
let
(
:another_worker
)
do
stub_const
(
'AnotherEventSubscriber'
,
Class
.
new
).
tap
do
|
klass
|
klass
.
class_eval
do
include
ApplicationWorker
include
Gitlab
::
EventStore
::
Subscriber
end
end
end
let
(
:unrelated_worker
)
do
stub_const
(
'UnrelatedEventSubscriber'
,
Class
.
new
).
tap
do
|
klass
|
klass
.
class_eval
do
include
ApplicationWorker
include
Gitlab
::
EventStore
::
Subscriber
end
end
end
before
do
event_klass
.
class_eval
do
def
schema
{
'required'
=>
%w[name id]
,
'type'
=>
'object'
,
'properties'
=>
{
'name'
=>
{
'type'
=>
'string'
},
'id'
=>
{
'type'
=>
'integer'
}
}
}
end
end
end
describe
'#subscribe'
do
it
'subscribes a worker to an event'
do
store
=
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
event_klass
end
subscriptions
=
store
.
subscriptions
[
event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
)
end
it
'subscribes multiple workers to an event'
do
store
=
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
event_klass
s
.
subscribe
another_worker
,
to:
event_klass
end
subscriptions
=
store
.
subscriptions
[
event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
,
another_worker
)
end
it
'subscribes a worker to multiple events is separate calls'
do
store
=
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
event_klass
s
.
subscribe
worker
,
to:
another_event_klass
end
subscriptions
=
store
.
subscriptions
[
event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
)
subscriptions
=
store
.
subscriptions
[
another_event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
)
end
it
'subscribes a worker to multiple events in a single call'
do
store
=
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
[
event_klass
,
another_event_klass
]
end
subscriptions
=
store
.
subscriptions
[
event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
)
subscriptions
=
store
.
subscriptions
[
another_event_klass
]
expect
(
subscriptions
.
map
(
&
:worker
)).
to
contain_exactly
(
worker
)
end
it
'subscribes a worker to an event with condition'
do
store
=
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
event_klass
,
if:
->
(
event
)
{
event
.
data
[
:name
]
==
'Alice'
}
end
subscriptions
=
store
.
subscriptions
[
event_klass
]
expect
(
subscriptions
.
size
).
to
eq
(
1
)
subscription
=
subscriptions
.
first
expect
(
subscription
).
to
be_an_instance_of
(
Gitlab
::
EventStore
::
Subscription
)
expect
(
subscription
.
worker
).
to
eq
(
worker
)
expect
(
subscription
.
condition
.
call
(
double
(
data:
{
name:
'Bob'
}))).
to
eq
(
false
)
expect
(
subscription
.
condition
.
call
(
double
(
data:
{
name:
'Alice'
}))).
to
eq
(
true
)
end
it
'refuses the subscription if the target is not an Event object'
do
expect
do
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
Integer
end
end
.
to
raise_error
(
Gitlab
::
EventStore
::
Error
,
/Event being subscribed to is not a subclass of Gitlab::EventStore::Event/
)
end
it
'refuses the subscription if the subscriber is not a worker'
do
expect
do
described_class
.
new
do
|
s
|
s
.
subscribe
double
,
to:
event_klass
end
end
.
to
raise_error
(
Gitlab
::
EventStore
::
Error
,
/Subscriber is not an ApplicationWorker/
)
end
end
describe
'#publish'
do
let
(
:data
)
{
{
name:
'Bob'
,
id:
123
}
}
context
'when event has a subscribed worker'
do
let
(
:store
)
do
described_class
.
new
do
|
store
|
store
.
subscribe
worker
,
to:
event_klass
store
.
subscribe
another_worker
,
to:
another_event_klass
end
end
it
'dispatches the event to the subscribed worker'
do
expect
(
worker
).
to
receive
(
:perform_async
).
with
(
'TestEvent'
,
data
)
expect
(
another_worker
).
not_to
receive
(
:perform_async
)
store
.
publish
(
event
)
end
context
'when other workers subscribe to the same event'
do
let
(
:store
)
do
described_class
.
new
do
|
store
|
store
.
subscribe
worker
,
to:
event_klass
store
.
subscribe
another_worker
,
to:
event_klass
store
.
subscribe
unrelated_worker
,
to:
another_event_klass
end
end
it
'dispatches the event to each subscribed worker'
do
expect
(
worker
).
to
receive
(
:perform_async
).
with
(
'TestEvent'
,
data
)
expect
(
another_worker
).
to
receive
(
:perform_async
).
with
(
'TestEvent'
,
data
)
expect
(
unrelated_worker
).
not_to
receive
(
:perform_async
)
store
.
publish
(
event
)
end
end
context
'when an error is raised'
do
before
do
allow
(
worker
).
to
receive
(
:perform_async
).
and_raise
(
NoMethodError
,
'the error message'
)
end
it
'is rescued and tracked'
do
expect
(
Gitlab
::
ErrorTracking
)
.
to
receive
(
:track_and_raise_for_dev_exception
)
.
with
(
kind_of
(
NoMethodError
),
event_class:
event
.
class
.
name
,
event_data:
event
.
data
)
.
and_call_original
expect
{
store
.
publish
(
event
)
}.
to
raise_error
(
NoMethodError
,
'the error message'
)
end
end
it
'raises and tracks an error when event is published inside a database transaction'
do
expect
(
Gitlab
::
ErrorTracking
)
.
to
receive
(
:track_and_raise_for_dev_exception
)
.
at_least
(
:once
)
.
and_call_original
expect
do
ApplicationRecord
.
transaction
do
store
.
publish
(
event
)
end
end
.
to
raise_error
(
Sidekiq
::
Worker
::
EnqueueFromTransactionError
)
end
it
'refuses publishing if the target is not an Event object'
do
expect
{
store
.
publish
(
double
(
:event
))
}
.
to
raise_error
(
Gitlab
::
EventStore
::
Error
,
/Event being published is not an instance of Gitlab::EventStore::Event/
)
end
end
context
'when event has subscribed workers with condition'
do
let
(
:store
)
do
described_class
.
new
do
|
s
|
s
.
subscribe
worker
,
to:
event_klass
,
if:
->
(
event
)
{
event
.
data
[
:name
]
==
'Bob'
}
s
.
subscribe
another_worker
,
to:
event_klass
,
if:
->
(
event
)
{
event
.
data
[
:name
]
==
'Alice'
}
end
end
let
(
:event
)
{
event_klass
.
new
(
data:
data
)
}
it
'dispatches the event to the workers satisfying the condition'
do
expect
(
worker
).
to
receive
(
:perform_async
).
with
(
'TestEvent'
,
data
)
expect
(
another_worker
).
not_to
receive
(
:perform_async
)
store
.
publish
(
event
)
end
end
end
describe
'subscriber'
do
let
(
:data
)
{
{
name:
'Bob'
,
id:
123
}
}
let
(
:event_name
)
{
event
.
class
.
name
}
let
(
:worker_instance
)
{
worker
.
new
}
subject
{
worker_instance
.
perform
(
event_name
,
data
)
}
it
'handles the event'
do
expect
(
worker_instance
).
to
receive
(
:handle_event
).
with
(
instance_of
(
event
.
class
))
expect_any_instance_of
(
event
.
class
)
do
|
event
|
expect
(
event
).
to
receive
(
:data
).
and_return
(
data
)
end
subject
end
context
'when the event name does not exist'
do
let
(
:event_name
)
{
'UnknownClass'
}
it
'raises an error'
do
expect
{
subject
}.
to
raise_error
(
Gitlab
::
EventStore
::
InvalidEvent
)
end
end
context
'when the worker does not define handle_event method'
do
let
(
:worker_instance
)
{
another_worker
.
new
}
it
'raises an error'
do
expect
{
subject
}.
to
raise_error
(
NotImplementedError
)
end
end
end
end
spec/services/ci/create_pipeline_service_spec.rb
View file @
80863e73
...
@@ -146,138 +146,44 @@ RSpec.describe Ci::CreatePipelineService do
...
@@ -146,138 +146,44 @@ RSpec.describe Ci::CreatePipelineService do
end
end
context
'when merge requests already exist for this source branch'
do
context
'when merge requests already exist for this source branch'
do
let
(
:merge_request_1
)
do
let
!
(
:merge_request_1
)
do
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"master"
,
source_project:
project
)
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"master"
,
source_project:
project
)
end
end
let
(
:merge_request_2
)
do
let
!
(
:merge_request_2
)
do
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"v1.1.0"
,
source_project:
project
)
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"v1.1.0"
,
source_project:
project
)
end
end
context
'when related merge request is already merged'
do
let!
(
:merged_merge_request
)
do
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
"branch_2"
,
source_project:
project
,
state:
'merged'
)
end
it
'does not schedule update head pipeline job'
do
expect
(
UpdateHeadPipelineForMergeRequestWorker
).
not_to
receive
(
:perform_async
).
with
(
merged_merge_request
.
id
)
execute_service
end
end
context
'when the head pipeline sha equals merge request sha'
do
context
'when the head pipeline sha equals merge request sha'
do
it
'updates head pipeline of each merge request'
,
:sidekiq_might_not_need_inline
do
it
'updates head pipeline of each merge request'
,
:sidekiq_might_not_need_inline
do
merge_request_1
merge_request_2
head_pipeline
=
execute_service
(
ref:
'feature'
,
after:
nil
).
payload
head_pipeline
=
execute_service
(
ref:
'feature'
,
after:
nil
).
payload
expect
(
merge_request_1
.
reload
.
head_pipeline
).
to
eq
(
head_pipeline
)
expect
(
merge_request_1
.
reload
.
head_pipeline
).
to
eq
(
head_pipeline
)
expect
(
merge_request_2
.
reload
.
head_pipeline
).
to
eq
(
head_pipeline
)
expect
(
merge_request_2
.
reload
.
head_pipeline
).
to
eq
(
head_pipeline
)
end
end
end
context
'when the head pipeline sha does not equal merge request sha'
do
it
'does not update the head piepeline of MRs'
do
merge_request_1
merge_request_2
allow_any_instance_of
(
Ci
::
Pipeline
).
to
receive
(
:latest?
).
and_return
(
true
)
expect
{
execute_service
(
after:
'ae73cb07c9eeaf35924a10f713b364d32b2dd34f'
)
}.
not_to
raise_error
last_pipeline
=
Ci
::
Pipeline
.
last
# TODO: remove after ci_publish_pipeline_events FF is removed
# https://gitlab.com/gitlab-org/gitlab/-/issues/336752
expect
(
merge_request_1
.
reload
.
head_pipeline
).
not_to
eq
(
last_pipeline
)
it
'does not schedule sync update for the head pipeline of the merge request'
do
expect
(
merge_request_2
.
reload
.
head_pipeline
).
not_to
eq
(
last_pipeline
)
expect
(
UpdateHeadPipelineForMergeRequestWorker
)
end
.
not_to
receive
(
:perform_async
)
end
context
'when there is no pipeline for source branch'
do
it
"does not update merge request head pipeline"
do
merge_request
=
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"branch_1"
,
source_project:
project
)
head_pipeline
=
execute_service
.
payload
expect
(
merge_request
.
reload
.
head_pipeline
).
not_to
eq
(
head_pipeline
)
end
end
context
'when merge request target project is different from source project'
do
let!
(
:project
)
{
fork_project
(
target_project
,
nil
,
repository:
true
)
}
let!
(
:target_project
)
{
create
(
:project
,
:repository
)
}
let!
(
:user
)
{
create
(
:user
)
}
before
do
project
.
add_developer
(
user
)
end
it
'updates head pipeline for merge request'
,
:sidekiq_might_not_need_inline
do
merge_request
=
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"master"
,
source_project:
project
,
target_project:
target_project
)
head_pipeline
=
execute_service
(
ref:
'feature'
,
after:
nil
).
payload
expect
(
merge_request
.
reload
.
head_pipeline
).
to
eq
(
head_pipeline
)
end
end
context
'when the pipeline is not the latest for the branch'
do
it
'does not update merge request head pipeline'
do
merge_request
=
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
"branch_1"
,
source_project:
project
)
allow_any_instance_of
(
MergeRequest
)
.
to
receive
(
:find_actual_head_pipeline
)
{
}
execute_service
expect
(
merge_request
.
reload
.
head_pipeline
).
to
be_nil
end
end
context
'when pipeline has errors'
do
before
do
stub_ci_pipeline_yaml_file
(
'some invalid syntax'
)
end
it
'updates merge request head pipeline reference'
,
:sidekiq_might_not_need_inline
do
merge_request
=
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
'feature'
,
source_project:
project
)
head_pipeline
=
execute_service
.
payload
expect
(
head_pipeline
).
to
be_persisted
execute_service
(
ref:
'feature'
,
after:
nil
)
expect
(
head_pipeline
.
yaml_errors
).
to
be_present
expect
(
head_pipeline
.
messages
).
to
be_present
expect
(
merge_request
.
reload
.
head_pipeline
).
to
eq
head_pipeline
end
end
end
end
context
'when
pipeline has been skipp
ed'
do
context
'when
feature flag ci_publish_pipeline_events is disabl
ed'
do
before
do
before
do
allow_any_instance_of
(
Ci
::
Pipeline
)
stub_feature_flags
(
ci_publish_pipeline_events:
false
)
.
to
receive
(
:git_commit_message
)
.
and_return
(
'some commit [ci skip]'
)
end
end
it
'updates merge request head pipeline'
,
:sidekiq_might_not_need_inline
do
it
'schedules update for the head pipeline of the merge request'
do
merge_request
=
create
(
:merge_request
,
source_branch:
'master'
,
expect
(
UpdateHeadPipelineForMergeRequestWorker
)
target_branch:
'feature'
,
.
to
receive
(
:perform_async
).
with
(
merge_request_1
.
id
)
source_project:
project
)
expect
(
UpdateHeadPipelineForMergeRequestWorker
)
.
to
receive
(
:perform_async
).
with
(
merge_request_2
.
id
)
head_pipeline
=
execute_service
.
payload
expect
(
head_pipeline
).
to
be_skipped
execute_service
(
ref:
'feature'
,
after:
nil
)
expect
(
head_pipeline
).
to
be_persisted
expect
(
merge_request
.
reload
.
head_pipeline
).
to
eq
head_pipeline
end
end
end
end
end
end
...
@@ -1655,7 +1561,7 @@ RSpec.describe Ci::CreatePipelineService do
...
@@ -1655,7 +1561,7 @@ RSpec.describe Ci::CreatePipelineService do
expect
(
pipeline
.
target_sha
).
to
be_nil
expect
(
pipeline
.
target_sha
).
to
be_nil
end
end
it
'schedules update for the head pipeline of the merge request'
do
it
'schedules update for the head pipeline of the merge request'
,
:sidekiq_inline
do
expect
(
UpdateHeadPipelineForMergeRequestWorker
)
expect
(
UpdateHeadPipelineForMergeRequestWorker
)
.
to
receive
(
:perform_async
).
with
(
merge_request
.
id
)
.
to
receive
(
:perform_async
).
with
(
merge_request
.
id
)
...
...
spec/workers/merge_requests/update_head_pipeline_worker_spec.rb
0 → 100644
View file @
80863e73
# frozen_string_literal: true
require
'spec_helper'
RSpec
.
describe
MergeRequests
::
UpdateHeadPipelineWorker
do
include
ProjectForksHelper
let_it_be
(
:project
)
{
create
(
:project
,
:repository
)
}
let
(
:ref
)
{
'master'
}
let
(
:pipeline
)
{
create
(
:ci_pipeline
,
project:
project
,
ref:
ref
)
}
let
(
:event
)
{
Ci
::
PipelineCreatedEvent
.
new
(
data:
{
pipeline_id:
pipeline
.
id
})
}
subject
{
consume_event
(
event
)
}
def
consume_event
(
event
)
described_class
.
new
.
perform
(
event
.
class
.
name
,
event
.
data
)
end
context
'when merge requests already exist for this source branch'
,
:sidekiq_inline
do
let
(
:merge_request_1
)
do
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"master"
,
source_project:
project
)
end
let
(
:merge_request_2
)
do
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"v1.1.0"
,
source_project:
project
)
end
context
'when related merge request is already merged'
do
let!
(
:merged_merge_request
)
do
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
"branch_2"
,
source_project:
project
,
state:
'merged'
)
end
it
'does not schedule update head pipeline job'
do
expect
(
UpdateHeadPipelineForMergeRequestWorker
).
not_to
receive
(
:perform_async
).
with
(
merged_merge_request
.
id
)
subject
end
end
context
'when the head pipeline sha equals merge request sha'
do
let
(
:ref
)
{
'feature'
}
before
do
pipeline
.
update!
(
sha:
project
.
repository
.
commit
(
ref
).
id
)
end
it
'updates head pipeline of each merge request'
do
merge_request_1
merge_request_2
subject
expect
(
merge_request_1
.
reload
.
head_pipeline
).
to
eq
(
pipeline
)
expect
(
merge_request_2
.
reload
.
head_pipeline
).
to
eq
(
pipeline
)
end
end
context
'when the head pipeline sha does not equal merge request sha'
do
let
(
:ref
)
{
'feature'
}
it
'does not update the head piepeline of MRs'
do
merge_request_1
merge_request_2
subject
expect
(
merge_request_1
.
reload
.
head_pipeline
).
not_to
eq
(
pipeline
)
expect
(
merge_request_2
.
reload
.
head_pipeline
).
not_to
eq
(
pipeline
)
end
end
context
'when there is no pipeline for source branch'
do
it
"does not update merge request head pipeline"
do
merge_request
=
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"branch_1"
,
source_project:
project
)
subject
expect
(
merge_request
.
reload
.
head_pipeline
).
not_to
eq
(
pipeline
)
end
end
context
'when merge request target project is different from source project'
do
let
(
:project
)
{
fork_project
(
target_project
,
nil
,
repository:
true
)
}
let
(
:target_project
)
{
create
(
:project
,
:repository
)
}
let
(
:user
)
{
create
(
:user
)
}
let
(
:ref
)
{
'feature'
}
before
do
project
.
add_developer
(
user
)
pipeline
.
update!
(
sha:
project
.
repository
.
commit
(
ref
).
id
)
end
it
'updates head pipeline for merge request'
do
merge_request
=
create
(
:merge_request
,
source_branch:
'feature'
,
target_branch:
"master"
,
source_project:
project
,
target_project:
target_project
)
subject
expect
(
merge_request
.
reload
.
head_pipeline
).
to
eq
(
pipeline
)
end
end
context
'when the pipeline is not the latest for the branch'
do
it
'does not update merge request head pipeline'
do
merge_request
=
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
"branch_1"
,
source_project:
project
)
create
(
:ci_pipeline
,
project:
pipeline
.
project
,
ref:
pipeline
.
ref
)
subject
expect
(
merge_request
.
reload
.
head_pipeline
).
to
be_nil
end
end
context
'when pipeline has errors'
do
before
do
pipeline
.
update!
(
yaml_errors:
'some errors'
,
status: :failed
)
end
it
'updates merge request head pipeline reference'
do
merge_request
=
create
(
:merge_request
,
source_branch:
'master'
,
target_branch:
'feature'
,
source_project:
project
)
subject
expect
(
merge_request
.
reload
.
head_pipeline
).
to
eq
(
pipeline
)
end
end
end
end
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment