pact_broker 2.75.0 → 2.78.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (193) hide show
  1. checksums.yaml +4 -4
  2. data/.codeclimate.yml +12 -0
  3. data/.github/workflows/test-ruby-3.yml +19 -0
  4. data/.github/workflows/test.yml +19 -4
  5. data/.gitignore +3 -1
  6. data/CHANGELOG.md +48 -0
  7. data/DEVELOPER_SETUP.md +62 -3
  8. data/Dockerfile +1 -0
  9. data/ISSUES.md +13 -5
  10. data/README.md +1 -1
  11. data/config.ru +1 -0
  12. data/db/ddl_statements/head_pact_tags.rb +24 -1
  13. data/db/ddl_statements/latest_tagged_pact_consumer_version_orders.rb +11 -0
  14. data/db/ddl_statements/latest_tagged_pact_publications.rb +6 -0
  15. data/db/ddl_statements/latest_verification_ids_for_consumer_version_tags.rb +13 -0
  16. data/db/migrations/20210117_add_branch_to_version.rb +9 -0
  17. data/db/migrations/20210202_add_created_at_to_head_pact_tags.rb +14 -0
  18. data/db/migrations/20210205_add_pacticipant_id_to_tag.rb +17 -0
  19. data/db/migrations/20210206_add_index_to_tags_and_versions.rb +27 -0
  20. data/db/migrations/20210207_optimise_latest_verification_ids_for_consumer_version_tags.rb +13 -0
  21. data/db/migrations/20210208_optimise_latest_tagged_pact_cv_orders.rb +13 -0
  22. data/db/migrations/20210210_create_environments_table.rb +16 -0
  23. data/docker-compose-issue-repro-with-pact-broker-docker-image.yml +44 -0
  24. data/lib/pact_broker/api.rb +7 -2
  25. data/lib/pact_broker/api/contracts/dry_validation_predicates.rb +8 -0
  26. data/lib/pact_broker/api/contracts/environment_schema.rb +49 -0
  27. data/lib/pact_broker/api/decorators/base_decorator.rb +11 -0
  28. data/lib/pact_broker/api/decorators/dashboard_decorator.rb +5 -1
  29. data/lib/pact_broker/api/decorators/environment_decorator.rb +30 -0
  30. data/lib/pact_broker/api/decorators/environments_decorator.rb +21 -0
  31. data/lib/pact_broker/api/decorators/matrix_decorator.rb +8 -2
  32. data/lib/pact_broker/api/decorators/reason_decorator.rb +2 -2
  33. data/lib/pact_broker/api/decorators/verifiable_pacts_query_decorator.rb +2 -0
  34. data/lib/pact_broker/api/decorators/version_decorator.rb +15 -2
  35. data/lib/pact_broker/api/pact_broker_urls.rb +8 -0
  36. data/lib/pact_broker/api/paths.rb +5 -0
  37. data/lib/pact_broker/api/resources/default_base_resource.rb +15 -2
  38. data/lib/pact_broker/api/resources/environment.rb +76 -0
  39. data/lib/pact_broker/api/resources/environments.rb +75 -0
  40. data/lib/pact_broker/api/resources/index.rb +20 -0
  41. data/lib/pact_broker/api/resources/latest_version.rb +27 -0
  42. data/lib/pact_broker/api/resources/provider_pacts_for_verification.rb +1 -0
  43. data/lib/pact_broker/api/resources/verifications.rb +5 -2
  44. data/lib/pact_broker/api/resources/version.rb +15 -9
  45. data/lib/pact_broker/api/resources/webhook_execution.rb +1 -1
  46. data/lib/pact_broker/app.rb +3 -0
  47. data/lib/pact_broker/certificates/certificate.rb +1 -1
  48. data/lib/pact_broker/config/setting.rb +1 -1
  49. data/lib/pact_broker/config/space_delimited_integer_list.rb +25 -0
  50. data/lib/pact_broker/configuration.rb +18 -1
  51. data/lib/pact_broker/db/data_migrations/helpers.rb +4 -0
  52. data/lib/pact_broker/db/data_migrations/set_extra_columns_for_tags.rb +29 -0
  53. data/lib/pact_broker/db/migrate_data.rb +1 -0
  54. data/lib/pact_broker/db/seed_example_data.rb +13 -13
  55. data/lib/pact_broker/deployments/environment.rb +15 -0
  56. data/lib/pact_broker/deployments/environment_service.rb +39 -0
  57. data/lib/pact_broker/doc/controllers/app.rb +1 -1
  58. data/lib/pact_broker/doc/views/index/environment.markdown +37 -0
  59. data/lib/pact_broker/doc/views/index/environments.markdown +53 -0
  60. data/lib/pact_broker/doc/views/index/latest-pact-versions.markdown +1 -1
  61. data/lib/pact_broker/doc/views/index/pacticipant-version-tag.markdown +1 -0
  62. data/lib/pact_broker/doc/views/index/pacticipant-version.markdown +13 -0
  63. data/lib/pact_broker/domain/index_item.rb +18 -4
  64. data/lib/pact_broker/domain/pacticipant.rb +9 -5
  65. data/lib/pact_broker/domain/tag.rb +131 -71
  66. data/lib/pact_broker/domain/verification.rb +3 -2
  67. data/lib/pact_broker/domain/version.rb +58 -23
  68. data/lib/pact_broker/domain/webhook.rb +12 -9
  69. data/lib/pact_broker/hash_refinements.rb +4 -0
  70. data/lib/pact_broker/index/service.rb +55 -49
  71. data/lib/pact_broker/locale/en.yml +3 -1
  72. data/lib/pact_broker/matrix/quick_row.rb +8 -0
  73. data/lib/pact_broker/metrics/service.rb +1 -1
  74. data/lib/pact_broker/pacts/eager_loaders.rb +52 -0
  75. data/lib/pact_broker/pacts/latest_pact_publication_id_for_consumer_version.rb +18 -13
  76. data/lib/pact_broker/pacts/lazy_loaders.rb +14 -0
  77. data/lib/pact_broker/pacts/pact_publication.rb +38 -84
  78. data/lib/pact_broker/pacts/pact_publication_dataset_module.rb +297 -0
  79. data/lib/pact_broker/pacts/pact_version.rb +1 -2
  80. data/lib/pact_broker/pacts/pacts_for_verification_repository.rb +286 -0
  81. data/lib/pact_broker/pacts/repository.rb +5 -231
  82. data/lib/pact_broker/pacts/selected_pact.rb +4 -0
  83. data/lib/pact_broker/pacts/selector.rb +56 -1
  84. data/lib/pact_broker/pacts/selectors.rb +16 -0
  85. data/lib/pact_broker/pacts/service.rb +6 -8
  86. data/lib/pact_broker/pacts/squash_pacts_for_verification.rb +1 -4
  87. data/lib/pact_broker/pacts/verifiable_pact.rb +45 -2
  88. data/lib/pact_broker/pacts/verifiable_pact_messages.rb +56 -16
  89. data/lib/pact_broker/repositories/helpers.rb +4 -0
  90. data/lib/pact_broker/services.rb +9 -0
  91. data/lib/pact_broker/tags/eager_loaders.rb +47 -0
  92. data/lib/pact_broker/tags/repository.rb +3 -1
  93. data/lib/pact_broker/tags/service.rb +0 -3
  94. data/lib/pact_broker/tags/tag_with_latest_flag.rb +1 -0
  95. data/lib/pact_broker/test/http_test_data_builder.rb +101 -35
  96. data/lib/pact_broker/test/test_data_builder.rb +50 -3
  97. data/lib/pact_broker/ui/app.rb +6 -0
  98. data/lib/pact_broker/ui/controllers/base_controller.rb +5 -1
  99. data/lib/pact_broker/ui/controllers/pacts.rb +18 -0
  100. data/lib/pact_broker/ui/view_models/index_item.rb +9 -0
  101. data/lib/pact_broker/ui/view_models/matrix_line.rb +36 -0
  102. data/lib/pact_broker/ui/views/index/show-with-tags.haml +8 -0
  103. data/lib/pact_broker/ui/views/matrix/show.haml +10 -0
  104. data/lib/pact_broker/verifications/latest_verification_id_for_pact_version_and_provider_version.rb +7 -5
  105. data/lib/pact_broker/verifications/service.rb +2 -1
  106. data/lib/pact_broker/version.rb +1 -1
  107. data/lib/pact_broker/versions/eager_loaders.rb +71 -0
  108. data/lib/pact_broker/versions/lazy_loaders.rb +13 -0
  109. data/lib/pact_broker/versions/repository.rb +22 -2
  110. data/lib/pact_broker/versions/service.rb +5 -1
  111. data/lib/pact_broker/webhooks/execution.rb +3 -2
  112. data/lib/pact_broker/webhooks/latest_triggered_webhook.rb +2 -0
  113. data/lib/pact_broker/webhooks/pact_and_verification_parameters.rb +10 -3
  114. data/lib/pact_broker/webhooks/service.rb +8 -7
  115. data/lib/pact_broker/webhooks/trigger_service.rb +56 -23
  116. data/lib/pact_broker/webhooks/triggered_webhook.rb +14 -5
  117. data/lib/pact_broker/webhooks/webhook.rb +1 -1
  118. data/lib/pact_broker/webhooks/webhook_event.rb +1 -1
  119. data/lib/pact_broker/webhooks/webhook_execution_result.rb +4 -1
  120. data/lib/pact_broker/webhooks/webhook_request_logger.rb +5 -1
  121. data/lib/rack/pact_broker/set_base_url.rb +22 -0
  122. data/lib/sequel/plugins/upsert.rb +18 -4
  123. data/public/stylesheets/index.css +22 -1
  124. data/public/stylesheets/matrix.css +0 -21
  125. data/regression/can_i_deploy_spec.rb +5 -4
  126. data/regression/index_spec.rb +26 -0
  127. data/regression/regression_helper.rb +29 -3
  128. data/regression/script/clear.sh +3 -0
  129. data/regression/script/run.sh +3 -0
  130. data/script/demonstrate-version-branches.rb +33 -0
  131. data/script/pry.rb +2 -2
  132. data/script/reproduce-issue-starting-up.rb +13 -23
  133. data/script/reproduce-issue.rb +18 -14
  134. data/script/trigger-release.sh +1 -1
  135. data/script/webhook-server.ru +5 -5
  136. data/spec/features/create_environment_spec.rb +47 -0
  137. data/spec/features/create_tag_spec.rb +32 -0
  138. data/spec/features/create_version_spec.rb +70 -0
  139. data/spec/features/delete_environment_spec.rb +16 -0
  140. data/spec/features/end_deployment_spec.rb +29 -0
  141. data/spec/features/get_environment_spec.rb +19 -0
  142. data/spec/features/get_environments_spec.rb +20 -0
  143. data/spec/features/record_deployment_spec.rb +28 -0
  144. data/spec/features/update_environment_spec.rb +44 -0
  145. data/spec/fixtures/approvals/modifiable_resources.approved.json +6 -0
  146. data/spec/fixtures/dashboard.json +4 -2
  147. data/spec/integration/ui/index_spec.rb +4 -4
  148. data/spec/lib/pact_broker/api/contracts/environment_schema_spec.rb +83 -0
  149. data/spec/lib/pact_broker/api/decorators/dashboard_decorator_spec.rb +4 -2
  150. data/spec/lib/pact_broker/api/decorators/matrix_decorator_spec.rb +11 -6
  151. data/spec/lib/pact_broker/api/decorators/reason_decorator_spec.rb +6 -6
  152. data/spec/lib/pact_broker/api/decorators/verifiable_pacts_query_decorator_spec.rb +6 -0
  153. data/spec/lib/pact_broker/api/decorators/version_decorator_spec.rb +18 -0
  154. data/spec/lib/pact_broker/api/resources/default_base_resource_approval_spec.rb +1 -1
  155. data/spec/lib/pact_broker/api/resources/default_base_resource_spec.rb +1 -12
  156. data/spec/lib/pact_broker/api/resources/provider_pacts_for_verification_spec.rb +4 -0
  157. data/spec/lib/pact_broker/api/resources/verifications_spec.rb +2 -5
  158. data/spec/lib/pact_broker/api/resources/webhook_execution_result_spec.rb +56 -0
  159. data/spec/lib/pact_broker/api/resources/webhook_execution_spec.rb +3 -2
  160. data/spec/lib/pact_broker/config/space_delimited_integer_list_spec.rb +47 -0
  161. data/spec/lib/pact_broker/configuration_spec.rb +12 -0
  162. data/spec/lib/pact_broker/doc/controllers/app_spec.rb +3 -5
  163. data/spec/lib/pact_broker/domain/tag_spec.rb +101 -27
  164. data/spec/lib/pact_broker/domain/version_spec.rb +103 -15
  165. data/spec/lib/pact_broker/domain/webhook_spec.rb +7 -7
  166. data/spec/lib/pact_broker/index/service_spec.rb +89 -15
  167. data/spec/lib/pact_broker/pacts/pact_publication_dataset_module_spec.rb +400 -0
  168. data/spec/lib/pact_broker/pacts/pact_publication_spec.rb +434 -14
  169. data/spec/lib/pact_broker/pacts/repository_find_for_verification_fallback_spec.rb +1 -1
  170. data/spec/lib/pact_broker/pacts/repository_find_for_verification_spec.rb +1 -1
  171. data/spec/lib/pact_broker/pacts/repository_find_wip_pact_versions_for_provider_branch_spec.rb +224 -0
  172. data/spec/lib/pact_broker/pacts/repository_find_wip_pact_versions_for_provider_spec.rb +46 -7
  173. data/spec/lib/pact_broker/pacts/selector_spec.rb +3 -2
  174. data/spec/lib/pact_broker/pacts/service_find_for_verification_spec.rb +2 -3
  175. data/spec/lib/pact_broker/pacts/service_spec.rb +9 -7
  176. data/spec/lib/pact_broker/pacts/verifiable_pact_messages_spec.rb +57 -10
  177. data/spec/lib/pact_broker/tags/repository_spec.rb +2 -0
  178. data/spec/lib/pact_broker/verifications/service_spec.rb +4 -1
  179. data/spec/lib/pact_broker/versions/repository_spec.rb +54 -0
  180. data/spec/lib/pact_broker/webhooks/render_spec.rb +6 -5
  181. data/spec/lib/pact_broker/webhooks/service_spec.rb +13 -9
  182. data/spec/lib/pact_broker/webhooks/trigger_service_spec.rb +60 -18
  183. data/spec/lib/pact_broker/webhooks/webhook_request_logger_spec.rb +8 -0
  184. data/spec/lib/sequel/plugins/upsert_spec.rb +31 -3
  185. data/spec/service_consumers/hal_relation_proxy_app.rb +3 -1
  186. data/spec/service_consumers/provider_states_for_pact_broker_client.rb +16 -0
  187. data/spec/spec_helper.rb +17 -5
  188. data/spec/support/approvals.rb +24 -0
  189. data/spec/support/shared_examples_for_responses.rb +11 -0
  190. data/tasks/db.rake +1 -0
  191. data/tasks/rspec.rake +1 -1
  192. metadata +69 -4
  193. data/docker-compose-issue-repro.yml +0 -22
