shipit-engine 0.28.1 → 0.33.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 +41 -2
  3. data/Rakefile +4 -2
  4. data/app/assets/images/archive-solid.svg +1 -0
  5. data/app/assets/images/magic-solid.svg +1 -0
  6. data/app/assets/javascripts/shipit/repositories_search.js.coffee +60 -0
  7. data/app/assets/javascripts/shipit/{search.js.coffee → stack_search.js.coffee} +0 -0
  8. data/app/assets/stylesheets/_pages/_repositories.scss +148 -0
  9. data/app/assets/stylesheets/_pages/_stacks.scss +95 -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 +13 -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 +4 -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 +33 -0
  28. data/app/controllers/shipit/api/stacks_controller.rb +50 -5
  29. data/app/controllers/shipit/api/tasks_controller.rb +6 -5
  30. data/app/controllers/shipit/api_clients_controller.rb +50 -0
  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 +5 -4
  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 +33 -28
  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 +78 -14
  43. data/app/controllers/shipit/status_controller.rb +2 -1
  44. data/app/controllers/shipit/tasks_controller.rb +9 -8
  45. data/app/controllers/shipit/webhooks_controller.rb +5 -132
  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 -0
  51. data/app/helpers/shipit/stacks_helper.rb +9 -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 +4 -1
  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 +3 -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 +1 -0
  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 +5 -4
  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 +2 -1
  70. data/app/jobs/shipit/{mark_deploy_healty_job.rb → 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 -90
  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 +4 -3
  83. data/app/models/concerns/shipit/deferred_touch.rb +4 -3
  84. data/app/models/shipit/anonymous_user.rb +9 -0
  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 +5 -4
  88. data/app/models/shipit/command_line_user.rb +5 -0
  89. data/app/models/shipit/commit.rb +41 -22
  90. data/app/models/shipit/commit_checks.rb +2 -0
  91. data/app/models/shipit/commit_deployment.rb +17 -12
  92. data/app/models/shipit/commit_deployment_status.rb +8 -3
  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 +41 -10
  96. data/app/models/shipit/deploy_spec.rb +38 -7
  97. data/app/models/shipit/deploy_spec/bundler_discovery.rb +2 -1
  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 +1 -0
  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 +58 -0
  106. data/app/models/shipit/duration.rb +3 -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 +8 -5
  111. data/app/models/shipit/membership.rb +3 -2
  112. data/app/models/shipit/merge_request.rb +302 -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 +27 -260
  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 +3 -2
  121. data/app/models/shipit/repository.rb +97 -0
  122. data/app/models/shipit/review_stack.rb +116 -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 +130 -57
  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 +1 -0
  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 +118 -14
  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 +110 -0
  135. data/app/models/shipit/team.rb +2 -1
  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 +15 -8
  139. data/app/models/shipit/variable_definition.rb +1 -0
  140. data/app/models/shipit/webhooks.rb +43 -0
  141. data/app/models/shipit/webhooks/handlers/check_suite_handler.rb +20 -0
  142. data/app/models/shipit/webhooks/handlers/handler.rb +41 -0
  143. data/app/models/shipit/webhooks/handlers/membership_handler.rb +46 -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 +21 -0
  154. data/app/models/shipit/webhooks/handlers/status_handler.rb +27 -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 +12 -5
  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 +4 -3
  171. data/app/validators/subset_validator.rb +1 -0
  172. data/app/views/layouts/_head.html.erb +0 -0
  173. data/app/views/layouts/shipit.html.erb +5 -3
  174. data/app/views/shipit/_variables.html.erb +1 -1
  175. data/app/views/shipit/api_clients/index.html.erb +36 -0
  176. data/app/views/shipit/api_clients/new.html.erb +33 -0
  177. data/app/views/shipit/api_clients/show.html.erb +35 -0
  178. data/app/views/shipit/ccmenu/project.xml.builder +2 -1
  179. data/app/views/shipit/deploys/show.html.erb +2 -2
  180. data/app/views/shipit/merge_requests/_merge_request.html.erb +29 -0
  181. data/app/views/shipit/{pull_requests → merge_requests}/index.html.erb +2 -2
  182. data/app/views/shipit/merge_requests/merge_requests/_pull_request.html.erb +29 -0
  183. data/app/views/shipit/merge_requests/merge_requests/index.html.erb +20 -0
  184. data/app/views/shipit/merge_status/_merge_queue_button.html.erb +3 -3
  185. data/app/views/shipit/merge_status/backlogged.html.erb +1 -1
  186. data/app/views/shipit/merge_status/failure.html.erb +1 -1
  187. data/app/views/shipit/merge_status/locked.html.erb +1 -1
  188. data/app/views/shipit/merge_status/logged_out.erb +1 -1
  189. data/app/views/shipit/merge_status/success.html.erb +2 -2
  190. data/app/views/shipit/repositories/_header.html.erb +19 -0
  191. data/app/views/shipit/repositories/index.html.erb +31 -0
  192. data/app/views/shipit/repositories/new.html.erb +23 -0
  193. data/app/views/shipit/repositories/settings.html.erb +53 -0
  194. data/app/views/shipit/repositories/show.html.erb +30 -0
  195. data/app/views/shipit/stacks/_banners.html.erb +13 -0
  196. data/app/views/shipit/stacks/_header.html.erb +30 -12
  197. data/app/views/shipit/stacks/_links.html.erb +1 -0
  198. data/app/views/shipit/stacks/_stack.html.erb +8 -0
  199. data/app/views/shipit/stacks/all_tasks.html.erb +28 -0
  200. data/app/views/shipit/stacks/index.html.erb +9 -3
  201. data/app/views/shipit/stacks/settings.html.erb +22 -3
  202. data/app/views/shipit/stacks/show.html.erb +1 -1
  203. data/app/views/shipit/stacks/statistics.html.erb +82 -0
  204. data/app/views/shipit/tasks/_task_output.html.erb +1 -1
  205. data/app/views/shipit/tasks/show.html.erb +1 -1
  206. data/config/initializers/inflections.rb +2 -1
  207. data/config/locales/en.yml +18 -5
  208. data/config/routes.rb +29 -7
  209. data/db/migrate/20191209231045_create_shipit_repositories.rb +12 -0
  210. data/db/migrate/20191209231307_add_repository_reference_to_stacks.rb +15 -0
  211. data/db/migrate/20191216162728_backfill_repository_data.rb +22 -0
  212. data/db/migrate/20191216163010_remove_repository_information_from_stacks.rb +20 -0
  213. data/db/migrate/20191219205202_add_archived_since_to_stacks.rb +6 -0
  214. data/db/migrate/20200102175621_optional_task_commits.rb +6 -0
  215. data/db/migrate/20200109132519_add_sha_to_commit_deployments.rb +5 -0
  216. data/db/migrate/20200226211925_add_index_to_tasks_status.rb +5 -0
  217. data/db/migrate/20200427135152_add_pull_request_head_sha_to_commit.rb +5 -0
  218. data/db/migrate/20200615181558_add_rollback_once_aborted_to.rb +5 -0
  219. data/db/migrate/20200706145406_add_review_stacks.rb +12 -0
  220. data/db/migrate/20200804144639_rename_pull_request_to_merge_request.rb +7 -0
  221. data/db/migrate/20200804161512_rename_commits_pull_request_id_to_merge_request_id.rb +5 -0
  222. data/db/migrate/20200813134712_recreate_shipit_pull_requests.rb +22 -0
  223. data/db/migrate/20200813194056_create_pull_request_assignments.rb +8 -0
  224. data/db/migrate/20201001125502_add_provision_pr_stacks_flag_to_repositories.rb +7 -0
  225. data/db/migrate/20201008145809_add_retry_attempt_to_tasks.rb +5 -0
  226. data/db/migrate/20201008152744_add_max_retries_to_tasks.rb +5 -0
  227. data/lib/shipit.rb +23 -3
  228. data/lib/shipit/cast_value.rb +1 -0
  229. data/lib/shipit/command.rb +14 -18
  230. data/lib/shipit/commands.rb +5 -4
  231. data/lib/shipit/csv_serializer.rb +1 -0
  232. data/lib/shipit/deploy_commands.rb +1 -0
  233. data/lib/shipit/engine.rb +11 -2
  234. data/lib/shipit/environment_variables.rb +11 -1
  235. data/lib/shipit/first_parent_commits_iterator.rb +1 -0
  236. data/lib/shipit/flock.rb +1 -0
  237. data/lib/shipit/github_app.rb +41 -10
  238. data/lib/shipit/github_http_cache_middleware.rb +1 -0
  239. data/lib/shipit/null_serializer.rb +1 -0
  240. data/lib/shipit/octokit_check_runs.rb +3 -2
  241. data/lib/shipit/octokit_iterator.rb +3 -2
  242. data/lib/shipit/paginator.rb +3 -2
  243. data/lib/shipit/review_stack_commands.rb +8 -0
  244. data/lib/shipit/rollback_commands.rb +1 -0
  245. data/lib/shipit/same_site_cookie_middleware.rb +29 -0
  246. data/lib/shipit/simple_message_verifier.rb +1 -0
  247. data/lib/shipit/stack_commands.rb +12 -4
  248. data/lib/shipit/stat.rb +1 -0
  249. data/lib/shipit/task_commands.rb +23 -14
  250. data/lib/shipit/version.rb +2 -1
  251. data/lib/snippets/release-gem +5 -1
  252. data/lib/tasks/cron.rake +13 -2
  253. data/lib/tasks/dev.rake +3 -2
  254. data/lib/tasks/shipit.rake +16 -17
  255. data/lib/tasks/teams.rake +1 -0
  256. data/test/controllers/api/base_controller_test.rb +3 -2
  257. data/test/controllers/api/ccmenu_controller_test.rb +9 -8
  258. data/test/controllers/api/commits_controller_test.rb +3 -2
  259. data/test/controllers/api/deploys_controller_test.rb +15 -14
  260. data/test/controllers/api/hooks_controller_test.rb +8 -7
  261. data/test/controllers/api/locks_controller_test.rb +7 -6
  262. data/test/controllers/api/{pull_requests_controller_test.rb → merge_requests_controller_test.rb} +17 -16
  263. data/test/controllers/api/outputs_controller_test.rb +4 -2
  264. data/test/controllers/api/release_statuses_controller_test.rb +2 -1
  265. data/test/controllers/api/rollback_controller_test.rb +113 -0
  266. data/test/controllers/api/stacks_controller_test.rb +65 -16
  267. data/test/controllers/api/tasks_controller_test.rb +13 -12
  268. data/test/controllers/api_clients_controller_test.rb +104 -0
  269. data/test/controllers/ccmenu_controller_test.rb +4 -3
  270. data/test/controllers/commit_checks_controller_test.rb +4 -3
  271. data/test/controllers/commits_controller_test.rb +3 -2
  272. data/test/controllers/deploys_controller_test.rb +33 -22
  273. data/test/controllers/github_authentication_controller_test.rb +1 -0
  274. data/test/controllers/merge_requests_controller_test.rb +32 -0
  275. data/test/controllers/merge_status_controller_test.rb +27 -9
  276. data/test/controllers/release_statuses_controller_test.rb +3 -2
  277. data/test/controllers/repositories_controller_test.rb +71 -0
  278. data/test/controllers/rollbacks_controller_test.rb +9 -8
  279. data/test/controllers/stacks_controller_test.rb +72 -15
  280. data/test/controllers/status_controller_test.rb +1 -0
  281. data/test/controllers/tasks_controller_test.rb +33 -20
  282. data/test/controllers/webhooks_controller_test.rb +36 -9
  283. data/test/dummy/config/application.rb +7 -2
  284. data/test/dummy/config/environments/development.rb +23 -6
  285. data/test/dummy/config/environments/test.rb +2 -5
  286. data/test/dummy/db/schema.rb +76 -24
  287. data/test/dummy/db/seeds.rb +30 -16
  288. data/test/fixtures/payloads/check_suite_master.json +2 -2
  289. data/test/fixtures/payloads/invalid_pull_request.json +117 -0
  290. data/test/fixtures/payloads/provision_disabled_pull_request.json +454 -0
  291. data/test/fixtures/payloads/pull_request_assigned.json +480 -0
  292. data/test/fixtures/payloads/pull_request_closed.json +454 -0
  293. data/test/fixtures/payloads/pull_request_labeled.json +461 -0
  294. data/test/fixtures/payloads/pull_request_opened.json +454 -0
  295. data/test/fixtures/payloads/pull_request_reopened.json +454 -0
  296. data/test/fixtures/payloads/pull_request_unlabeled.json +454 -0
  297. data/test/fixtures/payloads/pull_request_with_no_repo.json +454 -0
  298. data/test/fixtures/shipit/commit_deployment_statuses.yml +4 -4
  299. data/test/fixtures/shipit/commit_deployments.yml +8 -8
  300. data/test/fixtures/shipit/commits.yml +52 -1
  301. data/test/fixtures/shipit/merge_requests.yml +141 -0
  302. data/test/fixtures/shipit/pull_request_assignments.yml +3 -0
  303. data/test/fixtures/shipit/pull_requests.yml +10 -131
  304. data/test/fixtures/shipit/repositories.yml +28 -0
  305. data/test/fixtures/shipit/stacks.yml +335 -30
  306. data/test/fixtures/shipit/statuses.yml +9 -0
  307. data/test/fixtures/shipit/tasks.yml +69 -3
  308. data/test/fixtures/shipit/users.yml +7 -0
  309. data/test/fixtures/timeout +2 -1
  310. data/test/helpers/api_helper.rb +1 -0
  311. data/test/helpers/fixture_aliases_helper.rb +1 -0
  312. data/test/helpers/hooks_helper.rb +2 -1
  313. data/test/helpers/json_helper.rb +15 -11
  314. data/test/helpers/links_helper.rb +4 -3
  315. data/test/helpers/payloads_helper.rb +5 -0
  316. data/test/helpers/queries_helper.rb +3 -2
  317. data/test/jobs/cache_deploy_spec_job_test.rb +2 -1
  318. data/test/jobs/chunk_rollup_job_test.rb +16 -1
  319. data/test/jobs/deliver_hook_job_test.rb +1 -0
  320. data/test/jobs/destroy_repository_job_test.rb +27 -0
  321. data/test/jobs/destroy_stack_job_test.rb +10 -0
  322. data/test/jobs/emit_event_job_test.rb +2 -1
  323. data/test/jobs/fetch_commit_stats_job_test.rb +1 -0
  324. data/test/jobs/fetch_deployed_revision_job_test.rb +1 -0
  325. data/test/jobs/github_sync_job_test.rb +1 -0
  326. data/test/jobs/mark_deploy_healthy_job_test.rb +1 -0
  327. data/test/jobs/perform_task_job_test.rb +12 -11
  328. data/test/jobs/{merge_pull_requests_job_test.rb → process_merge_requests_job_test.rb} +19 -18
  329. data/test/jobs/purge_old_deliveries_job_test.rb +1 -0
  330. data/test/jobs/reap_dead_tasks_job_test.rb +68 -0
  331. data/test/jobs/refresh_github_user_job_test.rb +1 -0
  332. data/test/jobs/refresh_status_job_test.rb +1 -0
  333. data/test/jobs/unique_job_test.rb +1 -0
  334. data/test/jobs/update_github_last_deployed_ref_job_test.rb +1 -0
  335. data/test/lib/shipit/deploy_commands_test.rb +16 -0
  336. data/test/lib/shipit/task_commands_test.rb +17 -0
  337. data/test/middleware/same_site_cookie_middleware_test.rb +52 -0
  338. data/test/models/api_client_test.rb +1 -0
  339. data/test/models/commit_checks_test.rb +1 -0
  340. data/test/models/commit_deployment_status_test.rb +34 -4
  341. data/test/models/commit_deployment_test.rb +9 -11
  342. data/test/models/commits_test.rb +116 -21
  343. data/test/models/delivery_test.rb +2 -1
  344. data/test/models/deploy_spec_test.rb +103 -65
  345. data/test/models/deploy_stats_test.rb +113 -0
  346. data/test/models/deploys_test.rb +207 -26
  347. data/test/models/duration_test.rb +1 -0
  348. data/test/models/github_hook_test.rb +1 -0
  349. data/test/models/hook_test.rb +20 -16
  350. data/test/models/membership_test.rb +1 -0
  351. data/test/models/{pull_request_test.rb → merge_request_test.rb} +48 -41
  352. data/test/models/pull_request_assignment_test.rb +16 -0
  353. data/test/models/release_statuses_test.rb +1 -0
  354. data/test/models/rollbacks_test.rb +1 -0
  355. data/test/models/shipit/check_run_test.rb +1 -0
  356. data/test/models/shipit/provisioning_handler/base_test.rb +33 -0
  357. data/test/models/shipit/provisioning_handler/unregistered_provisioning_handler_test.rb +49 -0
  358. data/test/models/shipit/provisioning_handler_test.rb +64 -0
  359. data/test/models/shipit/pull_request_test.rb +52 -0
  360. data/test/models/shipit/repository_test.rb +81 -0
  361. data/test/models/shipit/review_stack_provision_status_test.rb +77 -0
  362. data/test/models/shipit/review_stack_provisioning_queue_test.rb +63 -0
  363. data/test/models/shipit/review_stack_test.rb +59 -0
  364. data/test/models/{stacks_test.rb → shipit/stacks_test.rb} +120 -60
  365. data/test/models/shipit/webhooks/handlers/pull_request/assigned_handler_test.rb +45 -0
  366. data/test/models/shipit/webhooks/handlers/pull_request/closed_handler_test.rb +192 -0
  367. data/test/models/shipit/webhooks/handlers/pull_request/edited_handler_test.rb +47 -0
  368. data/test/models/shipit/webhooks/handlers/pull_request/label_capturing_handler_test.rb +209 -0
  369. data/test/models/shipit/webhooks/handlers/pull_request/labeled_handler_test.rb +332 -0
  370. data/test/models/shipit/webhooks/handlers/pull_request/opened_handler_test.rb +238 -0
  371. data/test/models/shipit/webhooks/handlers/pull_request/reopened_handler_test.rb +282 -0
  372. data/test/models/shipit/webhooks/handlers/pull_request/review_stack_adapter_test.rb +83 -0
  373. data/test/models/shipit/webhooks/handlers/pull_request/unlabeled_handler_test.rb +324 -0
  374. data/test/models/shipit/webhooks/handlers_test.rb +27 -0
  375. data/test/models/status/group_test.rb +1 -0
  376. data/test/models/status/missing_test.rb +1 -0
  377. data/test/models/status_test.rb +1 -0
  378. data/test/models/task_definitions_test.rb +9 -8
  379. data/test/models/tasks_test.rb +59 -1
  380. data/test/models/team_test.rb +4 -2
  381. data/test/models/undeployed_commits_test.rb +14 -0
  382. data/test/models/users_test.rb +13 -5
  383. data/test/serializers/shipit/pull_request_serializer_test.rb +29 -0
  384. data/test/test_command_integration.rb +3 -2
  385. data/test/test_helper.rb +37 -32
  386. data/test/unit/anonymous_user_serializer_test.rb +14 -0
  387. data/test/unit/command_test.rb +15 -10
  388. data/test/unit/commands_test.rb +1 -0
  389. data/test/unit/commit_serializer_test.rb +16 -0
  390. data/test/unit/csv_serializer_test.rb +3 -2
  391. data/test/unit/deploy_commands_test.rb +14 -4
  392. data/test/unit/deploy_serializer_test.rb +17 -0
  393. data/test/unit/environment_variables_test.rb +5 -4
  394. data/test/unit/github_app_test.rb +137 -0
  395. data/test/unit/github_url_helper_test.rb +6 -0
  396. data/test/unit/rollback_commands_test.rb +2 -1
  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 +1 -0
  400. data/test/unit/user_serializer_test.rb +14 -0
  401. data/test/unit/variable_definition_test.rb +1 -0
  402. metadata +334 -169
  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
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+ module Shipit
3
+ class ReapDeadTasksJob < BackgroundJob
4
+ include BackgroundJob::Unique
5
+ queue_as :default
6
+
7
+ def perform
8
+ Rails.logger.info("Reaping #{zombie_tasks.size} running tasks.")
9
+ zombie_tasks.each do |task|
10
+ Rails.logger.info("Reaping task #{task.id}: #{task.title}")
11
+ task.report_dead!
12
+ end
13
+ end
14
+
15
+ private
16
+
17
+ def zombie_tasks
18
+ @zombie_tasks ||= Task.zombies
19
+ end
20
+ end
21
+ end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class RefreshCheckRunsJob < BackgroundJob
3
4
  queue_as :default
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class RefreshGithubUserJob < BackgroundJob
3
4
  queue_as :default
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+ module Shipit
3
+ class RefreshMergeRequestJob < BackgroundJob
4
+ queue_as :default
5
+
6
+ def perform(merge_request)
7
+ merge_request.refresh!
8
+ ProcessMergeRequestsJob.perform_later(merge_request.stack)
9
+ end
10
+ end
11
+ end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class RefreshStatusesJob < BackgroundJob
3
4
  queue_as :default
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class SetupGithubHookJob < BackgroundJob
3
4
  queue_as :default
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class UpdateEstimatedDeployDurationJob < BackgroundJob
3
4
  queue_as :default
@@ -1,10 +1,11 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class UpdateGithubLastDeployedRefJob < BackgroundJob
3
4
  queue_as :default
4
5
 
5
6
  # We do not prefix 'refs/' because Octokit methods will do this automatically.
6
- BRANCH_REF_PREFIX = 'heads'.freeze
7
- DEPLOY_PREFIX = 'shipit-deploy'.freeze
7
+ BRANCH_REF_PREFIX = 'heads'
8
+ DEPLOY_PREFIX = "#{Shipit.app_name.downcase}-deploy"
8
9
 
9
10
  def perform(stack)
10
11
  stack_sha = stack.last_successful_deploy_commit&.sha
@@ -33,7 +34,7 @@ module Shipit
33
34
  client.update_ref(repo_name, ref, new_sha)
34
35
  rescue Octokit::UnprocessableEntity => e
35
36
  error_msg = e.message
36
- if error_msg.include? "Reference does not exist"
37
+ if error_msg.include?("Reference does not exist")
37
38
  create_ref(client: client, repo_name: repo_name, ref: ref, sha: new_sha)
38
39
  else
39
40
  raise
@@ -1,10 +1,11 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  module DeferredTouch
3
4
  extend ActiveSupport::Concern
4
5
 
5
- SET_KEY = 'shipit:deferred_touches'.freeze
6
- TMP_KEY = "#{SET_KEY}:updating".freeze
7
- CACHE_KEY = "#{SET_KEY}:scheduled".freeze
6
+ SET_KEY = 'shipit:deferred_touches'
7
+ TMP_KEY = "#{SET_KEY}:updating"
8
+ CACHE_KEY = "#{SET_KEY}:scheduled"
8
9
  THROTTLE_TTL = 1.second
9
10
 
10
11
  included do
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class AnonymousUser
3
4
  def present?
@@ -34,6 +35,10 @@ module Shipit
34
35
  Shipit.authentication_disabled?
35
36
  end
36
37
 
38
+ def repositories_contributed_to
39
+ []
40
+ end
41
+
37
42
  def stacks_contributed_to
38
43
  []
39
44
  end
@@ -54,5 +59,9 @@ module Shipit
54
59
  def github_api
55
60
  Shipit.github.api
56
61
  end
62
+
63
+ def serializer_class
64
+ AnonymousUserSerializer
65
+ end
57
66
  end
58
67
  end
@@ -1,5 +1,6 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class ApiClient < ActiveRecord::Base
3
+ class ApiClient < Record
3
4
  InsufficientPermission = Class.new(StandardError)
4
5
 
5
6
  belongs_to :creator, class_name: 'User'
@@ -16,7 +17,7 @@ module Shipit
16
17
  read:hook
17
18
  write:hook
18
19
  ).freeze
