-
-
Notifications
You must be signed in to change notification settings - Fork 178
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(matrix): optimise query for can-i-deploy, again
- Loading branch information
Showing
6 changed files
with
308 additions
and
20 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,255 @@ | ||
=begin | ||
The Matrix::Row is based on the matrix view which does a join of every pact/verification | ||
and then selects the relevant ones. | ||
The Matrix::QuickRow starts with the relevant rows, and builds the matrix query from that, | ||
making it much quicker. | ||
=end | ||
require 'sequel' | ||
require 'pact_broker/repositories/helpers' | ||
require 'pact_broker/logging' | ||
require 'pact_broker/pacts/pact_version' | ||
require 'pact_broker/domain/pacticipant' | ||
require 'pact_broker/domain/version' | ||
require 'pact_broker/domain/verification' | ||
require 'pact_broker/pacts/pact_publication' | ||
require 'pact_broker/tags/tag_with_latest_flag' | ||
|
||
module PactBroker | ||
module Matrix | ||
LV = :latest_verification_id_for_pact_version_and_provider_version | ||
LP = :latest_pact_publication_ids_for_consumer_versions | ||
|
||
CONSUMER_COLUMNS = [Sequel[:lp][:consumer_id], Sequel[:consumers][:name].as(:consumer_name), Sequel[:lp][:pact_publication_id], Sequel[:lp][:pact_version_id]] | ||
PROVIDER_COLUMNS = [Sequel[:lp][:provider_id], Sequel[:providers][:name].as(:provider_name), Sequel[:lv][:verification_id]] | ||
CONSUMER_VERSION_COLUMNS = [Sequel[:lp][:consumer_version_id], Sequel[:cv][:number].as(:consumer_version_number), Sequel[:cv][:order].as(:consumer_version_order)] | ||
PROVIDER_VERSION_COLUMNS = [Sequel[:lv][:provider_version_id], Sequel[:pv][:number].as(:provider_version_number), Sequel[:pv][:order].as(:provider_version_order)] | ||
ALL_COLUMNS = CONSUMER_COLUMNS + CONSUMER_VERSION_COLUMNS + PROVIDER_COLUMNS + PROVIDER_VERSION_COLUMNS | ||
|
||
LP_LV_JOIN = { Sequel[:lp][:pact_version_id] => Sequel[:lv][:pact_version_id] } | ||
CONSUMER_JOIN = { Sequel[:lp][:consumer_id] => Sequel[:consumers][:id] } | ||
PROVIDER_JOIN = { Sequel[:lp][:provider_id] => Sequel[:providers][:id] } | ||
CONSUMER_VERSION_JOIN = { Sequel[:lp][:consumer_version_id] => Sequel[:cv][:id] } | ||
PROVIDER_VERSION_JOIN = { Sequel[:lv][:provider_version_id] => Sequel[:pv][:id] } | ||
|
||
RAW_QUERY = Sequel::Model.db[Sequel.as(LP, :lp)] | ||
.select(*ALL_COLUMNS) | ||
.left_outer_join(LV, LP_LV_JOIN, { table_alias: :lv } ) | ||
.join(:pacticipants, CONSUMER_JOIN, { table_alias: :consumers }) | ||
.join(:pacticipants, PROVIDER_JOIN, { table_alias: :providers }) | ||
.join(:versions, CONSUMER_VERSION_JOIN, { table_alias: :cv }) | ||
.left_outer_join(:versions, PROVIDER_VERSION_JOIN, { table_alias: :pv } ) | ||
|
||
ALIASED_QUERY = Sequel.as(RAW_QUERY, :smart_rows) | ||
|
||
class QuickRow < Sequel::Model(ALIASED_QUERY) | ||
CONSUMER_ID = Sequel[:smart_rows][:consumer_id] | ||
PROVIDER_ID = Sequel[:smart_rows][:provider_id] | ||
CONSUMER_VERSION_ID = Sequel[:smart_rows][:consumer_version_id] | ||
PROVIDER_VERSION_ID = Sequel[:smart_rows][:provider_version_id] | ||
PACT_PUBLICATION_ID = Sequel[:smart_rows][:pact_publication_id] | ||
VERIFICATION_ID = Sequel[:smart_rows][:verification_id] | ||
|
||
associate(:many_to_one, :pact_publication, :class => "PactBroker::Pacts::PactPublication", :key => :pact_publication_id, :primary_key => :id) | ||
associate(:many_to_one, :provider, :class => "PactBroker::Domain::Pacticipant", :key => :provider_id, :primary_key => :id) | ||
associate(:many_to_one, :consumer, :class => "PactBroker::Domain::Pacticipant", :key => :consumer_id, :primary_key => :id) | ||
associate(:many_to_one, :consumer_version, :class => "PactBroker::Domain::Version", :key => :consumer_version_id, :primary_key => :id) | ||
associate(:many_to_one, :provider_version, :class => "PactBroker::Domain::Version", :key => :provider_version_id, :primary_key => :id) | ||
associate(:many_to_one, :pact_version, class: "PactBroker::Pacts::PactVersion", :key => :pact_version_id, :primary_key => :id) | ||
associate(:many_to_one, :verification, class: "PactBroker::Domain::Verification", :key => :verification_id, :primary_key => :id) | ||
associate(:one_to_many, :consumer_version_tags, :class => "PactBroker::Tags::TagWithLatestFlag", primary_key: :consumer_version_id, key: :version_id) | ||
associate(:one_to_many, :provider_version_tags, :class => "PactBroker::Tags::TagWithLatestFlag", primary_key: :provider_version_id, key: :version_id) | ||
|
||
dataset_module do | ||
include PactBroker::Repositories::Helpers | ||
include PactBroker::Logging | ||
|
||
def consumer_id consumer_id | ||
where(CONSUMER_ID => consumer_id) | ||
end | ||
|
||
def matching_selectors selectors | ||
if selectors.size == 1 | ||
where_consumer_or_provider_is(selectors.first) | ||
else | ||
where_consumer_and_provider_in(selectors) | ||
end | ||
end | ||
|
||
# find rows where (the consumer (and optional version) matches any of the selectors) | ||
# AND | ||
# the (provider (and optional version) matches any of the selectors OR the provider matches | ||
# and the verification is missing (and hence the provider version is null)) | ||
def where_consumer_and_provider_in selectors | ||
where{ | ||
Sequel.&( | ||
Sequel.|( | ||
*QueryHelper.consumer_and_maybe_consumer_version_match_any_selector(selectors) | ||
), | ||
Sequel.|( | ||
*QueryHelper.provider_and_maybe_provider_version_match_any_selector_or_verification_is_missing(selectors) | ||
), | ||
QueryHelper.either_consumer_or_provider_was_specified_in_query(selectors) | ||
) | ||
} | ||
end | ||
|
||
# Can't access other dataset_module methods from inside the Sequel `where{ ... }` block, so make a private class | ||
# with some helper methods | ||
class QueryHelper | ||
def self.consumer_and_maybe_consumer_version_match_any_selector(selectors) | ||
selectors.collect { |s| consumer_and_maybe_consumer_version_match_selector(s) } | ||
end | ||
|
||
def self.consumer_and_maybe_consumer_version_match_selector(s) | ||
if s[:pact_publication_ids] | ||
Sequel.&(PACT_PUBLICATION_ID => s[:pact_publication_ids]) | ||
elsif s[:pacticipant_version_id] | ||
Sequel.&(CONSUMER_ID => s[:pacticipant_id], CONSUMER_VERSION_ID => s[:pacticipant_version_id]) | ||
else | ||
Sequel.&(CONSUMER_ID => s[:pacticipant_id]) | ||
end | ||
end | ||
|
||
def self.provider_and_maybe_provider_version_match_selector(s) | ||
if s[:verification_ids] | ||
Sequel.&(VERIFICATION_ID => s[:verification_ids]) | ||
elsif s[:pacticipant_version_id] | ||
Sequel.&(PROVIDER_ID => s[:pacticipant_id], PROVIDER_VERSION_ID => s[:pacticipant_version_id]) | ||
else | ||
Sequel.&(PROVIDER_ID => s[:pacticipant_id]) | ||
end | ||
end | ||
|
||
# if the pact for a consumer version has never been verified, it exists in the matrix as a row | ||
# with a blank provider version id | ||
def self.provider_verification_is_missing_for_matching_selector(s) | ||
Sequel.&(PROVIDER_ID => s[:pacticipant_id], PROVIDER_VERSION_ID => nil) | ||
end | ||
|
||
def self.provider_and_maybe_provider_version_match_any_selector_or_verification_is_missing(selectors) | ||
selectors.collect { |s| | ||
provider_and_maybe_provider_version_match_selector(s) | ||
} + selectors.collect { |s| | ||
provider_verification_is_missing_for_matching_selector(s) | ||
} | ||
end | ||
|
||
# Some selecters are specified in the query, others are implied (when only one pacticipant is specified, | ||
# the integrations are automatically worked out, and the selectors for these are of type :implied ) | ||
# When there are 3 pacticipants that each have dependencies on each other (A->B, A->C, B->C), the query | ||
# to deploy C (implied A, implied B, specified C) was returning the A->B row because it matched the | ||
# implied selectors as well. | ||
# This extra filter makes sure that every row that is returned actually matches one of the specified | ||
# selectors. | ||
def self.either_consumer_or_provider_was_specified_in_query(selectors) | ||
specified_pacticipant_ids = selectors.select{ |s| s[:type] == :specified }.collect{ |s| s[:pacticipant_id] } | ||
Sequel.|({ CONSUMER_ID => specified_pacticipant_ids } , { PROVIDER_ID => specified_pacticipant_ids }) | ||
end | ||
end | ||
|
||
def where_consumer_or_provider_is s | ||
where{ | ||
Sequel.|( | ||
s[:pacticipant_version_id] ? Sequel.&(CONSUMER_ID => s[:pacticipant_id], CONSUMER_VERSION_ID => s[:pacticipant_version_id]) : Sequel.&(CONSUMER_ID => s[:pacticipant_id]), | ||
s[:pacticipant_version_id] ? Sequel.&(PROVIDER_ID => s[:pacticipant_id], PROVIDER_VERSION_ID => s[:pacticipant_version_id]) : Sequel.&(PROVIDER_ID => s[:pacticipant_id]) | ||
) | ||
} | ||
end | ||
|
||
def order_by_names_ascending_most_recent_first | ||
order( | ||
Sequel.asc(:consumer_name), | ||
Sequel.desc(:consumer_version_order), | ||
Sequel.asc(:provider_name), | ||
Sequel.desc(:provider_version_order), | ||
Sequel.desc(:verification_id)) | ||
end | ||
|
||
def eager_all_the_things | ||
eager(:consumer) | ||
.eager(:provider) | ||
.eager(:consumer_version) | ||
.eager(:provider_version) | ||
.eager(:verification) | ||
.eager(:pact_publication) | ||
.eager(:pact_version) | ||
end | ||
|
||
end | ||
|
||
def success | ||
verification&.success | ||
end | ||
|
||
def pact_version_sha | ||
pact_version.sha | ||
end | ||
|
||
def pact_revision_number | ||
pact_publication.revision_number | ||
end | ||
|
||
def verification_number | ||
verification&.number | ||
end | ||
|
||
def success | ||
verification&.success | ||
end | ||
|
||
def pact_created_at | ||
pact_publication.created_at | ||
end | ||
|
||
def verification_executed_at | ||
verification.execution_date | ||
end | ||
|
||
# Add logic for ignoring case | ||
def <=> other | ||
comparisons = [ | ||
compare_name_asc(consumer_name, other.consumer_name), | ||
compare_number_desc(consumer_version_order, other.consumer_version_order), | ||
compare_number_desc(pact_revision_number, other.pact_revision_number), | ||
compare_name_asc(provider_name, other.provider_name), | ||
compare_number_desc(provider_version_order, other.provider_version_order), | ||
compare_number_desc(verification_id, other.verification_id) | ||
] | ||
|
||
comparisons.find{|c| c != 0 } || 0 | ||
end | ||
|
||
def compare_name_asc name1, name2 | ||
name1 <=> name2 | ||
end | ||
|
||
def to_s | ||
"#{consumer_name} v#{consumer_version_number} #{provider_name} #{provider_version_number} #{success}" | ||
end | ||
|
||
def compare_number_desc number1, number2 | ||
if number1 && number2 | ||
number2 <=> number1 | ||
elsif number1 | ||
1 | ||
else | ||
-1 | ||
end | ||
end | ||
|
||
def eql?(obj) | ||
(obj.class == model) && (obj.values == values) | ||
end | ||
|
||
def pacticipant_names | ||
[consumer_name, provider_name] | ||
end | ||
|
||
def involves_pacticipant_with_name?(pacticipant_name) | ||
pacticipant_name.include?(pacticipant_name) | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
require 'pact_broker/matrix/quick_row' | ||
|
||
module PactBroker | ||
module Matrix | ||
describe QuickRow do | ||
before do | ||
td.create_pact_with_hierarchy("A", "1", "B") | ||
.create_verification(provider_version: '1', success: false) | ||
.create_verification(provider_version: '1', number: 2, success: true) | ||
.create_verification(provider_version: '2', number: 3, success: true) | ||
.create_provider("C") | ||
.create_pact | ||
.create_verification(provider_version: '1') | ||
.create_consumer_version("2") | ||
.create_pact | ||
.create_verification(provider_version: '3') | ||
.use_provider("B") | ||
.create_pact | ||
end | ||
|
||
it "behaves like a Row, except smarter" do | ||
a_id = QuickRow.db[:pacticipants].where(name: "A").select(:id).single_record[:id] | ||
rows = QuickRow.consumer_id(a_id).eager(:consumer).eager(:verification).all | ||
expect(rows.first.consumer).to be rows.last.consumer | ||
expect(rows.first.verification).to_not be nil | ||
expect(rows.first.consumer_name).to_not be nil | ||
expect(rows.first.provider_name).to_not be nil | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.