@@ -1,5 +1,6 @@
1
1
  require 'pact_broker/ui/controllers/index'
2
2
  require 'pact_broker/ui/controllers/groups'
3
+ require 'pact_broker/ui/controllers/pacts'
3
4
  require 'pact_broker/ui/controllers/matrix'
4
5
  require 'pact_broker/ui/controllers/can_i_deploy'
5
6
  require 'pact_broker/ui/controllers/error_test'
@@ -47,6 +48,11 @@ module PactBroker
47
48
  run PactBroker::UI::Controllers::CanIDeploy
48
49
  end
49
50
 
51
+ map "/pacts/" do
52
+ use PathInfoFixer
53
+ run PactBroker::UI::Controllers::Pacts
54
+ end
55
+
50
56
  map "/test/error" do
51
57
  use PathInfoFixer
52
58
  run PactBroker::UI::Controllers::ErrorTest
@@ -12,7 +12,11 @@ module PactBroker
12
12
  set :dump_errors, false # The padrino logger logs these for us. If this is enabled we get duplicate logging.
13
13
 
14
14
  def base_url
15
- PactBroker.configuration.base_url || ''
15
+ # Using the X-Forwarded headers in the UI can leave the app vulnerable
16
+ # https://www.acunetix.com/blog/articles/automated-detection-of-host-header-attacks/
17
+ # Either use the explicitly configured base url or an empty string,
18
+ # rather than request.base_url, which uses the X-Forwarded headers.
19
+ env["pactbroker.base_url"] || ''
16
20
  end
