shipit-engine 0.30.0 → 0.34.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (410) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +19 -4
  3. data/Rakefile +4 -2
  4. data/app/assets/images/magic-solid.svg +1 -0
  5. data/app/assets/javascripts/shipit/repositories_search.js.coffee +60 -0
  6. data/app/assets/javascripts/shipit/{search.js.coffee → stack_search.js.coffee} +0 -0
  7. data/app/assets/stylesheets/_pages/_deploy.scss +0 -2
  8. data/app/assets/stylesheets/_pages/_repositories.scss +148 -0
  9. data/app/assets/stylesheets/_pages/_stacks.scss +19 -3
  10. data/app/assets/stylesheets/merge_status.scss +0 -3
  11. data/app/assets/stylesheets/shipit.scss +1 -0
  12. data/app/controllers/concerns/shipit/active_model_serializers_patch.rb +1 -0
  13. data/app/controllers/concerns/shipit/api/cacheable.rb +1 -0
  14. data/app/controllers/concerns/shipit/api/paginable.rb +3 -2
  15. data/app/controllers/concerns/shipit/api/rendering.rb +5 -4
  16. data/app/controllers/concerns/shipit/authentication.rb +3 -2
  17. data/app/controllers/concerns/shipit/pagination.rb +2 -1
  18. data/app/controllers/shipit/api/base_controller.rb +11 -6
  19. data/app/controllers/shipit/api/ccmenu_controller.rb +2 -1
  20. data/app/controllers/shipit/api/commits_controller.rb +2 -1
  21. data/app/controllers/shipit/api/deploys_controller.rb +6 -3
  22. data/app/controllers/shipit/api/hooks_controller.rb +6 -5
  23. data/app/controllers/shipit/api/locks_controller.rb +5 -4
  24. data/app/controllers/shipit/api/merge_requests_controller.rb +37 -0
  25. data/app/controllers/shipit/api/outputs_controller.rb +2 -1
  26. data/app/controllers/shipit/api/release_statuses_controller.rb +3 -2
  27. data/app/controllers/shipit/api/rollbacks_controller.rb +34 -0
  28. data/app/controllers/shipit/api/stacks_controller.rb +32 -5
  29. data/app/controllers/shipit/api/tasks_controller.rb +6 -5
  30. data/app/controllers/shipit/api_clients_controller.rb +4 -3
  31. data/app/controllers/shipit/ccmenu_url_controller.rb +4 -3
  32. data/app/controllers/shipit/commit_checks_controller.rb +2 -1
  33. data/app/controllers/shipit/commits_controller.rb +2 -1
  34. data/app/controllers/shipit/deploys_controller.rb +4 -3
  35. data/app/controllers/shipit/github_authentication_controller.rb +4 -3
  36. data/app/controllers/shipit/merge_requests_controller.rb +31 -0
  37. data/app/controllers/shipit/merge_status_controller.rb +31 -30
  38. data/app/controllers/shipit/release_statuses_controller.rb +3 -2
  39. data/app/controllers/shipit/repositories_controller.rb +74 -0
  40. data/app/controllers/shipit/rollbacks_controller.rb +3 -2
  41. data/app/controllers/shipit/shipit_controller.rb +2 -1
  42. data/app/controllers/shipit/stacks_controller.rb +24 -9
  43. data/app/controllers/shipit/status_controller.rb +2 -1
  44. data/app/controllers/shipit/tasks_controller.rb +7 -6
  45. data/app/controllers/shipit/webhooks_controller.rb +26 -6
  46. data/app/helpers/shipit/chunks_helper.rb +3 -2
  47. data/app/helpers/shipit/deploys_helper.rb +4 -3
  48. data/app/helpers/shipit/github_url_helper.rb +9 -0
  49. data/app/helpers/shipit/merge_status_helper.rb +1 -0
  50. data/app/helpers/shipit/shipit_helper.rb +1 -1
  51. data/app/helpers/shipit/stacks_helper.rb +5 -0
  52. data/app/helpers/shipit/tasks_helper.rb +1 -0
  53. data/app/jobs/shipit/background_job.rb +4 -0
  54. data/app/jobs/shipit/background_job/unique.rb +1 -0
  55. data/app/jobs/shipit/cache_deploy_spec_job.rb +1 -0
  56. data/app/jobs/shipit/chunk_rollup_job.rb +4 -0
  57. data/app/jobs/shipit/clear_git_cache_job.rb +1 -0
  58. data/app/jobs/shipit/continuous_delivery_job.rb +2 -1
  59. data/app/jobs/shipit/create_on_github_job.rb +7 -1
  60. data/app/jobs/shipit/create_release_statuses_job.rb +1 -0
  61. data/app/jobs/shipit/deferred_touch_job.rb +4 -0
  62. data/app/jobs/shipit/deliver_hook_job.rb +2 -1
  63. data/app/jobs/shipit/destroy_job.rb +1 -0
  64. data/app/jobs/shipit/destroy_repository_job.rb +24 -0
  65. data/app/jobs/shipit/destroy_stack_job.rb +3 -2
  66. data/app/jobs/shipit/emit_event_job.rb +2 -1
  67. data/app/jobs/shipit/fetch_commit_stats_job.rb +1 -0
  68. data/app/jobs/shipit/fetch_deployed_revision_job.rb +1 -0
  69. data/app/jobs/shipit/github_sync_job.rb +15 -10
  70. data/app/jobs/shipit/mark_deploy_healthy_job.rb +1 -0
  71. data/app/jobs/shipit/perform_commit_checks_job.rb +1 -0
  72. data/app/jobs/shipit/perform_task_job.rb +5 -92
  73. data/app/jobs/shipit/process_merge_requests_job.rb +32 -0
  74. data/app/jobs/shipit/purge_old_deliveries_job.rb +1 -0
  75. data/app/jobs/shipit/reap_dead_tasks_job.rb +21 -0
  76. data/app/jobs/shipit/refresh_check_runs_job.rb +1 -0
  77. data/app/jobs/shipit/refresh_github_user_job.rb +1 -0
  78. data/app/jobs/shipit/refresh_merge_request_job.rb +11 -0
  79. data/app/jobs/shipit/refresh_statuses_job.rb +1 -0
  80. data/app/jobs/shipit/setup_github_hook_job.rb +1 -0
  81. data/app/jobs/shipit/update_estimated_deploy_duration_job.rb +1 -0
  82. data/app/jobs/shipit/update_github_last_deployed_ref_job.rb +5 -4
  83. data/app/models/concerns/shipit/deferred_touch.rb +4 -3
  84. data/app/models/shipit/anonymous_user.rb +15 -2
  85. data/app/models/shipit/api_client.rb +3 -2
  86. data/app/models/shipit/application_record.rb +2 -1
  87. data/app/models/shipit/check_run.rb +41 -4
  88. data/app/models/shipit/command_line_user.rb +5 -0
  89. data/app/models/shipit/commit.rb +42 -24
  90. data/app/models/shipit/commit_checks.rb +15 -13
  91. data/app/models/shipit/commit_deployment.rb +6 -5
  92. data/app/models/shipit/commit_deployment_status.rb +5 -4
  93. data/app/models/shipit/commit_message.rb +1 -0
  94. data/app/models/shipit/delivery.rb +4 -3
  95. data/app/models/shipit/deploy.rb +23 -28
  96. data/app/models/shipit/deploy_spec.rb +38 -7
  97. data/app/models/shipit/deploy_spec/bundler_discovery.rb +1 -0
  98. data/app/models/shipit/deploy_spec/capistrano_discovery.rb +1 -0
  99. data/app/models/shipit/deploy_spec/file_system.rb +20 -7
  100. data/app/models/shipit/deploy_spec/kubernetes_discovery.rb +1 -0
  101. data/app/models/shipit/deploy_spec/lerna_discovery.rb +13 -4
  102. data/app/models/shipit/deploy_spec/npm_discovery.rb +5 -4
  103. data/app/models/shipit/deploy_spec/pypi_discovery.rb +1 -0
  104. data/app/models/shipit/deploy_spec/rubygems_discovery.rb +1 -0
  105. data/app/models/shipit/deploy_stats.rb +2 -1
  106. data/app/models/shipit/duration.rb +5 -2
  107. data/app/models/shipit/ephemeral_commit_checks.rb +1 -0
  108. data/app/models/shipit/github_hook.rb +2 -1
  109. data/app/models/shipit/github_status.rb +2 -1
  110. data/app/models/shipit/hook.rb +34 -7
  111. data/app/models/shipit/membership.rb +3 -2
  112. data/app/models/shipit/merge_request.rb +304 -0
  113. data/app/models/shipit/output_chunk.rb +7 -2
  114. data/app/models/shipit/provisioning_handler.rb +32 -0
  115. data/app/models/shipit/provisioning_handler/base.rb +30 -0
  116. data/app/models/shipit/provisioning_handler/unregistered_provisioning_handler.rb +35 -0
  117. data/app/models/shipit/pull_request.rb +28 -266
  118. data/app/models/shipit/pull_request_assignment.rb +10 -0
  119. data/app/models/shipit/record.rb +18 -0
  120. data/app/models/shipit/release_status.rb +4 -3
  121. data/app/models/shipit/repository.rb +71 -6
  122. data/app/models/shipit/review_stack.rb +130 -0
  123. data/app/models/shipit/review_stack_provisioning_queue.rb +39 -0
  124. data/app/models/shipit/rollback.rb +1 -0
  125. data/app/models/shipit/stack.rb +144 -44
  126. data/app/models/shipit/status.rb +3 -2
  127. data/app/models/shipit/status/common.rb +7 -6
  128. data/app/models/shipit/status/group.rb +2 -1
  129. data/app/models/shipit/status/missing.rb +2 -1
  130. data/app/models/shipit/status/unknown.rb +2 -1
  131. data/app/models/shipit/task.rb +98 -12
  132. data/app/models/shipit/task_definition.rb +1 -0
  133. data/app/models/shipit/task_execution_strategy/base.rb +20 -0
  134. data/app/models/shipit/task_execution_strategy/default.rb +109 -0
  135. data/app/models/shipit/team.rb +6 -3
  136. data/app/models/shipit/undeployed_commit.rb +1 -0
  137. data/app/models/shipit/unlimited_api_client.rb +1 -0
  138. data/app/models/shipit/user.rb +19 -8
  139. data/app/models/shipit/variable_definition.rb +1 -0
  140. data/app/models/shipit/webhooks.rb +11 -0
  141. data/app/models/shipit/webhooks/handlers/check_suite_handler.rb +1 -0
  142. data/app/models/shipit/webhooks/handlers/handler.rb +1 -0
  143. data/app/models/shipit/webhooks/handlers/membership_handler.rb +1 -0
  144. data/app/models/shipit/webhooks/handlers/pull_request/assigned_handler.rb +74 -0
  145. data/app/models/shipit/webhooks/handlers/pull_request/closed_handler.rb +68 -0
  146. data/app/models/shipit/webhooks/handlers/pull_request/edited_handler.rb +74 -0
  147. data/app/models/shipit/webhooks/handlers/pull_request/label_capturing_handler.rb +127 -0
  148. data/app/models/shipit/webhooks/handlers/pull_request/labeled_handler.rb +106 -0
  149. data/app/models/shipit/webhooks/handlers/pull_request/opened_handler.rb +83 -0
  150. data/app/models/shipit/webhooks/handlers/pull_request/reopened_handler.rb +88 -0
  151. data/app/models/shipit/webhooks/handlers/pull_request/review_stack_adapter.rb +103 -0
  152. data/app/models/shipit/webhooks/handlers/pull_request/unlabeled_handler.rb +107 -0
  153. data/app/models/shipit/webhooks/handlers/push_handler.rb +5 -1
  154. data/app/models/shipit/webhooks/handlers/status_handler.rb +1 -0
  155. data/app/serializers/concerns/shipit/conditional_attributes.rb +1 -0
  156. data/app/serializers/shipit/anonymous_user_serializer.rb +1 -0
  157. data/app/serializers/shipit/command_line_user_serializer.rb +1 -0
  158. data/app/serializers/shipit/commit_serializer.rb +1 -0
  159. data/app/serializers/shipit/deploy_serializer.rb +8 -1
  160. data/app/serializers/shipit/hook_serializer.rb +1 -0
  161. data/app/serializers/shipit/merge_request_serializer.rb +21 -0
  162. data/app/serializers/shipit/pull_request_serializer.rb +6 -8
  163. data/app/serializers/shipit/review_stack_serializer.rb +7 -0
  164. data/app/serializers/shipit/rollback_serializer.rb +1 -0
  165. data/app/serializers/shipit/short_commit_serializer.rb +1 -0
  166. data/app/serializers/shipit/stack_serializer.rb +8 -6
  167. data/app/serializers/shipit/tail_task_serializer.rb +11 -2
  168. data/app/serializers/shipit/task_serializer.rb +2 -17
  169. data/app/serializers/shipit/user_serializer.rb +6 -1
  170. data/app/validators/ascii_only_validator.rb +1 -0
  171. data/app/validators/subset_validator.rb +2 -1
  172. data/app/views/layouts/merge_status.html.erb +1 -1
  173. data/app/views/layouts/shipit.html.erb +1 -1
  174. data/app/views/shipit/_variables.html.erb +1 -1
  175. data/app/views/shipit/ccmenu/project.xml.builder +2 -1
  176. data/app/views/shipit/deploys/show.html.erb +2 -2
  177. data/app/views/shipit/merge_requests/_merge_request.html.erb +29 -0
  178. data/app/views/shipit/{pull_requests → merge_requests}/index.html.erb +2 -2
  179. data/app/views/shipit/merge_requests/merge_requests/_pull_request.html.erb +29 -0
  180. data/app/views/shipit/merge_requests/merge_requests/index.html.erb +20 -0
  181. data/app/views/shipit/merge_status/_merge_queue_button.html.erb +3 -3
  182. data/app/views/shipit/merge_status/backlogged.html.erb +1 -1
  183. data/app/views/shipit/merge_status/failure.html.erb +1 -1
  184. data/app/views/shipit/merge_status/locked.html.erb +1 -1
  185. data/app/views/shipit/merge_status/success.html.erb +2 -2
  186. data/app/views/shipit/repositories/_header.html.erb +19 -0
  187. data/app/views/shipit/repositories/index.html.erb +31 -0
  188. data/app/views/shipit/repositories/new.html.erb +23 -0
  189. data/app/views/shipit/repositories/settings.html.erb +53 -0
  190. data/app/views/shipit/repositories/show.html.erb +30 -0
  191. data/app/views/shipit/stacks/_banners.html.erb +15 -1
  192. data/app/views/shipit/stacks/_header.html.erb +20 -7
  193. data/app/views/shipit/stacks/_stack.html.erb +8 -0
  194. data/app/views/shipit/stacks/all_tasks.html.erb +28 -0
  195. data/app/views/shipit/stacks/index.html.erb +3 -2
  196. data/app/views/shipit/stacks/new.html.erb +1 -1
  197. data/app/views/shipit/stacks/settings.html.erb +5 -5
  198. data/app/views/shipit/stacks/show.html.erb +1 -1
  199. data/app/views/shipit/tasks/_task_output.html.erb +1 -1
  200. data/app/views/shipit/tasks/show.html.erb +1 -1
  201. data/config/initializers/inflections.rb +2 -1
  202. data/config/locales/en.yml +4 -3
  203. data/config/routes.rb +25 -7
  204. data/config/secrets.development.example.yml +24 -0
  205. data/config/secrets.development.shopify.yml +20 -9
  206. data/db/migrate/20200226211925_add_index_to_tasks_status.rb +5 -0
  207. data/db/migrate/20200427135152_add_pull_request_head_sha_to_commit.rb +5 -0
  208. data/db/migrate/20200615181558_add_rollback_once_aborted_to.rb +5 -0
  209. data/db/migrate/20200706145406_add_review_stacks.rb +12 -0
  210. data/db/migrate/20200804144639_rename_pull_request_to_merge_request.rb +7 -0
  211. data/db/migrate/20200804161512_rename_commits_pull_request_id_to_merge_request_id.rb +5 -0
  212. data/db/migrate/20200813134712_recreate_shipit_pull_requests.rb +22 -0
  213. data/db/migrate/20200813194056_create_pull_request_assignments.rb +8 -0
  214. data/db/migrate/20201001125502_add_provision_pr_stacks_flag_to_repositories.rb +7 -0
  215. data/db/migrate/20201008145809_add_retry_attempt_to_tasks.rb +5 -0
  216. data/db/migrate/20201008152744_add_max_retries_to_tasks.rb +5 -0
  217. data/db/migrate/20210325194053_remove_stacks_branch_default.rb +5 -0
  218. data/db/migrate/20210504200438_add_github_updated_at_to_check_runs.rb +5 -0
  219. data/lib/shipit.rb +61 -17
  220. data/lib/shipit/cast_value.rb +1 -0
  221. data/lib/shipit/command.rb +20 -21
  222. data/lib/shipit/commands.rb +14 -6
  223. data/lib/shipit/csv_serializer.rb +1 -0
  224. data/lib/shipit/deploy_commands.rb +1 -0
  225. data/lib/shipit/engine.rb +9 -2
  226. data/lib/shipit/environment_variables.rb +11 -1
  227. data/lib/shipit/first_parent_commits_iterator.rb +1 -0
  228. data/lib/shipit/flock.rb +9 -1
  229. data/lib/shipit/github_app.rb +15 -12
  230. data/lib/shipit/github_http_cache_middleware.rb +1 -0
  231. data/lib/shipit/null_serializer.rb +1 -0
  232. data/lib/shipit/octokit_check_runs.rb +3 -2
  233. data/lib/shipit/octokit_iterator.rb +4 -3
  234. data/lib/shipit/paginator.rb +3 -2
  235. data/lib/shipit/review_stack_commands.rb +8 -0
  236. data/lib/shipit/rollback_commands.rb +1 -0
  237. data/lib/shipit/same_site_cookie_middleware.rb +29 -0
  238. data/lib/shipit/simple_message_verifier.rb +3 -2
  239. data/lib/shipit/stack_commands.rb +37 -7
  240. data/lib/shipit/stat.rb +1 -0
  241. data/lib/shipit/task_commands.rb +23 -16
  242. data/lib/shipit/version.rb +2 -1
  243. data/lib/snippets/publish-lerna-independent-packages +35 -34
  244. data/lib/snippets/publish-lerna-independent-packages-legacy +39 -0
  245. data/lib/snippets/release-gem +5 -1
  246. data/lib/tasks/cron.rake +13 -2
  247. data/lib/tasks/dev.rake +3 -2
  248. data/lib/tasks/shipit.rake +15 -14
  249. data/lib/tasks/teams.rake +1 -0
  250. data/test/controllers/api/base_controller_test.rb +3 -2
  251. data/test/controllers/api/ccmenu_controller_test.rb +9 -8
  252. data/test/controllers/api/commits_controller_test.rb +3 -2
  253. data/test/controllers/api/deploys_controller_test.rb +32 -14
  254. data/test/controllers/api/hooks_controller_test.rb +8 -7
  255. data/test/controllers/api/locks_controller_test.rb +7 -6
  256. data/test/controllers/api/{pull_requests_controller_test.rb → merge_requests_controller_test.rb} +17 -16
  257. data/test/controllers/api/outputs_controller_test.rb +3 -1
  258. data/test/controllers/api/release_statuses_controller_test.rb +2 -1
  259. data/test/controllers/api/rollback_controller_test.rb +113 -0
  260. data/test/controllers/api/stacks_controller_test.rb +71 -16
  261. data/test/controllers/api/tasks_controller_test.rb +13 -12
  262. data/test/controllers/api_clients_controller_test.rb +5 -4
  263. data/test/controllers/ccmenu_controller_test.rb +4 -3
  264. data/test/controllers/commit_checks_controller_test.rb +4 -3
  265. data/test/controllers/commits_controller_test.rb +3 -2
  266. data/test/controllers/deploys_controller_test.rb +32 -21
  267. data/test/controllers/github_authentication_controller_test.rb +1 -0
  268. data/test/controllers/merge_requests_controller_test.rb +32 -0
  269. data/test/controllers/merge_status_controller_test.rb +7 -6
  270. data/test/controllers/release_statuses_controller_test.rb +3 -2
  271. data/test/controllers/repositories_controller_test.rb +71 -0
  272. data/test/controllers/rollbacks_controller_test.rb +9 -8
  273. data/test/controllers/stacks_controller_test.rb +41 -19
  274. data/test/controllers/status_controller_test.rb +1 -0
  275. data/test/controllers/tasks_controller_test.rb +32 -19
  276. data/test/controllers/webhooks_controller_test.rb +33 -17
  277. data/test/dummy/app/assets/config/manifest.js +3 -0
  278. data/test/dummy/config/application.rb +7 -2
  279. data/test/dummy/config/database.yml +9 -0
  280. data/test/dummy/config/environments/development.rb +3 -4
  281. data/test/dummy/config/environments/test.rb +2 -5
  282. data/test/dummy/config/secrets_double_github_app.yml +79 -0
  283. data/test/dummy/db/schema.rb +59 -17
  284. data/test/dummy/db/seeds.rb +2 -1
  285. data/test/fixtures/payloads/check_suite_master.json +4 -32
  286. data/test/fixtures/payloads/invalid_pull_request.json +117 -0
  287. data/test/fixtures/payloads/provision_disabled_pull_request.json +454 -0
  288. data/test/fixtures/payloads/pull_request_assigned.json +480 -0
  289. data/test/fixtures/payloads/pull_request_closed.json +454 -0
  290. data/test/fixtures/payloads/pull_request_labeled.json +461 -0
  291. data/test/fixtures/payloads/pull_request_opened.json +454 -0
  292. data/test/fixtures/payloads/pull_request_reopened.json +454 -0
  293. data/test/fixtures/payloads/pull_request_unlabeled.json +454 -0
  294. data/test/fixtures/payloads/pull_request_with_no_repo.json +454 -0
  295. data/test/fixtures/payloads/push_master.json +1 -1
  296. data/test/fixtures/payloads/push_not_master.json +1 -1
  297. data/test/fixtures/shipit/commits.yml +31 -3
  298. data/test/fixtures/shipit/hooks.yml +1 -0
  299. data/test/fixtures/shipit/merge_requests.yml +141 -0
  300. data/test/fixtures/shipit/pull_request_assignments.yml +3 -0
  301. data/test/fixtures/shipit/pull_requests.yml +10 -131
  302. data/test/fixtures/shipit/repositories.yml +5 -0
  303. data/test/fixtures/shipit/stacks.yml +235 -14
  304. data/test/fixtures/shipit/statuses.yml +9 -0
  305. data/test/fixtures/shipit/tasks.yml +4 -1
  306. data/test/fixtures/shipit/users.yml +7 -0
  307. data/test/fixtures/timeout +2 -1
  308. data/test/helpers/api_helper.rb +1 -0
  309. data/test/helpers/fixture_aliases_helper.rb +1 -0
  310. data/test/helpers/hooks_helper.rb +2 -1
  311. data/test/helpers/json_helper.rb +20 -12
  312. data/test/helpers/links_helper.rb +4 -3
  313. data/test/helpers/payloads_helper.rb +5 -0
  314. data/test/helpers/queries_helper.rb +3 -2
  315. data/test/jobs/cache_deploy_spec_job_test.rb +2 -1
  316. data/test/jobs/chunk_rollup_job_test.rb +16 -1
  317. data/test/jobs/deliver_hook_job_test.rb +1 -0
  318. data/test/jobs/destroy_repository_job_test.rb +27 -0
  319. data/test/jobs/destroy_stack_job_test.rb +1 -0
  320. data/test/jobs/emit_event_job_test.rb +2 -1
  321. data/test/jobs/fetch_commit_stats_job_test.rb +1 -0
  322. data/test/jobs/fetch_deployed_revision_job_test.rb +1 -0
  323. data/test/jobs/github_sync_job_test.rb +3 -1
  324. data/test/jobs/mark_deploy_healthy_job_test.rb +1 -0
  325. data/test/jobs/perform_task_job_test.rb +12 -11
  326. data/test/jobs/{merge_pull_requests_job_test.rb → process_merge_requests_job_test.rb} +19 -18
  327. data/test/jobs/purge_old_deliveries_job_test.rb +1 -0
  328. data/test/jobs/reap_dead_tasks_job_test.rb +68 -0
  329. data/test/jobs/refresh_github_user_job_test.rb +1 -0
  330. data/test/jobs/refresh_status_job_test.rb +1 -0
  331. data/test/jobs/unique_job_test.rb +1 -0
  332. data/test/jobs/update_github_last_deployed_ref_job_test.rb +1 -0
  333. data/test/lib/shipit/deploy_commands_test.rb +16 -0
  334. data/test/lib/shipit/task_commands_test.rb +17 -0
  335. data/test/middleware/same_site_cookie_middleware_test.rb +52 -0
  336. data/test/models/api_client_test.rb +1 -0
  337. data/test/models/commit_checks_test.rb +1 -0
  338. data/test/models/commit_deployment_status_test.rb +4 -3
  339. data/test/models/commit_deployment_test.rb +2 -1
  340. data/test/models/commits_test.rb +96 -19
  341. data/test/models/delivery_test.rb +2 -1
  342. data/test/models/deploy_spec_test.rb +110 -65
  343. data/test/models/deploy_stats_test.rb +1 -0
  344. data/test/models/deploys_test.rb +197 -36
  345. data/test/models/duration_test.rb +1 -0
  346. data/test/models/github_hook_test.rb +1 -0
  347. data/test/models/hook_test.rb +47 -10
  348. data/test/models/membership_test.rb +1 -0
  349. data/test/models/{pull_request_test.rb → merge_request_test.rb} +53 -37
  350. data/test/models/pull_request_assignment_test.rb +16 -0
  351. data/test/models/release_statuses_test.rb +1 -0
  352. data/test/models/rollbacks_test.rb +1 -0
  353. data/test/models/shipit/check_run_test.rb +125 -5
  354. data/test/models/shipit/provisioning_handler/base_test.rb +33 -0
  355. data/test/models/shipit/provisioning_handler/unregistered_provisioning_handler_test.rb +49 -0
  356. data/test/models/shipit/provisioning_handler_test.rb +64 -0
  357. data/test/models/shipit/pull_request_test.rb +52 -0
  358. data/test/models/shipit/repository_test.rb +6 -1
  359. data/test/models/shipit/review_stack_provision_status_test.rb +77 -0
  360. data/test/models/shipit/review_stack_provisioning_queue_test.rb +63 -0
  361. data/test/models/shipit/review_stack_test.rb +91 -0
  362. data/test/models/{stacks_test.rb → shipit/stacks_test.rb} +121 -16
  363. data/test/models/shipit/webhooks/handlers/pull_request/assigned_handler_test.rb +45 -0
  364. data/test/models/shipit/webhooks/handlers/pull_request/closed_handler_test.rb +192 -0
  365. data/test/models/shipit/webhooks/handlers/pull_request/edited_handler_test.rb +47 -0
  366. data/test/models/shipit/webhooks/handlers/pull_request/label_capturing_handler_test.rb +209 -0
  367. data/test/models/shipit/webhooks/handlers/pull_request/labeled_handler_test.rb +332 -0
  368. data/test/models/shipit/webhooks/handlers/pull_request/opened_handler_test.rb +238 -0
  369. data/test/models/shipit/webhooks/handlers/pull_request/reopened_handler_test.rb +282 -0
  370. data/test/models/shipit/webhooks/handlers/pull_request/review_stack_adapter_test.rb +107 -0
  371. data/test/models/shipit/webhooks/handlers/pull_request/unlabeled_handler_test.rb +324 -0
  372. data/test/models/shipit/{wehbooks → webhooks}/handlers_test.rb +1 -0
  373. data/test/models/status/group_test.rb +1 -0
  374. data/test/models/status/missing_test.rb +1 -0
  375. data/test/models/status_test.rb +1 -0
  376. data/test/models/task_definitions_test.rb +9 -8
  377. data/test/models/tasks_test.rb +81 -1
  378. data/test/models/team_test.rb +4 -2
  379. data/test/models/undeployed_commits_test.rb +1 -0
  380. data/test/models/users_test.rb +13 -5
  381. data/test/serializers/shipit/pull_request_serializer_test.rb +29 -0
  382. data/test/test_command_integration.rb +3 -2
  383. data/test/test_helper.rb +49 -31
  384. data/test/unit/anonymous_user_serializer_test.rb +14 -0
  385. data/test/unit/command_test.rb +16 -10
  386. data/test/unit/commands_test.rb +1 -0
  387. data/test/unit/commit_serializer_test.rb +16 -0
  388. data/test/unit/csv_serializer_test.rb +3 -2
  389. data/test/unit/deploy_commands_test.rb +75 -18
  390. data/test/unit/deploy_serializer_test.rb +17 -0
  391. data/test/unit/environment_variables_test.rb +5 -4
  392. data/test/unit/github_app_test.rb +3 -3
  393. data/test/unit/github_apps_test.rb +416 -0
  394. data/test/unit/github_url_helper_test.rb +6 -0
  395. data/test/unit/rollback_commands_test.rb +2 -1
  396. data/test/unit/shipit_deployment_checks_test.rb +77 -0
  397. data/test/unit/shipit_helper_test.rb +17 -0
  398. data/test/unit/shipit_task_execution_strategy_test.rb +47 -0
  399. data/test/unit/shipit_test.rb +15 -0
  400. data/test/unit/user_serializer_test.rb +14 -0
  401. data/test/unit/variable_definition_test.rb +1 -0
  402. metadata +320 -178
  403. data/app/controllers/shipit/api/pull_requests_controller.rb +0 -36
  404. data/app/controllers/shipit/pull_requests_controller.rb +0 -30
  405. data/app/jobs/shipit/merge_pull_requests_job.rb +0 -31
  406. data/app/jobs/shipit/refresh_pull_request_job.rb +0 -10
  407. data/app/views/shipit/pull_requests/_pull_request.html.erb +0 -29
  408. data/test/controllers/pull_requests_controller_test.rb +0 -31
  409. data/test/fixtures/shipit/output_chunks.yml +0 -47
  410. data/test/models/output_chunk_test.rb +0 -20
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  module BundlerDiscovery
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  module CapistranoDiscovery
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  class FileSystem < DeploySpec
@@ -12,6 +13,7 @@ module Shipit
12
13
  def initialize(app_dir, env)