19
- validates :permissions, subset: {of: PERMISSIONS}
20
+ validates :permissions, subset: { of: PERMISSIONS }
20
21
 
21
22
  class << self
22
23
  def authenticate(token)
@@ -1,5 +1,6 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class ApplicationRecord < ActiveRecord::Base
3
+ class ApplicationRecord < Record
3
4
  self.abstract_class = true
4
5
  end
5
6
  end
@@ -1,6 +1,7 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class CheckRun < ApplicationRecord
3
- CONCLUSIONS = %w(success failure neutral cancelled timed_out action_required).freeze
4
+ CONCLUSIONS = %w(success failure neutral cancelled timed_out action_required stale skipped).freeze
4
5
  include DeferredTouch
5
6
  include Status::Common
6
7
 
@@ -9,7 +10,7 @@ module Shipit
9
10
 
10
11
  deferred_touch commit: :updated_at
11
12
 
12
- validates :conclusion, inclusion: {in: CONCLUSIONS, allow_nil: true}
13
+ validates :conclusion, inclusion: { in: CONCLUSIONS, allow_nil: true }
13
14
 
14
15
  after_create :enable_ci_on_stack
15
16
 
@@ -43,9 +44,9 @@ module Shipit
43
44
  case conclusion
44
45
  when nil, 'action_required'