17
21
  end
18
22
  end
@@ -0,0 +1,18 @@
1
+ require 'pact_broker/ui/controllers/base_controller'
2
+
3
+ module PactBroker
4
+ module UI
5
+ module Controllers
6
+ class Pacts < Base
7
+ include PactBroker::Services
8
+
9
+ get "/provider/:provider_name/consumer/:consumer_name/pact-version/:pact_version/verification-results/:number" do
10
+ url = URI.parse("#{env["pactbroker.base_url"]}/hal-browser/browser.html")
11
+ url.fragment = "#{env["pactbroker.base_url"]}#{env["SCRIPT_NAME"]}#{env["PATH_INFO"]}"
12
+ response.headers["Location"] = url.to_s
13
+ response.status = 302
14
+ end
15
+ end
16
+ end
17
+ end
18
+ end
@@ -3,11 +3,16 @@ require 'pact_broker/ui/helpers/url_helper'
3
3
  require 'pact_broker/date_helper'
4
4
  require 'pact_broker/versions/abbreviate_number'
5
5
  require 'pact_broker/configuration'
6
+ require 'forwardable'
6
7
 
7
8
  module PactBroker
8
9
  module UI
9
10
  module ViewDomain
10
11
  class IndexItem
12
+ extend Forwardable
13
+
14
+ delegate [:consumer_version_branch, :provider_version_branch, :latest_for_branch?] => :relationship
15
+
11
16
 