13
14
  @app_dir = Pathname(app_dir)
14
15
  @env = env
16
+ super(nil)
15
17
  end
16
18
 
17
19
  def cacheable
@@ -31,10 +33,10 @@ module Shipit
31
33
  def cacheable_config
32
34
  (config || {}).deep_merge(
33
35
  'merge' => {
34
- 'require' => pull_request_required_statuses,
35
- 'ignore' => pull_request_ignored_statuses,
36
- 'revalidate_after' => revalidate_pull_requests_after&.to_i,
37
- 'method' => pull_request_merge_method,
36
+ 'require' => merge_request_required_statuses,
37
+ 'ignore' => merge_request_ignored_statuses,
38
+ 'revalidate_after' => revalidate_merge_requests_after&.to_i,
39
+ 'method' => merge_request_merge_method,
38
40
  'max_divergence' => {
39
41
  'commits' => max_divergence_commits&.to_i,
40
42
  'age' => max_divergence_age&.to_i,
@@ -61,14 +63,19 @@ module Shipit
61
63
  'context' => release_status_context,
62
64
  'delay' => release_status_delay,
63
65
  },
64
- 'dependencies' => {'override' => dependencies_steps},
66
+ 'dependencies' => { 'override' => dependencies_steps },
67
+ 'provision' => { 'handler_name' => provisioning_handler_name },
65
68
  'deploy' => {
66
69
  'override' => deploy_steps,
67
70
  'variables' => deploy_variables.map(&:to_h),
68
71
  'max_commits' => maximum_commits_per_deploy,
69
72
  'interval' => pause_between_deploys,
73
+ 'retries' => retries_on_deploy,
74
+ },
75
+ 'rollback' => {
76
+ 'override' => rollback_steps,
77
+ 'retries' => retries_on_rollback,
70
78
  },
71
- 'rollback' => {'override' => rollback_steps},
72
79
  'fetch' => fetch_deployed_revision_steps,
73
80
  'tasks' => cacheable_tasks,
74
81
  )
@@ -84,10 +91,16 @@ module Shipit
84
91
  end
85
92
 
86
93
  def load_config
87
- read_config(file("shipit.#{@env}.yml", root: true)) ||
94
+ read_config(file("#{app_name}.#{@env}.yml", root: true)) ||
95
+ read_config(file("#{app_name}.yml", root: true)) ||
96
+ read_config(file("shipit.#{@env}.yml", root: true)) ||
88
97
  read_config(file('shipit.yml', root: true))
89
98
  end
90
99
 
100
+ def app_name
101
+ @app_name ||= Shipit.app_name.downcase
102
+ end
103
+
91
104
  def read_config(path)
92
105
  SafeYAML.load(path.read) if path.exist?
93
106
  end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  module KubernetesDiscovery
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  require 'json'
2
3
 
3
4
  module Shipit
@@ -75,10 +76,18 @@ module Shipit
75
76
  end
76
77
 
77
78
  def publish_independent_packages
78
- [
79
- 'assert-lerna-independent-version-tags',
80
- 'publish-lerna-independent-packages',
81
- ]
79
+ command = if lerna_lerna >= LATEST_MAJOR_VERSION
80
+ [
81
+ 'assert-lerna-independent-version-tags',
82
+ 'publish-lerna-independent-packages',
83
+ ]
84
+ else
85
+ [
86
+ 'assert-lerna-independent-version-tags',
87
+ 'publish-lerna-independent-packages-legacy',
88
+ ]
89
+ end
90
+ command
82
91
  end
83
92
 
84
93
  def publish_fixed_version_packages
@@ -1,13 +1,14 @@
1
+ # frozen_string_literal: true
1
2
  require 'json'
2
3
 
3
4
  module Shipit
4
5
  class DeploySpec
5
6
  module NpmDiscovery
6
7
  # https://docs.npmjs.com/cli/publish
7
- PUBLIC = 'public'.freeze
8
- PRIVATE = 'restricted'.freeze
8
+ PUBLIC = 'public'
9
+ PRIVATE = 'restricted'
9
10
  VALID_ACCESS = [PUBLIC, PRIVATE].freeze
10
- NPM_REGISTRY = "https://registry.npmjs.org/".freeze
11
+ NPM_REGISTRY = "https://registry.npmjs.org/"
11
12
 
12
13
  def discover_dependencies_steps
13
14
  discover_package_json || super
@@ -57,7 +58,7 @@ module Shipit
57
58
  # are treated as 'next' npm dist-tags.
58
59
  # An 1.0.0-beta.1 would be installable using both:
59
60
  # `yarn add package@1.0.0-beta.1` and `yarn add package@next`
60
- return 'next' if ['-beta', '-alpha', '-rc', '-next'].any? { |tag| version.include? tag }
61
+ return 'next' if ['-beta', '-alpha', '-rc', '-next'].any? { |tag| version.include?(tag) }
61
62
  'latest'
62
63
  end
63
64
 
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  module PypiDiscovery
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeploySpec
3
4
  module RubygemsDiscovery
@@ -1,10 +1,11 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class DeployStats
3
4
  delegate :empty?, to: :@deploys
4
5
 
5
6
  def initialize(deploys)
6
7
  @deploys = deploys
7
- @durations = @deploys.map { |d| d.duration.value }.compact
8
+ @durations = @deploys.map { |d| d.duration&.value }.compact
8
9
  end
9
10
 
10
11
  def count
@@ -1,15 +1,16 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class Duration < ActiveSupport::Duration
3
4
  ParseError = Class.new(ArgumentError)
4
5
 
5
- FORMAT = /
6
+ FORMAT = %r{
6
7
  \A
7
8
  (?<days>\d+d)?
8
9
  (?<hours>\d+h)?
9
10
  (?<minutes>\d+m)?
10
11
  (?<seconds>\d+s?)?
11
12
  \z
12
- /x
13
+ }x
13
14
  UNITS = {
14
15
  's' => :seconds,
15
16
  'm' => :minutes,
@@ -19,6 +20,8 @@ module Shipit
19
20
 
20
21
  class << self
21
22
  def parse(value)
23
+ return new(-1) if value.to_s == "-1"
24
+
22
25
  unless match = FORMAT.match(value.to_s)
23
26
  raise ParseError, "not a duration: #{value.inspect}"
24
27
  end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class EphemeralCommitChecks
3
4
  FINAL_STATUSES = %w(failed error success).freeze
@@ -1,5 +1,6 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class GithubHook < ActiveRecord::Base
3
+ class GithubHook < Record
3
4
  # TODO: app-migration, delete class
4
5
  belongs_to :stack, required: false # Required for fixtures
5
6
 
@@ -1,6 +1,7 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  module GithubStatus
3
- CACHE_KEY = 'github::status'.freeze
4
+ CACHE_KEY = 'github::status'
4
5
 
5
6
  class << self
6
7
  def status
@@ -1,11 +1,28 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class Hook < ActiveRecord::Base
3
+ class Hook < Record
4
+ class DeliverySigner
5
+ attr_reader :secret
6
+
7
+ ALGORITHM = 'sha256'
8
+
9
+ def initialize(secret)
10
+ @secret = secret
11
+ end
12
+
13
+ def sign(payload)
14
+ hmac = OpenSSL::HMAC.hexdigest(ALGORITHM, secret, payload)
15
+ "#{ALGORITHM}=#{hmac}"
16
+ end
17
+ end
18
+
3
19
  class DeliverySpec
4
- def initialize(event:, url:, content_type:, payload:)
20
+ def initialize(event:, url:, content_type:, payload:, secret:)
5
21
  @event = event
6
22
  @url = url
7
23
  @content_type = content_type
8
24
  @payload = payload
25
+ @secret = secret
9
26
  end
10
27
 
11
28
  def send!
@@ -14,12 +31,12 @@ module Shipit
14
31
 
15
32
  private
16
33
 
17
- attr_reader :event, :url, :content_type, :payload
34
+ attr_reader :event, :url, :content_type, :payload, :secret
18
35
 
19
36
  def http
20
37
  Faraday::Connection.new do |connection|
21
38
  connection.headers = headers
22
- connection.adapter Faraday.default_adapter
39
+ connection.adapter(Faraday.default_adapter)
23
40
  end
24
41
  end
25
42
 
@@ -28,9 +45,16 @@ module Shipit
28
45
  'User-Agent' => 'Shipit Webhook',
29
46
  'Content-Type' => content_type,
30
47
  'X-Shipit-Event' => event,
48
+ 'X-Shipit-Signature' => signature,
31
49
  'Accept' => '*/*',
32
50
  }
33
51
  end
52
+
53
+ def signature
54
+ return nil if secret.blank?
55
+
56
+ DeliverySigner.new(secret).sign(payload)
57
+ end
34
58
  end
35
59
 
36
60
  default_scope { order :id }
@@ -44,6 +68,7 @@ module Shipit
44
68
 
45
69
  EVENTS = %w(
46
70
  stack
71
+ review_stack
47
72
  task
48
73
  deploy
49
74
  rollback
@@ -52,14 +77,15 @@ module Shipit
52
77
  deployable_status
53
78
  merge_status
54
79
  merge
80
+ pull_request
55
81
  ).freeze
56
82
 
57
83
  belongs_to :stack, required: false
58
84
  has_many :deliveries
59
85
 
60
- validates :delivery_url, presence: true, url: {no_local: true, allow_blank: true}
61
- validates :content_type, presence: true, inclusion: {in: CONTENT_TYPES.keys}
62
- validates :events, presence: true, subset: {of: EVENTS}
86
+ validates :delivery_url, presence: true, url: { no_local: true, allow_blank: true }
87
+ validates :content_type, presence: true, inclusion: { in: CONTENT_TYPES.keys }
88
+ validates :events, presence: true, subset: { of: EVENTS }
63
89
 
64
90
  serialize :events, Shipit::CSVSerializer
65
91
 
@@ -116,6 +142,7 @@ module Shipit
116
142
  url: delivery_url,
117
143
  content_type: CONTENT_TYPES[content_type],
118
144
  payload: serialize_payload(payload),
145
+ secret: secret,
119
146
  )