45
46
  'pending'
46
- when 'success', 'neutral'
47
+ when 'success', 'neutral', 'skipped'
47
48
  'success'
48
- when 'failure', 'cancelled'
49
+ when 'failure', 'cancelled', 'stale'
49
50
  'failure'
50
51
  when 'timed_out'
51
52
  'error'
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class CommandLineUser
3
4
  def present?
@@ -34,6 +35,10 @@ module Shipit
34
35
  Shipit.authentication_disabled?
35
36
  end
36
37
 
38
+ def repositories_contributed_to
39
+ []
40
+ end
41
+
37
42
  def stacks_contributed_to
38
43
  []
39
44
  end
@@ -1,7 +1,10 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class Commit < ActiveRecord::Base
3
+ class Commit < Record
3
4
  include DeferredTouch
4
5
 
6
+ RECENT_COMMIT_THRESHOLD = 10.seconds
7
+
5
8
  AmbiguousRevision = Class.new(StandardError)
6
9
 
7
10
  belongs_to :stack
@@ -10,11 +13,11 @@ module Shipit
10
13
  has_many :check_runs, -> { order(created_at: :desc) }, dependent: :destroy, inverse_of: :commit
11
14
  has_many :commit_deployments, dependent: :destroy
12
15
  has_many :release_statuses, dependent: :destroy