12
17
  include PactBroker::Api::PactBrokerUrls
13
18
 
@@ -186,6 +191,10 @@ module PactBroker
186
191
  def base_url
187
192
  @options[:base_url]
188
193
  end
194
+
195
+ private
196
+
197
+ attr_reader :relationship
189
198
  end
190
199
  end
191
200
  end
@@ -4,6 +4,7 @@ require 'pact_broker/date_helper'
4
4
  require 'pact_broker/ui/view_models/matrix_tag'
5
5
  require 'pact_broker/versions/abbreviate_number'
6
6
  require 'pact_broker/messages'
7
+ require 'forwardable'
7
8
 
8
9
  module PactBroker
9
10
  module UI
@@ -11,6 +12,9 @@ module PactBroker
11
12
  class MatrixLine
12
13
  include PactBroker::Api::PactBrokerUrls
13
14
  include PactBroker::Messages
15
+ extend Forwardable
16
+
17
+ delegate [:consumer_version_branch, :provider_version_branch] => :line
14
18
 
15
19
  def initialize line, options = {}
16
20
  @line = line
@@ -97,27 +101,47 @@ module PactBroker
97
101
  end
98
102
  end
99
103
 
104
+ def consumer_version_branch_tooltip
105
+ branch_tooltip(consumer_name, consumer_version_branch, consumer_version_latest_for_branch?)
106
+ end
107
+
108
+ def consumer_version_latest_for_branch?
109
+ @line.consumer_version.latest_for_branch?
110
+ end
111
+
112
+ def provider_version_branch_tooltip
113
+ branch_tooltip(provider_name, provider_version_branch, provider_version_latest_for_branch?)
114
+ end
115
+
116
+ def provider_version_latest_for_branch?
117
+ @line.provider_version.latest_for_branch?
118
+ end
119
+
100
120
  def latest_consumer_version_tags