120
147
  end
121
148
 
@@ -1,8 +1,9 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class Membership < ActiveRecord::Base
3
+ class Membership < Record
3
4
  belongs_to :team, required: true
4
5
  belongs_to :user, required: true
5
6
 
6
- validates :user_id, uniqueness: {scope: :team_id}
7
+ validates :user_id, uniqueness: { scope: :team_id }
7
8
  end
8
9
  end
@@ -0,0 +1,304 @@
1
+ # frozen_string_literal: true
2
+ module Shipit
3
+ class MergeRequest < ApplicationRecord
4
+ include DeferredTouch
5
+
6
+ MERGE_REQUEST_FIELD = 'Merge-Requested-By'
7
+
8
+ WAITING_STATUSES = %w(fetching pending).freeze
9
+ QUEUED_STATUSES = %w(pending revalidating).freeze
10
+ REJECTION_REASONS = %w(ci_missing ci_failing merge_conflict requires_rebase).freeze
11
+ InvalidTransition = Class.new(StandardError)
12
+ NotReady = Class.new(StandardError)
13
+
14
+ class StatusChecker < Status::Group
15
+ def initialize(commit, statuses, deploy_spec)
16
+ @deploy_spec = deploy_spec
17
+ super(commit, statuses)
18
+ end
19
+
20
+ private
21
+
22
+ attr_reader :deploy_spec
23
+
24
+ def reject_hidden(statuses)
25
+ statuses.reject { |s| ignored_statuses.include?(s.context) }
26
+ end
27
+
28
+ def reject_allowed_to_fail(statuses)
29
+ statuses.reject { |s| ignored_statuses.include?(s.context) }
30
+ end
31
+
32
+ def ignored_statuses
33
+ deploy_spec&.merge_request_ignored_statuses || []
34
+ end
35
+
36
+ def required_statuses
37
+ deploy_spec&.merge_request_required_statuses || []
38
+ end
39
+ end
40
+
41
+ belongs_to :stack
42
+ belongs_to :head, class_name: 'Shipit::Commit', optional: true
43
+ belongs_to :base_commit, class_name: 'Shipit::Commit', optional: true
44
+ belongs_to :merge_requested_by, class_name: 'Shipit::User', optional: true
45
+ has_one :merge_commit, class_name: 'Shipit::Commit'
46
+
47
+ deferred_touch stack: :updated_at
48
+
49
+ validates :number, presence: true, uniqueness: { scope: :stack_id }
50
+
51
+ scope :waiting, -> { where(merge_status: WAITING_STATUSES) }
52
+ scope :pending, -> { where(merge_status: 'pending') }
53
+ scope :to_be_merged, -> { pending.order(merge_requested_at: :asc) }
54
+ scope :queued, -> { where(merge_status: QUEUED_STATUSES).order(merge_requested_at: :asc) }
55
+
56
+ after_save :record_merge_status_change
57
+ after_commit :emit_hooks
58
+
59
+ state_machine :merge_status, initial: :fetching do
60
+ state :fetching
61
+ state :pending
62
+ state :rejected
63
+ state :canceled
64
+ state :merged
65
+ state :revalidating
66
+
67
+ event :fetched do
68
+ transition fetching: :pending
69
+ end
70
+
71
+ event :reject do
72
+ transition pending: :rejected
73
+ end
74
+
75
+ event :revalidate do
76
+ transition pending: :revalidating
77
+ end
78
+
79
+ event :cancel do
80
+ transition any => :canceled
81
+ end
82
+
83
+ event :complete do
84
+ transition pending: :merged
85
+ end
86
+
87
+ event :retry do
88
+ transition %i(rejected canceled revalidating) => :pending
89
+ end
90
+
91
+ before_transition rejected: any do |pr|
92
+ pr.rejection_reason = nil
93
+ end
94
+
95
+ before_transition %i(fetching rejected canceled) => :pending do |pr|
96
+ pr.merge_requested_at = Time.now.utc
97
+ end
98
+
99
+ before_transition any => :pending do |pr|
100
+ pr.revalidated_at = Time.now.utc
101
+ end
102
+
103
+ before_transition %i(pending) => :merged do |pr|
104
+ Stack.increment_counter(:undeployed_commits_count, pr.stack_id)
105
+ end
106
+ end
107
+
108
+ def self.schedule_merges
109
+ Shipit::Stack.where(merge_queue_enabled: true).find_each(&:schedule_merges)
110
+ end
111
+
112
+ def self.extract_number(stack, number_or_url)
113
+ org = stack.repository.owner
114
+ case number_or_url
115
+ when /\A#?(\d+)\z/
116
+ $1.to_i
117
+ when %r{\Ahttps://#{Regexp.escape(Shipit.github(organization: org).domain)}/([^/]+)/([^/]+)/pull/(\d+)}
118
+ return unless $1.downcase == stack.repo_owner.downcase
119
+ return unless $2.downcase == stack.repo_name.downcase
120
+ $3.to_i
121
+ end
122
+ end
123
+
124
+ def self.request_merge!(stack, number, user)
125
+ now = Time.now.utc
126
+ merge_request = begin
127
+ create_with(
128
+ merge_requested_at: now,
129
+ merge_requested_by: user.presence,
130
+ ).find_or_create_by!(
131
+ stack: stack,
132
+ number: number,
133
+ )
134
+ rescue ActiveRecord::RecordNotUnique
135
+ retry
136
+ end
137
+ merge_request.update!(merge_requested_by: user.presence)
138
+ merge_request.retry! if merge_request.rejected? || merge_request.canceled? || merge_request.revalidating?
139
+ merge_request.schedule_refresh!
140
+ merge_request
141
+ end
142
+
143
+ def reject!(reason)
144
+ unless REJECTION_REASONS.include?(reason)
145
+ raise ArgumentError, "invalid reason: #{reason.inspect}, must be one of: #{REJECTION_REASONS.inspect}"
146
+ end
147
+ self.rejection_reason = reason.presence
148
+ super()
149
+ true
150
+ end
151
+
152
+ def reject_unless_mergeable!
153
+ return reject!('merge_conflict') if merge_conflict?
154
+ return reject!('ci_missing') if any_status_checks_missing?
155
+ return reject!('ci_failing') if any_status_checks_failed?
156
+ return reject!('requires_rebase') if stale?
157
+ false
158
+ end
159
+
160
+ def merge!
161
+ raise InvalidTransition unless pending?
162
+
163
+ raise NotReady if not_mergeable_yet?
164
+
165
+ stack.github_api.merge_pull_request(
166
+ stack.github_repo_name,
167
+ number,
168
+ merge_message,
169
+ sha: head.sha,
170
+ commit_message: 'Merged by Shipit',
171
+ merge_method: stack.merge_method,
172
+ )
173
+ begin
174
+ if stack.github_api.pull_requests(stack.github_repo_name, base: branch).empty?
175
+ stack.github_api.delete_branch(stack.github_repo_name, branch)
176
+ end
177
+ rescue Octokit::UnprocessableEntity
178
+ # branch was already deleted somehow
179
+ end
180
+ complete!
181
+ true
182
+ rescue Octokit::MethodNotAllowed # merge conflict
183
+ reject!('merge_conflict')
184
+ false
185
+ rescue Octokit::Conflict # shas didn't match, PR was updated.
186
+ raise NotReady
187
+ end
188
+
189
+ def all_status_checks_passed?
190
+ return false unless head
191
+ StatusChecker.new(head, head.statuses_and_check_runs, stack.cached_deploy_spec).success?
192
+ end
193
+
194
+ def any_status_checks_failed?
195
+ status = StatusChecker.new(head, head.statuses_and_check_runs, stack.cached_deploy_spec)
196
+ status.failure? || status.error?
197
+ end
198
+
199
+ def any_status_checks_missing?
200
+ StatusChecker.new(head, head.statuses_and_check_runs, stack.cached_deploy_spec).missing?
201
+ end
202
+
203
+ def waiting?
204
+ WAITING_STATUSES.include?(merge_status)
205
+ end
206
+
207
+ def need_revalidation?
208
+ timeout = stack.cached_deploy_spec&.revalidate_merge_requests_after
209
+ return false unless timeout
210
+ (revalidated_at + timeout).past?
211
+ end
212
+
213
+ def merge_conflict?
214
+ mergeable == false
215
+ end
216
+
217
+ def not_mergeable_yet?
218
+ mergeable.nil?
219
+ end
220
+
221
+ def schedule_refresh!
222
+ RefreshMergeRequestJob.perform_later(self)
223
+ end
224
+
225
+ def closed?
226
+ state == "closed"
227
+ end
228
+
229
+ def merged_upstream?
230
+ closed? && merged_at
231
+ end
232
+
233
+ def refresh!
234
+ update!(github_pull_request: stack.github_api.pull_request(stack.github_repo_name, number))
235
+ head.refresh_statuses!
236
+ head.refresh_check_runs!
237
+ fetched! if fetching?
238
+ @comparison = nil
239
+ end
240
+
241
+ def github_pull_request=(github_pull_request)
242
+ self.github_id = github_pull_request.id
243
+ self.api_url = github_pull_request.url
244
+ self.title = github_pull_request.title
245
+ self.state = github_pull_request.state
246
+ self.mergeable = github_pull_request.mergeable
247
+ self.additions = github_pull_request.additions
248
+ self.deletions = github_pull_request.deletions
249
+ self.branch = github_pull_request.head.ref
250
+ self.head = find_or_create_commit_from_github_by_sha!(github_pull_request.head.sha, detached: true)
251
+ self.merged_at = github_pull_request.merged_at
252
+ self.base_ref = github_pull_request.base.ref
253
+ self.base_commit = find_or_create_commit_from_github_by_sha!(github_pull_request.base.sha, detached: true)
254
+ end
255
+
256
+ def merge_message
257
+ return title unless merge_requested_by
258
+ "#{title}\n\n#{MERGE_REQUEST_FIELD}: #{merge_requested_by.login}\n"
259
+ end
260
+
261
+ def stale?
262
+ return false unless base_commit
263
+ spec = stack.cached_deploy_spec
264
+ if max_branch_age = spec.max_divergence_age
265
+ return true if Time.now.utc - head.committed_at > max_branch_age
266
+ end
267
+ if commit_count_limit = spec.max_divergence_commits
268
+ return true if comparison.behind_by > commit_count_limit
269
+ end
270
+ false
271
+ end
272
+
273
+ def comparison
274
+ @comparison ||= stack.github_api.compare(
275
+ stack.github_repo_name,
276
+ base_ref,
277
+ head.sha,
278
+ )
279
+ end
280
+
281
+ private
282
+
283
+ def record_merge_status_change
284
+ @merge_status_changed ||= saved_change_to_attribute?(:merge_status)
285
+ end
286
+
287
+ def emit_hooks
288
+ return unless @merge_status_changed
289
+ @merge_status_changed = nil
290
+ Hook.emit('merge', stack, merge_request: self, status: merge_status, stack: stack)
291
+ end
292
+
293
+ def find_or_create_commit_from_github_by_sha!(sha, attributes)
294
+ if commit = stack.commits.by_sha(sha)
295
+ commit
296
+ else
297
+ github_commit = stack.github_api.commit(stack.github_repo_name, sha)
298
+ stack.commits.create_from_github!(github_commit, attributes)
299
+ end
300
+ rescue ActiveRecord::RecordNotUnique
301
+ retry
302
+ end
303
+ end
304
+ end