13
- belongs_to :pull_request, inverse_of: :merge_commit, optional: true
16
+ belongs_to :merge_request, inverse_of: :merge_commit, optional: true
14
17
 
15
18
  deferred_touch stack: :updated_at
16
19
 
17
- before_create :identify_pull_request
20
+ before_create :identify_merge_request
18
21
  after_commit { broadcast_update }
19
22
  after_create { stack.update_undeployed_commits_count }
20
23
 
@@ -90,7 +93,7 @@ module Shipit
90
93
  committer = User.find_or_create_committer_from_github_commit(commit)
91
94
  committer ||= Anonymous.new
92
95
 
93
- new(
96
+ record = new(
94
97
  sha: commit.sha,
95
98
  message: commit.commit.message,
96
99
  author: author,
@@ -100,6 +103,20 @@ module Shipit
100
103
  additions: commit.stats&.additions,
101
104
  deletions: commit.stats&.deletions,
102
105
  )
106
+
107
+ if record.pull_request?
108
+ record.pull_request_head_sha = commit.parents.last.sha
109
+ end
110
+
111
+ record
112
+ end
113
+
114
+ def message=(message)
115
+ limit = self.class.columns_hash['message'].limit
116
+ if limit && message && message.bytesize > limit
117
+ message = message.truncate_bytes(limit)
118
+ end
119
+ super(message)
103
120
  end
104
121
 
105
122
  def reload(*)
@@ -126,7 +143,9 @@ module Shipit
126
143
  end
127
144
 
128
145
  def refresh_statuses!
129
- github_statuses = stack.handle_github_redirections { Shipit.github.api.statuses(github_repo_name, sha) }
146
+ github_statuses = stack.handle_github_redirections do
147
+ Shipit.github.api.statuses(github_repo_name, sha, per_page: 100)
148
+ end
130
149
  github_statuses.each do |status|
131
150
  create_status_from_github!(status)
132
151
  end
@@ -177,13 +196,7 @@ module Shipit
177
196
  def active?
178
197
  return false unless stack.active_task?
179
198
 
180
- active_task = stack.active_task
181
-
182
- if active_task.since_commit == active_task.until_commit
183
- id == active_task.since_commit.id
184
- else
185
- id > active_task.since_commit.id && id <= active_task.until_commit.id
186
- end
199
+ stack.active_task.includes_commit?(self)
187
200
  end
188
201
 
189
202
  def deployable?
@@ -242,7 +255,9 @@ module Shipit
242
255
 
243
256
  def schedule_continuous_delivery
244
257
  return unless deployable? && stack.continuous_deployment? && stack.deployable?
245
- ContinuousDeliveryJob.perform_later(stack)
258
+ # This buffer is to allow for statuses and checks to be refreshed before evaluating if the commit is deployable
259
+ # - e.g. if the commit was fast-forwarded with already passing CI.
260
+ ContinuousDeliveryJob.set(wait: RECENT_COMMIT_THRESHOLD).perform_later(stack)
246
261
  end
247
262
 
248
263
  def github_commit
@@ -272,13 +287,13 @@ module Shipit
272
287
  stack.deploys.unsuccessful.where(until_commit_id: id).any?
273
288
  end
274
289
 
275
- def identify_pull_request
290
+ def identify_merge_request
276
291
  return unless message_parser.pull_request?
277
- if pull_request = stack.pull_requests.find_by(number: message_parser.pull_request_number)
278
- self.pull_request = pull_request
279
- self.pull_request_number = pull_request.number
280
- self.pull_request_title = pull_request.title
281
- self.author = pull_request.merge_requested_by if pull_request.merge_requested_by
292
+ if merge_request = stack.merge_requests.find_by(number: message_parser.pull_request_number)
293
+ self.merge_request = merge_request
294
+ self.pull_request_number = merge_request.number
295
+ self.pull_request_title = merge_request.title
296
+ self.author = merge_request.merge_requested_by if merge_request.merge_requested_by
282
297
  end
283
298
 
284
299
  self.pull_request_number = message_parser.pull_request_number unless self[:pull_request_number]
@@ -286,8 +301,8 @@ module Shipit
286
301
  end
287
302
 
288
303
  def deploy_requested_at
289
- if pull_request&.merged?
290
- pull_request.merge_requested_at
304
+ if merge_request&.merged?
305
+ merge_request.merge_requested_at
291
306
  else
292
307
  created_at
293
308
  end
@@ -311,6 +326,10 @@ module Shipit
311
326
  update!(locked: false, lock_author: nil)
312
327
  end
313
328
 
329
+ def recently_pushed?
330
+ created_at > RECENT_COMMIT_THRESHOLD.ago
331
+ end
332
+
314
333
  private
315
334
 
316
335
  def message_parser
@@ -326,7 +345,7 @@ module Shipit
326
345
  new_status = status
327
346
 
328
347
  unless already_deployed
329
- payload = {commit: self, stack: stack, status: new_status.state}
348
+ payload = { commit: self, stack: stack, status: new_status.state }
330
349
  if previous_status != new_status
331
350
  Hook.emit(:commit_status, stack, payload.merge(commit_status: new_status))
332
351
  end
@@ -1,3 +1,4 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
3
  class CommitChecks < EphemeralCommitChecks
3
4
  OUTPUT_TTL = 10.minutes.to_i
@@ -5,6 +6,7 @@ module Shipit
5
6
 
6
7
  def initialize(commit)
7
8
  @commit = commit
9
+ super(commit)
8
10
  end
9
11
 
10
12
  def synchronize(&block)
@@ -1,6 +1,6 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class CommitDeployment < ActiveRecord::Base
3
- belongs_to :commit
3
+ class CommitDeployment < Record
4
4
  belongs_to :task
5
5
  has_many :statuses, dependent: :destroy, class_name: 'CommitDeploymentStatus'
6
6
 
@@ -9,11 +9,10 @@ module Shipit
9
9
  delegate :stack, :author, to: :task
10
10
 
11
11
  def create_on_github!
12
- return unless commit.pull_request?
13
-
14
12
  create_deployment_on_github!
15
13
  statuses.order(id: :asc).each(&:create_on_github!)
16
- rescue Octokit::NotFound, Octokit::Forbidden
14
+ rescue Octokit::NotFound, Octokit::Forbidden => error
15
+ Rails.logger.warn("Got #{error.class.name} creating deployment or statuses: #{error.message}")
17
16
  # If no one can create the deployment we can only give up
18
17
  end
19
18
 
@@ -34,25 +33,31 @@ module Shipit
34
33
  update!(github_id: response.id, api_url: response.url)
35
34
  end
36
35
 
37
- def pull_request_head
38
- pull_request = Shipit.github.api.pull_request(stack.github_repo_name, commit.pull_request_number)
39
- pull_request.head.sha
40
- end
41
-
42
36
  def schedule_create_on_github
43
37
  CreateOnGithubJob.perform_later(self)
44
38
  end
45
39
 
40
+ def short_sha
41
+ sha[0..9]
42
+ end
43
+
46
44
  private
47
45
 
48
46
  def create_deployment_on_github(client)
49
47
  client.create_deployment(
50
48
  stack.github_repo_name,
51
- pull_request_head,
49
+ sha,
52
50
  auto_merge: false,
53
51
  required_contexts: [],
54
- description: "Via Shipit",
52
+ description: "Via #{Shipit.app_name}",
55
53
  environment: stack.environment,
54
+ payload: {
55
+ shipit: {
56
+ task_id: task.id,
57
+ from_sha: task.since_commit.sha,
58
+ to_sha: task.until_commit.sha,
59
+ },
60
+ }.to_json,
56
61
  )
57
62
  end
58
63
  end
@@ -1,5 +1,8 @@
1
+ # frozen_string_literal: true
1
2
  module Shipit
2
- class CommitDeploymentStatus < ActiveRecord::Base
3
+ class CommitDeploymentStatus < Record
4
+ DESCRIPTION_CHARACTER_LIMIT_ON_GITHUB = 140
5
+
3
6
  belongs_to :commit_deployment
4
7
 
5
8
  after_commit :schedule_create_on_github, on: :create
@@ -25,7 +28,7 @@ module Shipit
25
28
  def description
26
29
  I18n.t(
27
30
  "deployment_description.#{task_type}.#{status}",
28
- sha: task.until_commit.sha,
31
+ sha: task.until_commit.short_sha,
29
32
  author: task.author.login,
30
33
  stack: stack.to_param,
31
34
  )
@@ -45,8 +48,10 @@ module Shipit
45
48
  client.create_deployment_status(
46
49
  commit_deployment.api_url,
47
50
  status,
51
+ accept: 'application/vnd.github.flash-preview+json',
48
52
  target_url: url_helpers.stack_deploy_url(stack, task),
49
- description: description,
53
+ description: description.truncate(DESCRIPTION_CHARACTER_LIMIT_ON_GITHUB),
54
+ environment_url: stack.deploy_url,
50
55
  )
51
56
  end
52
57