101
121
  @line.consumer_version_tags
102
122
  .select(&:latest)
123
+ .sort_by(&:created_at)
103
124
  .collect{ | tag | MatrixTag.new(tag.to_hash.merge(pacticipant_name: consumer_name, version_number: consumer_version_number)) }
104
125
  end
105
126
 
106
127
  def other_consumer_version_tags
107
128
  @line.consumer_version_tags
108
129
  .reject(&:latest)
130
+ .sort_by(&:created_at)
109
131
  .collect{ | tag | MatrixTag.new(tag.to_hash.merge(pacticipant_name: consumer_name, version_number: consumer_version_number)) }
110
132
  end
111
133
 
112
134
  def latest_provider_version_tags
113
135
  @line.provider_version_tags
114
136
  .select(&:latest)
137
+ .sort_by(&:created_at)
115
138
  .collect{ | tag | MatrixTag.new(tag.to_hash.merge(pacticipant_name: provider_name, version_number: provider_version_number)) }
116
139
  end
117
140
 
118
141
  def other_provider_version_tags
119
142
  @line.provider_version_tags
120
143
  .reject(&:latest)
144
+ .sort_by(&:created_at)
121
145
  .collect{ | tag | MatrixTag.new(tag.to_hash.merge(pacticipant_name: provider_name, version_number: provider_version_number)) }
122
146
  end
123
147
 
@@ -187,6 +211,18 @@ module PactBroker
187
211
  def base_url
188
212
  @options[:base_url]
189
213
  end
214
+
215
+ private
216
+
217
+ attr_reader :line
218
+
219
+ def branch_tooltip(pacticipant_name, branch, latest)
220
+ if latest
221
+ "This is the latest version of #{pacticipant_name} from branch \"#{branch}\"."
222
+ else
223
+ "A more recent version of #{pacticipant_name} from branch \"#{branch}\" exists."
224
+ end
225
+ end
190
226
  end
191
227
  end
192
228
  end
@@ -47,6 +47,10 @@
47
47
  - if index_item.latest?
48
48
  .tag.badge.badge-success
49
49
  latest
50
+ - if index_item.consumer_version_branch
51
+ - branch_class = index_item.latest_for_branch? ? "tag badge badge-dark" : "tag badge badge-secondary"
52
+ %div{"class": branch_class}
53
+ = "[" + index_item.consumer_version_branch + "]"
50
54
  - index_item.consumer_version_latest_tag_names.each do | tag_name |
51
55
  .tag.badge.badge-primary
52
56
  = tag_name
@@ -64,6 +68,10 @@
64
68
  - if index_item.provider_version_number
65
69
  %button.clippy.invisible{ title: "Copy to clipboard" }
66
70
  %span.copy-icon
71
+ - if index_item.provider_version_branch
72
+ - branch_class = "tag badge badge-dark"
73
+ %div{"class": branch_class}
74
+ = "[" + index_item.provider_version_branch + "]"
67
75
  - index_item.provider_version_latest_tag_names.each do | tag_name |
68
76
  .tag.badge.badge-primary
69
77
  = tag_name
@@ -118,6 +118,11 @@
118
118
  - if line.display_consumer_version_number
119
119
  %button.clippy.invisible{ title: "Copy to clipboard" }
120
120
  %span.copy-icon
121
+ - if line.consumer_version_branch
122
+ .tag-parent{"title": line.consumer_version_branch_tooltip, "data-toggle": "tooltip", "data-placement": "right"}
123
+ - branch_class = line.consumer_version_latest_for_branch? ? "tag badge badge-dark" : "tag badge badge-secondary"
124
+ %div{"class": branch_class}
125
+ = "[" + line.consumer_version_branch + "]"
121
126
  - line.latest_consumer_version_tags.each do | tag |
122
127
  .tag-parent{"title": tag.tooltip, "data-toggle": "tooltip", "data-placement": "right"}
123
128
  %a{href: tag.url}
@@ -145,6 +150,11 @@
145
150
  - if line.display_provider_version_number
146
151
  %button.clippy.invisible{ title: "Copy to clipboard" }
147
152
  %span.copy-icon
153
+ - if line.provider_version_branch
154
+ .tag-parent{"title": line.provider_version_branch_tooltip, "data-toggle": "tooltip", "data-placement": "right"}
155
+ - branch_class = line.provider_version_latest_for_branch? ? "tag badge badge-dark" : "tag badge badge-secondary"
156
+ %div{"class": branch_class}
157
+ = "[" + line.provider_version_branch + "]"
148
158
  - line.latest_provider_version_tags.each do | tag |
149
159
  .tag-parent{"title": tag.tooltip, "data-toggle": "tooltip", "data-placement": "right"}
150
160
  %a{href: tag.url}
@@ -16,12 +16,14 @@ module PactBroker
16
16
  end
17
17
 
18
18
  # Table: latest_verification_id_for_pact_version_and_provider_version
19
+ # Primary Key: (pact_version_id, provider_version_id)
19
20
  # Columns:
20
- # consumer_id | integer | NOT NULL
21
- # pact_version_id | integer | NOT NULL
22
- # provider_id | integer | NOT NULL
23
- # provider_version_id | integer | NOT NULL
24
- # verification_id | integer | NOT NULL
21
+ # consumer_id | integer | NOT NULL
22
+ # pact_version_id | integer | NOT NULL
23
+ # provider_id | integer | NOT NULL
24
+ # provider_version_id | integer | NOT NULL
25
+ # verification_id | integer | NOT NULL
26
+ # created_at | timestamp without time zone |
25
27
  # Indexes:
26
28
  # latest_v_id_for_pv_and_pv_pv_id_pv_id_unq | UNIQUE btree (pact_version_id, provider_version_id)
27
29
  # latest_v_id_for_pv_and_pv_v_id_unq | UNIQUE btree (verification_id)
@@ -20,7 +20,7 @@ module PactBroker
20
20
  verification_repository.next_number
21
21
  end
22
22
 
23
- def create next_verification_number, params, pact, webhook_options
23
+ def create next_verification_number, params, pact, event_context, webhook_options
24
24
  logger.info "Creating verification #{next_verification_number} for pact_id=#{pact.id}", payload: params.reject{ |k,_| k == "testResults"}
25
25
  verification = PactBroker::Domain::Verification.new
26
26
  provider_version_number = params.fetch('providerApplicationVersion')
@@ -35,6 +35,7 @@ module PactBroker
35
35
  webhook_trigger_service.trigger_webhooks_for_verification_results_publication(
36
36
  pact,
37
37
  verification,
38
+ event_context.merge(provider_version_tags: verification.provider_version_tag_names),
38
39
  webhook_options.deep_merge(webhook_execution_configuration: execution_configuration)
39
40
  )
40
41
  verification
@@ -1,3 +1,3 @@
1
1
  module PactBroker
2
- VERSION = '2.75.0'
2
+ VERSION = '2.78.0'
3
3
  end
@@ -0,0 +1,71 @@
1
+ module PactBroker
2
+ module Versions
3
+ module EagerLoaders
4
+ class LatestVersionForBranch
5
+ def self.call(eo, **other)
6
+ initialize_association(eo[:rows])
7
+ populate_associations(eo[:rows])
8
+ end
9
+
10
+ def self.initialize_association(versions)
11
+ versions.each{|version| version.associations[:latest_version_for_branch] = nil }
12
+ end
13
+
14
+ def self.populate_associations(versions)
15
+ group_by_pacticipant_id(versions).each do | pacticipant, versions |
16
+ populate_associations_by_pacticipant(pacticipant, versions)
17
+ end
18
+ end
19
+
20
+ def self.group_by_pacticipant_id(versions)
21
+ versions.to_a.group_by(&:pacticipant_id)
22
+ end
23
+
24
+ def self.populate_associations_by_pacticipant(pacticipant_id, versions)
25
+ latest_versions_for_branches = latest_versions_for_pacticipant_branches(
26
+ pacticipant_id,
27
+ versions.collect(&:branch).uniq.compact,
28
+ versions.first.class
29
+ )
30
+ self.populate_versions_with_branches(versions, latest_versions_for_branches)
31
+ end
32
+
33
+ def self.populate_versions_with_branches(versions, latest_versions_for_branches)
34
+ versions.select(&:branch).each do | version |
35
+ version.associations[:latest_version_for_branch] = latest_versions_for_branches[[version.pacticipant_id, version.branch]]
36
+ end
37
+ end
38
+
39
+ def self.latest_versions_for_pacticipant_branches(pacticipant_id, branches, version_class)
40
+ version_class.latest_versions_for_pacticipant_branches(pacticipant_id, branches).each_with_object({}) do | row, hash |
41
+ hash[[row.pacticipant_id, row.branch]] = row
42
+ end
43
+ end
44
+ end
45
+
46
+ class LatestVersionForPacticipant
47
+ def self.call(eo, **other)
48
+ populate_associations(eo[:rows])
49
+ end
50
+
51
+ def self.populate_associations(versions)
52
+ group_by_pacticipant(versions).each do | pacticipant, versions |
53
+ populate_associations_by_pacticipant(pacticipant, versions)
54
+ end
55
+ end
56
+
57
+ def self.group_by_pacticipant(versions)
58
+ versions.to_a.group_by(&:pacticipant)
59
+ end
60
+
61
+ def self.populate_associations_by_pacticipant(pacticipant, versions)
62
+ latest_version = versions.first.class.latest_version_for_pacticipant(pacticipant).single_record
63
+
64
+ versions.each do | version |
65
+ version.associations[:latest_version_for_pacticipant] = latest_version
66
+ end
67
+ end
68
+ end
69
+ end
70
+ end
71
+ end
@@ -0,0 +1,13 @@
1
+ module PactBroker
2
+ module Versions
3
+ module LazyLoaders
4
+ LATEST_VERSION_FOR_BRANCH = lambda {
5
+ self.class
6
+ .where(branch: branch, pacticipant_id: pacticipant_id)
7
+ .exclude(branch: nil)
8
+ .order(Sequel.desc(:order))
9
+ .limit(1)
10
+ }
11
+ end
12
+ end
13
+ end
@@ -9,6 +9,7 @@ module PactBroker
9
9
 
10
10
  include PactBroker::Logging
11
11
  include PactBroker::Repositories::Helpers
12
+ include PactBroker::Repositories
12
13
 
13
14
  def find_by_pacticipant_id_and_number pacticipant_id, number
14
15
  PactBroker::Domain::Version.where(number: number, pacticipant_id: pacticipant_id).single_record
@@ -57,7 +58,26 @@ module PactBroker
57
58
  updated_at: Sequel.datetime_class.now
58
59
  }
59
60
 
60
- PactBroker::Domain::Version.new(version_params).insert_ignore
61
+ PactBroker::Domain::Version.new(version_params).upsert
62
+ end
63
+
64
+ def create_or_overwrite(pacticipant, version_number, open_struct_version)
65
+ saved_version = PactBroker::Domain::Version.new(
66
+ number: version_number,
67
+ pacticipant: pacticipant,
68
+ build_url: open_struct_version.build_url,
69
+ branch: open_struct_version.branch
70
+ ).upsert
71
+
72
+ if open_struct_version.tags
73
+ tag_repository.delete_by_version_id(saved_version.id)
74
+ open_struct_version.tags.collect do | open_struct_tag |
75
+ tag_repository.create(version: saved_version, name: open_struct_tag.name)
76
+ end
77
+ saved_version.refresh
78
+ end
79
+
80
+ saved_version
61
81
  end
62
82
 
63
83
  def find_by_pacticipant_id_and_number_or_create pacticipant_id, number
@@ -79,7 +99,7 @@ module PactBroker
79
99
  version_ids_to_keep = (version_ids_with_pact_publications + version_ids_with_verifications).uniq
80
100
 
81
101
  PactBroker::Domain::Version
82
- .where(pacticipant_id: [consumer.id, provider.id])
102
+ .where(Sequel[:versions][:pacticipant_id] => [consumer.id, provider.id])
83
103
  .exclude(id: (version_ids_with_pact_publications + version_ids_with_verifications).uniq)
84
104
  .delete
85
105
  end
@@ -1,7 +1,6 @@
1
1
  require 'pact_broker/repositories'
2
2
 
3
3
  module PactBroker
4
-
5
4
  module Versions
6
5
  class Service
7
6
 
@@ -19,6 +18,11 @@ module PactBroker
19
18
  version_repository.find_by_pacticipant_name_and_latest_tag(pacticipant_name, tag)
20
19
  end
21
20
 
21
+ def self.create_or_overwrite(pacticipant_name, version_number, version)
22
+ pacticipant = pacticipant_repository.find_by_name_or_create(pacticipant_name)
23
+ version_repository.create_or_overwrite(pacticipant, version_number, version)
24
+ end
25
+
22
26
  def self.delete version
23
27
  tag_repository.delete_by_version_id version.id
24
28
  webhook_repository.delete_triggered_webhooks_by_version_id version.id
@@ -32,13 +32,14 @@ end
32
32
 
33
33
  # Table: webhook_executions
34
34
  # Columns:
35
- # id | integer | PRIMARY KEY DEFAULT nextval('webhook_executions_id_seq'::regclass)
35
+ # id | integer | PRIMARY KEY GENERATED BY DEFAULT AS IDENTITY
36
36
  # triggered_webhook_id | integer |
37
37
  # success | boolean | NOT NULL
38
38
  # logs | text |
39
39
  # created_at | timestamp without time zone | NOT NULL
40
40
  # Indexes:
41
- # webhook_executions_pkey | PRIMARY KEY btree (id)
41
+ # webhook_executions_pkey | PRIMARY KEY btree (id)
42
+ # webhook_executions_triggered_webhook_id_index | btree (triggered_webhook_id)
42
43
  # Foreign key constraints:
43
44
  # webhook_executions_consumer_id_fkey | (consumer_id) REFERENCES pacticipants(id)
44
45
  # webhook_executions_pact_publication_id_fkey | (pact_publication_id) REFERENCES pact_publications(id)
@@ -21,3 +21,5 @@ end
21
21
  # status | text |
22
22
  # created_at | timestamp without time zone |
23
23
  # updated_at | timestamp without time zone |
24
+ # verification_id | integer |
25
+ # event_name | text |
@@ -13,6 +13,7 @@ module PactBroker
13
13
  BITBUCKET_VERIFICATION_STATUS = 'pactbroker.bitbucketVerificationStatus'
14
14
  CONSUMER_LABELS = 'pactbroker.consumerLabels'
15
15
  PROVIDER_LABELS = 'pactbroker.providerLabels'
16
+ EVENT_NAME = 'pactbroker.eventName'
16
17
 
17
18
  ALL = [
18
19
  CONSUMER_NAME,
@@ -26,7 +27,8 @@ module PactBroker
26
27
  GITHUB_VERIFICATION_STATUS,
27
28
  BITBUCKET_VERIFICATION_STATUS,
28
29
  CONSUMER_LABELS,
29
- PROVIDER_LABELS
30
+ PROVIDER_LABELS,
31
+ EVENT_NAME
30
32
  ]
31
33
 
32
34
  def initialize(pact, trigger_verification, webhook_context)
@@ -49,7 +51,8 @@ module PactBroker
49
51
  GITHUB_VERIFICATION_STATUS => github_verification_status,
50
52
  BITBUCKET_VERIFICATION_STATUS => bitbucket_verification_status,
51
53
  CONSUMER_LABELS => pacticipant_labels(pact && pact.consumer),
52
- PROVIDER_LABELS => pacticipant_labels(pact && pact.provider)
54
+ PROVIDER_LABELS => pacticipant_labels(pact && pact.provider),
55
+ EVENT_NAME => event_name
53
56
  }
54
57
  end
55
58
 
@@ -116,6 +119,10 @@ module PactBroker
116
119
  def pacticipant_labels pacticipant
117
120
  pacticipant && pacticipant.labels ? pacticipant.labels.collect(&:name).join(", ") : ""
118
121
  end
122
+
123
+ def event_name
124
+ webhook_context.fetch(:event_name)
125
+ end
119
126
  end
120
127
  end
121
- end
128
+ end