kuroko2 0.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (338) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +15 -0
  4. data/Rakefile +26 -0
  5. data/app/assets/images/kuroko2/avatar.png +0 -0
  6. data/app/assets/images/kuroko2/kuroko-logo-horizontal.png +0 -0
  7. data/app/assets/javascripts/kuroko2/application.js +28 -0
  8. data/app/assets/javascripts/kuroko2/bootstrap.js +2363 -0
  9. data/app/assets/javascripts/kuroko2/definition_linker.js +20 -0
  10. data/app/assets/javascripts/kuroko2/instance_linker.js +10 -0
  11. data/app/assets/javascripts/kuroko2/job_definition_stats.js +67 -0
  12. data/app/assets/javascripts/kuroko2/job_definitions.js +79 -0
  13. data/app/assets/javascripts/kuroko2/job_instances.js +88 -0
  14. data/app/assets/javascripts/kuroko2/job_timelines.js.coffee +24 -0
  15. data/app/assets/javascripts/kuroko2/narrow_job_definitions.js +30 -0
  16. data/app/assets/stylesheets/kuroko2/admin-lte.css +3402 -0
  17. data/app/assets/stylesheets/kuroko2/application.scss +86 -0
  18. data/app/assets/stylesheets/kuroko2/bootstrap.css +6799 -0
  19. data/app/assets/stylesheets/kuroko2/fonts.scss +1 -0
  20. data/app/assets/stylesheets/kuroko2/job_definitions.scss +19 -0
  21. data/app/assets/stylesheets/kuroko2/users.scss +7 -0
  22. data/app/controllers/kuroko2/api/application_controller.rb +49 -0
  23. data/app/controllers/kuroko2/api/job_instances_controller.rb +46 -0
  24. data/app/controllers/kuroko2/api/stats_controller.rb +28 -0
  25. data/app/controllers/kuroko2/application_controller.rb +43 -0
  26. data/app/controllers/kuroko2/dashboard_controller.rb +27 -0
  27. data/app/controllers/kuroko2/execution_logs_controller.rb +19 -0
  28. data/app/controllers/kuroko2/executions_controller.rb +28 -0
  29. data/app/controllers/kuroko2/job_definition_stats_controller.rb +54 -0
  30. data/app/controllers/kuroko2/job_definitions_controller.rb +100 -0
  31. data/app/controllers/kuroko2/job_instances_controller.rb +87 -0
  32. data/app/controllers/kuroko2/job_schedules_controller.rb +39 -0
  33. data/app/controllers/kuroko2/job_suspend_schedules_controller.rb +38 -0
  34. data/app/controllers/kuroko2/job_timelines_controller.rb +56 -0
  35. data/app/controllers/kuroko2/logs_controller.rb +15 -0
  36. data/app/controllers/kuroko2/sessions_controller.rb +32 -0
  37. data/app/controllers/kuroko2/stars_controller.rb +30 -0
  38. data/app/controllers/kuroko2/tokens_controller.rb +47 -0
  39. data/app/controllers/kuroko2/users_controller.rb +62 -0
  40. data/app/controllers/kuroko2/workers_controller.rb +5 -0
  41. data/app/errors/http/bad_request.rb +4 -0
  42. data/app/errors/http/forbidden.rb +4 -0
  43. data/app/errors/http/unauthorized.rb +4 -0
  44. data/app/helpers/kuroko2/application_helper.rb +8 -0
  45. data/app/helpers/kuroko2/dashboard_helper.rb +4 -0
  46. data/app/helpers/kuroko2/executions_helper.rb +4 -0
  47. data/app/helpers/kuroko2/job_definitions_helper.rb +38 -0
  48. data/app/helpers/kuroko2/job_instances_helper.rb +71 -0
  49. data/app/helpers/kuroko2/job_schedules_helper.rb +4 -0
  50. data/app/helpers/kuroko2/logs_helper.rb +4 -0
  51. data/app/helpers/kuroko2/sessions_helper.rb +4 -0
  52. data/app/helpers/kuroko2/stars_helper.rb +4 -0
  53. data/app/helpers/kuroko2/tokens_helper.rb +4 -0
  54. data/app/helpers/kuroko2/users_helper.rb +4 -0
  55. data/app/helpers/kuroko2/workers_helper.rb +4 -0
  56. data/app/jobs/kuroko2/application_job.rb +4 -0
  57. data/app/mailers/kuroko2/application_mailer.rb +6 -0
  58. data/app/mailers/kuroko2/notifications.rb +63 -0
  59. data/app/models/concerns/kuroko2/table_name_customizable.rb +16 -0
  60. data/app/models/kuroko2/admin_assignment.rb +6 -0
  61. data/app/models/kuroko2/api/application_resource.rb +10 -0
  62. data/app/models/kuroko2/api/job_instance_resource.rb +7 -0
  63. data/app/models/kuroko2/application_record.rb +5 -0
  64. data/app/models/kuroko2/execution.rb +55 -0
  65. data/app/models/kuroko2/job_definition.rb +147 -0
  66. data/app/models/kuroko2/job_definition_tag.rb +6 -0
  67. data/app/models/kuroko2/job_instance.rb +118 -0
  68. data/app/models/kuroko2/job_schedule.rb +93 -0
  69. data/app/models/kuroko2/job_suspend_schedule.rb +35 -0
  70. data/app/models/kuroko2/log.rb +3 -0
  71. data/app/models/kuroko2/memory_consumption_log.rb +49 -0
  72. data/app/models/kuroko2/memory_expectancy.rb +21 -0
  73. data/app/models/kuroko2/process_signal.rb +14 -0
  74. data/app/models/kuroko2/star.rb +6 -0
  75. data/app/models/kuroko2/tag.rb +8 -0
  76. data/app/models/kuroko2/tick.rb +12 -0
  77. data/app/models/kuroko2/token.rb +101 -0
  78. data/app/models/kuroko2/user.rb +48 -0
  79. data/app/models/kuroko2/worker.rb +12 -0
  80. data/app/views/kaminari/history/_paginator.html.slim +15 -0
  81. data/app/views/kaminari/list/_paginator.html.slim +17 -0
  82. data/app/views/kuroko2/dashboard/_taglist.html.slim +13 -0
  83. data/app/views/kuroko2/dashboard/index.html.slim +49 -0
  84. data/app/views/kuroko2/execution_logs/index.json.jbuilder +16 -0
  85. data/app/views/kuroko2/executions/index.html.slim +35 -0
  86. data/app/views/kuroko2/job_definition_stats/execution_time.json.jbuilder +13 -0
  87. data/app/views/kuroko2/job_definition_stats/index.html.slim +48 -0
  88. data/app/views/kuroko2/job_definition_stats/memory.json.jbuilder +10 -0
  89. data/app/views/kuroko2/job_definitions/_alert.html.slim +7 -0
  90. data/app/views/kuroko2/job_definitions/_form.html.slim +93 -0
  91. data/app/views/kuroko2/job_definitions/_list.html.slim +26 -0
  92. data/app/views/kuroko2/job_definitions/_search_results.html.slim +51 -0
  93. data/app/views/kuroko2/job_definitions/_taglist.html.slim +13 -0
  94. data/app/views/kuroko2/job_definitions/edit.html.slim +15 -0
  95. data/app/views/kuroko2/job_definitions/index.html.slim +11 -0
  96. data/app/views/kuroko2/job_definitions/new.html.slim +12 -0
  97. data/app/views/kuroko2/job_definitions/show.html.slim +90 -0
  98. data/app/views/kuroko2/job_instances/_instance.html.slim +42 -0
  99. data/app/views/kuroko2/job_instances/index.html.slim +58 -0
  100. data/app/views/kuroko2/job_instances/show.html.slim +19 -0
  101. data/app/views/kuroko2/job_instances/working.html.slim +37 -0
  102. data/app/views/kuroko2/job_schedules/index.html.slim +32 -0
  103. data/app/views/kuroko2/job_suspend_schedules/index.html.slim +27 -0
  104. data/app/views/kuroko2/job_timelines/dataset.json.jbuilder +11 -0
  105. data/app/views/kuroko2/job_timelines/index.html.slim +31 -0
  106. data/app/views/kuroko2/kaminari/history/_paginator.html.slim +15 -0
  107. data/app/views/kuroko2/kaminari/list/_paginator.html.slim +17 -0
  108. data/app/views/kuroko2/layouts/application.html.slim +68 -0
  109. data/app/views/kuroko2/logs/index.html.slim +33 -0
  110. data/app/views/kuroko2/notifications/executor_not_assigned.text.erb +22 -0
  111. data/app/views/kuroko2/notifications/job_failure.html.slim +21 -0
  112. data/app/views/kuroko2/notifications/job_failure.text.erb +18 -0
  113. data/app/views/kuroko2/notifications/notify_long_elapsed_time.text.erb +9 -0
  114. data/app/views/kuroko2/notifications/process_absence.text.erb +22 -0
  115. data/app/views/kuroko2/notifications/remind_failure.html.slim +21 -0
  116. data/app/views/kuroko2/notifications/remind_failure.text.erb +10 -0
  117. data/app/views/kuroko2/sessions/new.html.slim +20 -0
  118. data/app/views/kuroko2/tokens/index.html.slim +42 -0
  119. data/app/views/kuroko2/users/index.html.slim +55 -0
  120. data/app/views/kuroko2/users/show.html.slim +76 -0
  121. data/app/views/kuroko2/workers/index.html.slim +40 -0
  122. data/app/views/layouts/kuroko2/application.html.slim +72 -0
  123. data/app/views/layouts/mailer.html.erb +13 -0
  124. data/app/views/layouts/mailer.text.erb +1 -0
  125. data/bin/cleanup_old_instances.rb +9 -0
  126. data/bin/remind_failure.rb +5 -0
  127. data/config/initializers/000_kuroko2.rb +16 -0
  128. data/config/initializers/assets.rb +9 -0
  129. data/config/initializers/garage.rb +2 -0
  130. data/config/initializers/kaminari_config.rb +3 -0
  131. data/config/initializers/omniauth.rb +5 -0
  132. data/config/locales/en.yml +28 -0
  133. data/config/routes.rb +54 -0
  134. data/db/migrate/001_create_job_definitions.rb +22 -0
  135. data/db/migrate/002_create_job_instances.rb +17 -0
  136. data/db/migrate/003_create_job_schedules.rb +14 -0
  137. data/db/migrate/004_create_ticks.rb +7 -0
  138. data/db/migrate/005_create_logs.rb +13 -0
  139. data/db/migrate/006_create_tokens.rb +21 -0
  140. data/db/migrate/007_create_executions.rb +26 -0
  141. data/db/migrate/008_create_process_signals.rb +15 -0
  142. data/db/migrate/009_create_users.rb +20 -0
  143. data/db/migrate/010_create_admin_assignments.rb +12 -0
  144. data/db/migrate/011_create_stars.rb +12 -0
  145. data/db/migrate/012_create_workers.rb +15 -0
  146. data/db/migrate/018_create_job_definition_tags.rb +13 -0
  147. data/db/migrate/019_create_tags.rb +11 -0
  148. data/db/migrate/021_create_memory_expectancies.rb +17 -0
  149. data/db/migrate/025_create_job_suspend_schedules.rb +12 -0
  150. data/lib/kuroko2/command/executor.rb +62 -0
  151. data/lib/kuroko2/command/kill.rb +24 -0
  152. data/lib/kuroko2/command/monitor.rb +109 -0
  153. data/lib/kuroko2/command/shell.rb +163 -0
  154. data/lib/kuroko2/configuration.rb +19 -0
  155. data/lib/kuroko2/engine.rb +59 -0
  156. data/lib/kuroko2/execution_logger/cloud_watch_logs.rb +92 -0
  157. data/lib/kuroko2/execution_logger/void.rb +13 -0
  158. data/lib/kuroko2/execution_logger.rb +20 -0
  159. data/lib/kuroko2/memory_sampler.rb +50 -0
  160. data/lib/kuroko2/return_to_validator.rb +14 -0
  161. data/lib/kuroko2/servers/base.rb +30 -0
  162. data/lib/kuroko2/servers/command_executor.rb +27 -0
  163. data/lib/kuroko2/servers/job_scheduler.rb +25 -0
  164. data/lib/kuroko2/servers/workflow_processor.rb +25 -0
  165. data/lib/kuroko2/util/logger.rb +19 -0
  166. data/lib/kuroko2/util/rails_logger_formatter.rb +9 -0
  167. data/lib/kuroko2/version.rb +3 -0
  168. data/lib/kuroko2/workflow/assertion_error.rb +6 -0
  169. data/lib/kuroko2/workflow/engine.rb +141 -0
  170. data/lib/kuroko2/workflow/engine_error.rb +6 -0
  171. data/lib/kuroko2/workflow/node.rb +124 -0
  172. data/lib/kuroko2/workflow/notifier/concerns/chat_message_builder.rb +39 -0
  173. data/lib/kuroko2/workflow/notifier/hipchat.rb +88 -0
  174. data/lib/kuroko2/workflow/notifier/mail.rb +44 -0
  175. data/lib/kuroko2/workflow/notifier/slack.rb +121 -0
  176. data/lib/kuroko2/workflow/notifier.rb +31 -0
  177. data/lib/kuroko2/workflow/processor.rb +40 -0
  178. data/lib/kuroko2/workflow/scheduler.rb +42 -0
  179. data/lib/kuroko2/workflow/script_parser.rb +66 -0
  180. data/lib/kuroko2/workflow/shell_scanner.rb +34 -0
  181. data/lib/kuroko2/workflow/syntax_error.rb +6 -0
  182. data/lib/kuroko2/workflow/task/auto_skip_error.rb +20 -0
  183. data/lib/kuroko2/workflow/task/base.rb +32 -0
  184. data/lib/kuroko2/workflow/task/env.rb +45 -0
  185. data/lib/kuroko2/workflow/task/execute.rb +128 -0
  186. data/lib/kuroko2/workflow/task/expected_time.rb +9 -0
  187. data/lib/kuroko2/workflow/task/fork.rb +45 -0
  188. data/lib/kuroko2/workflow/task/kuroko_runner.rb +24 -0
  189. data/lib/kuroko2/workflow/task/noop.rb +13 -0
  190. data/lib/kuroko2/workflow/task/queue.rb +27 -0
  191. data/lib/kuroko2/workflow/task/rails_env.rb +24 -0
  192. data/lib/kuroko2/workflow/task/sequence.rb +13 -0
  193. data/lib/kuroko2/workflow/task/sleep.rb +29 -0
  194. data/lib/kuroko2/workflow/task/sub_process.rb +53 -0
  195. data/lib/kuroko2/workflow/task/time_base.rb +44 -0
  196. data/lib/kuroko2/workflow/task/timeout.rb +9 -0
  197. data/lib/kuroko2/workflow/task/wait.rb +143 -0
  198. data/lib/kuroko2.rb +26 -0
  199. data/lib/tasks/kuroko2_tasks.rake +4 -0
  200. data/spec/command/kill_spec.rb +20 -0
  201. data/spec/command/monitor_spec.rb +68 -0
  202. data/spec/command/shell_spec.rb +87 -0
  203. data/spec/controllers/dashboard_controller_spec.rb +5 -0
  204. data/spec/controllers/executions_controller_spec.rb +23 -0
  205. data/spec/controllers/job_definition_stats_controller_spec.rb +95 -0
  206. data/spec/controllers/job_definitions_controller_spec.rb +89 -0
  207. data/spec/controllers/job_instances_controller_spec.rb +62 -0
  208. data/spec/controllers/job_schedules_controller_spec.rb +39 -0
  209. data/spec/controllers/job_suspend_schedules_controller_spec.rb +39 -0
  210. data/spec/controllers/job_timelines_controller_spec.rb +114 -0
  211. data/spec/controllers/logs_controller_spec.rb +5 -0
  212. data/spec/controllers/sessions_controller_spec.rb +58 -0
  213. data/spec/controllers/stars_controller_spec.rb +28 -0
  214. data/spec/controllers/tokens_controller_spec.rb +5 -0
  215. data/spec/controllers/users_controller_spec.rb +51 -0
  216. data/spec/controllers/workers_controller_spec.rb +5 -0
  217. data/spec/dummy/Rakefile +6 -0
  218. data/spec/dummy/app/assets/config/manifest.js +5 -0
  219. data/spec/dummy/app/assets/javascripts/application.js +13 -0
  220. data/spec/dummy/app/assets/javascripts/cable.js +13 -0
  221. data/spec/dummy/app/assets/stylesheets/application.css +15 -0
  222. data/spec/dummy/app/channels/application_cable/channel.rb +4 -0
  223. data/spec/dummy/app/channels/application_cable/connection.rb +4 -0
  224. data/spec/dummy/app/controllers/application_controller.rb +3 -0
  225. data/spec/dummy/app/helpers/application_helper.rb +2 -0
  226. data/spec/dummy/app/jobs/application_job.rb +2 -0
  227. data/spec/dummy/app/mailers/application_mailer.rb +4 -0
  228. data/spec/dummy/app/models/application_record.rb +3 -0
  229. data/spec/dummy/app/views/layouts/application.html.erb +14 -0
  230. data/spec/dummy/app/views/layouts/mailer.html.erb +13 -0
  231. data/spec/dummy/app/views/layouts/mailer.text.erb +1 -0
  232. data/spec/dummy/bin/bundle +3 -0
  233. data/spec/dummy/bin/rails +4 -0
  234. data/spec/dummy/bin/rake +4 -0
  235. data/spec/dummy/bin/setup +34 -0
  236. data/spec/dummy/bin/update +29 -0
  237. data/spec/dummy/config/application.rb +25 -0
  238. data/spec/dummy/config/boot.rb +3 -0
  239. data/spec/dummy/config/cable.yml +9 -0
  240. data/spec/dummy/config/database.yml +29 -0
  241. data/spec/dummy/config/environment.rb +5 -0
  242. data/spec/dummy/config/environments/development.rb +54 -0
  243. data/spec/dummy/config/environments/production.rb +86 -0
  244. data/spec/dummy/config/environments/test.rb +42 -0
  245. data/spec/dummy/config/initializers/application_controller_renderer.rb +6 -0
  246. data/spec/dummy/config/initializers/assets.rb +11 -0
  247. data/spec/dummy/config/initializers/backtrace_silencers.rb +7 -0
  248. data/spec/dummy/config/initializers/cookies_serializer.rb +5 -0
  249. data/spec/dummy/config/initializers/filter_parameter_logging.rb +4 -0
  250. data/spec/dummy/config/initializers/inflections.rb +16 -0
  251. data/spec/dummy/config/initializers/mime_types.rb +4 -0
  252. data/spec/dummy/config/initializers/new_framework_defaults.rb +23 -0
  253. data/spec/dummy/config/initializers/session_store.rb +3 -0
  254. data/spec/dummy/config/initializers/wrap_parameters.rb +14 -0
  255. data/spec/dummy/config/kuroko2.yml +35 -0
  256. data/spec/dummy/config/locales/en.yml +23 -0
  257. data/spec/dummy/config/puma.rb +47 -0
  258. data/spec/dummy/config/routes.rb +3 -0
  259. data/spec/dummy/config/secrets.yml +22 -0
  260. data/spec/dummy/config/spring.rb +6 -0
  261. data/spec/dummy/config.ru +5 -0
  262. data/spec/dummy/db/schema.rb +217 -0
  263. data/spec/dummy/lib/dummy_extention.rb +14 -0
  264. data/spec/dummy/lib/kuroko2/workflow/task/custom_task1.rb +13 -0
  265. data/spec/dummy/public/404.html +67 -0
  266. data/spec/dummy/public/422.html +67 -0
  267. data/spec/dummy/public/500.html +66 -0
  268. data/spec/dummy/public/apple-touch-icon-precomposed.png +0 -0
  269. data/spec/dummy/public/apple-touch-icon.png +0 -0
  270. data/spec/dummy/public/favicon.ico +0 -0
  271. data/spec/execution_logger/cloud_watch_logs_spec.rb +95 -0
  272. data/spec/factories/execution_factory.rb +13 -0
  273. data/spec/factories/job_definition_factory.rb +21 -0
  274. data/spec/factories/job_instance_factory.rb +4 -0
  275. data/spec/factories/job_schedule_factory.rb +10 -0
  276. data/spec/factories/job_suspend_schedule_factory.rb +8 -0
  277. data/spec/factories/memory_expectancy_factory.rb +5 -0
  278. data/spec/factories/process_signal_factory.rb +4 -0
  279. data/spec/factories/star_factory.rb +4 -0
  280. data/spec/factories/tick_factory.rb +4 -0
  281. data/spec/factories/token_factory.rb +10 -0
  282. data/spec/factories/user_factory.rb +11 -0
  283. data/spec/factories/worker_factory.rb +8 -0
  284. data/spec/features/dashborad_spec.rb +82 -0
  285. data/spec/features/job_definition_spec.rb +74 -0
  286. data/spec/features/job_instance_spec.rb +94 -0
  287. data/spec/features/sign_in_and_out_spec.rb +17 -0
  288. data/spec/features/users_spec.rb +90 -0
  289. data/spec/features/workers_spec.rb +44 -0
  290. data/spec/helpers/executions_helper_spec.rb +4 -0
  291. data/spec/helpers/job_definition_helper_spec.rb +42 -0
  292. data/spec/helpers/job_schedules_helper_spec.rb +4 -0
  293. data/spec/helpers/logs_helper_spec.rb +4 -0
  294. data/spec/helpers/tokens_helper_spec.rb +4 -0
  295. data/spec/helpers/users_helper_spec.rb +4 -0
  296. data/spec/helpers/workers_helper_spec.rb +4 -0
  297. data/spec/mailers/notifications_spec.rb +54 -0
  298. data/spec/memory_sampler_spec.rb +11 -0
  299. data/spec/models/admin_assignment_spec.rb +4 -0
  300. data/spec/models/execution_spec.rb +26 -0
  301. data/spec/models/job_definition_spec.rb +163 -0
  302. data/spec/models/job_instance_spec.rb +115 -0
  303. data/spec/models/job_schedule_spec.rb +121 -0
  304. data/spec/models/job_suspend_schedule_spec.rb +32 -0
  305. data/spec/models/memory_consumption_log_spec.rb +50 -0
  306. data/spec/models/memory_expectancy_spec.rb +26 -0
  307. data/spec/models/star_spec.rb +4 -0
  308. data/spec/models/tick_spec.rb +23 -0
  309. data/spec/models/token_spec.rb +54 -0
  310. data/spec/models/user_spec.rb +17 -0
  311. data/spec/models/worker_spec.rb +5 -0
  312. data/spec/rails_helper.rb +81 -0
  313. data/spec/requests/api/job_instances_spec.rb +96 -0
  314. data/spec/requests/api/stats_spec.rb +45 -0
  315. data/spec/return_to_validator_spec.rb +28 -0
  316. data/spec/settings_spec.rb +10 -0
  317. data/spec/spec_helper.rb +49 -0
  318. data/spec/support/feature_sign_in_helper.rb +31 -0
  319. data/spec/support/sign_in_helper.rb +5 -0
  320. data/spec/support/wait_for_ajax.rb +11 -0
  321. data/spec/workflow/engine_spec.rb +241 -0
  322. data/spec/workflow/node_spec.rb +62 -0
  323. data/spec/workflow/notifier/hipchat_spec.rb +117 -0
  324. data/spec/workflow/notifier/mail_spec.rb +86 -0
  325. data/spec/workflow/notifier/slack_spec.rb +110 -0
  326. data/spec/workflow/script_parser_spec.rb +119 -0
  327. data/spec/workflow/shell_scanner_spec.rb +47 -0
  328. data/spec/workflow/task/auto_skip_error_spec.rb +35 -0
  329. data/spec/workflow/task/env_spec.rb +47 -0
  330. data/spec/workflow/task/execute_spec.rb +127 -0
  331. data/spec/workflow/task/expected_time_spec.rb +52 -0
  332. data/spec/workflow/task/fork_spec.rb +30 -0
  333. data/spec/workflow/task/queue_spec.rb +45 -0
  334. data/spec/workflow/task/rails_env_spec.rb +30 -0
  335. data/spec/workflow/task/sleep_spec.rb +22 -0
  336. data/spec/workflow/task/sub_process_spec.rb +32 -0
  337. data/spec/workflow/task/wait_spec.rb +162 -0
  338. metadata +1038 -0
@@ -0,0 +1,119 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow
4
+ describe ScriptParser do
5
+ subject(:parser) { ScriptParser.new(script) }
6
+
7
+ let(:root) { parser.parse }
8
+
9
+ describe '#parse' do
10
+ context 'with simple sequence' do
11
+ around do |example|
12
+ Node.register(key: :task1, klass: Task::Base)
13
+ Node.register(key: :task2, klass: Task::Base)
14
+ Node.register(key: :task3, klass: Task::Base)
15
+ Node.register(key: :task4, klass: Task::Base)
16
+
17
+ example.run
18
+
19
+ Node.deregister(:task1)
20
+ Node.deregister(:task2)
21
+ Node.deregister(:task3)
22
+ Node.deregister(:task4)
23
+ end
24
+
25
+ let(:script) do
26
+ <<-EOF
27
+ # comment
28
+ task1: # comment
29
+ task2: option
30
+
31
+ task3: option # comment
32
+ task4: OPTION='#option' # comment
33
+ EOF
34
+ end
35
+
36
+ it 'returns parsed nodes' do
37
+ expect(root.children.size).to eq 3
38
+
39
+ task1 = root.children.first
40
+ expect(task1.type).to eq :task1
41
+ expect(task1.children.size).to eq 1
42
+
43
+ task2 = task1.children.first
44
+ expect(task2.type).to eq :task2
45
+ expect(task2.option).to eq 'option'
46
+
47
+ task3 = root.children.second
48
+ expect(task3.type).to eq :task3
49
+ expect(task3.option).to eq 'option'
50
+
51
+ task4 = root.children.third
52
+ expect(task4.type).to eq :task4
53
+ expect(task4.option).to eq "OPTION='#option'"
54
+ end
55
+ end
56
+
57
+ context 'with nested indentation' do
58
+ let(:script) do
59
+ <<-EOF
60
+ noop:
61
+ noop: option
62
+ noop: option
63
+ noop:
64
+ noop: option
65
+ noop: option
66
+ noop:
67
+ EOF
68
+ end
69
+
70
+ it do
71
+ expect(root.children.size).to eq 2
72
+ expect(root.children.first.children.size).to eq 2
73
+ expect(root.children.first.children.first.children.size).to eq 1
74
+ expect(root.children.first.children.second.children.size).to eq 2
75
+ end
76
+ end
77
+
78
+ context 'with bad syntax' do
79
+ let(:script) { ':' }
80
+
81
+ it { expect { root }.to raise_error(Kuroko2::Workflow::SyntaxError) }
82
+ end
83
+
84
+ context 'with bad syntax' do
85
+ let(:script) do
86
+ <<-EOF
87
+ noop
88
+ EOF
89
+ end
90
+
91
+ it { expect { root }.to raise_error(Kuroko2::Workflow::SyntaxError) }
92
+ end
93
+
94
+ context 'with inconsistent indentation' do
95
+ let(:script) do
96
+ <<-EOF
97
+ noop:
98
+ noop: option
99
+ noop: option
100
+ EOF
101
+ end
102
+
103
+ it { expect { root }.to raise_error(Kuroko2::Workflow::SyntaxError) }
104
+ end
105
+
106
+ context 'with bad semantics' do
107
+ let(:script) do
108
+ <<-EOF
109
+ fork:
110
+ timeout: a
111
+ noop:
112
+ EOF
113
+ end
114
+
115
+ it { expect { root }.to raise_error(Kuroko2::Workflow::AssertionError) }
116
+ end
117
+ end
118
+ end
119
+ end
@@ -0,0 +1,47 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow
4
+ describe ShellScanner do
5
+ subject(:scanner) { ShellScanner.new(text) }
6
+
7
+ describe '#strip_comment' do
8
+ let(:text) { 'foo # bar' }
9
+
10
+ it 'strips text after #' do
11
+ expect(scanner.strip_comment).to eq('foo ')
12
+ end
13
+
14
+ context 'given nil' do
15
+ let(:text) { nil }
16
+
17
+ it 'returns nil' do
18
+ expect(scanner.strip_comment).to eq(nil)
19
+ end
20
+ end
21
+
22
+ context 'given string literal including #' do
23
+ let(:text) { 'FOO="b\"ar#baz" # comment' }
24
+
25
+ it 'preserves # in string literal' do
26
+ expect(scanner.strip_comment).to eq('FOO="b\"ar#baz" ')
27
+ end
28
+ end
29
+
30
+ context 'given string literal including # in single quotes' do
31
+ let(:text) { %q[FOO='b\'ar#baz' # comment] }
32
+
33
+ it 'preserves # in string literal' do
34
+ expect(scanner.strip_comment).to eq(%q[FOO='b\'ar#baz' ])
35
+ end
36
+ end
37
+
38
+ context 'given unmatched quote' do
39
+ let(:text) { %q["' # foo] }
40
+
41
+ it 'returns text until #' do
42
+ expect(scanner.strip_comment).to eq(%q["' ])
43
+ end
44
+ end
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,35 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe AutoSkipError do
5
+ describe '#execute' do
6
+ let(:node) { Kuroko2::Workflow::Node.new('auto_skip_error', 'true') }
7
+ let(:token) { build(:token, context: {}) }
8
+
9
+ subject { AutoSkipError.new(node, token).execute }
10
+
11
+ it "sets AUTO_SKIP_ERROR context" do
12
+ subject
13
+ expect(token.context).to eq({ 'AUTO_SKIP_ERROR' => true })
14
+ end
15
+
16
+ context 'Without options' do
17
+ let(:node) { Kuroko2::Workflow::Node.new('auto_skip_error') }
18
+
19
+ it "sets AUTO_SKIP_ERROR context to false" do
20
+ subject
21
+ expect(token.context).to eq({ 'AUTO_SKIP_ERROR' => false })
22
+ end
23
+ end
24
+
25
+ context 'With false options' do
26
+ let(:node) { Kuroko2::Workflow::Node.new('auto_skip_error', 'false') }
27
+
28
+ it "sets AUTO_SKIP_ERROR context to false" do
29
+ subject
30
+ expect(token.context).to eq({ 'AUTO_SKIP_ERROR' => false })
31
+ end
32
+ end
33
+ end
34
+ end
35
+ end
@@ -0,0 +1,47 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe Env do
5
+ describe '#execute' do
6
+ let(:token) { build(:token, script: 'env:', context: { 'ENV' => { 'EXISTING' => 'true' } }) }
7
+
8
+ context 'with valid syntax' do
9
+ before { Env.new(node, token).execute }
10
+ subject { token.context }
11
+
12
+ context 'with nil option' do
13
+ let(:node) { Kuroko2::Workflow::Node.new(:env) }
14
+
15
+ context 'with exisiting env' do
16
+ it { is_expected.to eq({ 'ENV' => { 'EXISTING' => 'true' } }) }
17
+ end
18
+
19
+ context 'without exisiting env' do
20
+ let(:token) { build(:token, script: 'env:') }
21
+
22
+ it { is_expected.to eq({}) }
23
+ end
24
+ end
25
+
26
+ context 'with valid option' do
27
+ let(:node) { Kuroko2::Workflow::Node.new(:env, 'A=1 B=2') }
28
+
29
+ it { is_expected.to eq({ 'ENV' => { 'A' => '1', 'B' => '2', 'EXISTING' => 'true' } }) }
30
+ end
31
+
32
+ context 'with quote' do
33
+ let(:node) { Kuroko2::Workflow::Node.new(:env, 'A="B=2 C=3" D="E=5 F=\"6"' " G='\\'7'") }
34
+
35
+ it { is_expected.to eq({ 'ENV' => { 'A' => 'B=2 C=3', 'D' => 'E=5 F="6', 'G' => "'7", 'EXISTING' => 'true' } }) }
36
+ end
37
+ end
38
+
39
+ context 'with invalid syntax' do
40
+ let(:node) { Kuroko2::Workflow::Node.new(:env, 'A="B=2 C=3" D="E=5 F=\"6') }
41
+
42
+ it { expect { Env.new(node, token).execute }.to raise_error(Kuroko2::Workflow::AssertionError) }
43
+ end
44
+ end
45
+
46
+ end
47
+ end
@@ -0,0 +1,127 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe Execute do
5
+ subject { Execute.new(node, token).execute }
6
+
7
+ let(:context) { { 'ENV' => { 'NAME' => 'alice' } } }
8
+ let(:node) { Kuroko2::Workflow::Node.new(:execute, shell) }
9
+
10
+ let(:token) { create(:token, path: '/', script: 'execute:', context: context, job_definition: definition, job_instance: instance) }
11
+ let(:execution) { Kuroko2::Execution.take }
12
+ let(:definition) { create(:job_definition, script: "execute: echo HELLO\n") }
13
+ let(:instance) do
14
+ create(:job_instance, job_definition: definition).tap do |instance|
15
+ instance.tokens.destroy
16
+ end
17
+ end
18
+
19
+ context 'with shell script' do
20
+ let(:shell) { 'echo $NAME' }
21
+
22
+ specify do
23
+ is_expected.to eq :pass
24
+
25
+ expect(Kuroko2::Execution.all.size).to eq 1
26
+ expect(execution.token).to eql token
27
+ expect(execution.shell).to eq shell
28
+ expect(execution.context['ENV']).to eq context['ENV']
29
+ end
30
+ end
31
+
32
+ context 'with successfully completed' do
33
+ before do
34
+ create(:execution, token: token)
35
+ end
36
+
37
+ let(:shell) { 'echo $NAME' }
38
+
39
+ specify do
40
+ is_expected.to eq :next
41
+
42
+ expect(Kuroko2::Execution.all.size).to eq 0
43
+ expect(Kuroko2::Log.all.count).to eq 1
44
+ end
45
+ end
46
+
47
+ context 'With TIMEOUT' do
48
+ let(:shell) { 'sleep 5 && echo $NAME' }
49
+ let(:pid) { 1 }
50
+ let(:hostname) { 'myhost' }
51
+
52
+ around do |example|
53
+ Execute.new(node, token).execute
54
+
55
+ execution = Kuroko2::Execution.of(token).take
56
+ execution.update!(pid: pid)
57
+ create(:worker, hostname: hostname, execution_id: execution.id)
58
+
59
+ token.context['TIMEOUT'] = '1' # 1 minute
60
+ Timecop.travel(2.minutes.since) { example.run }
61
+ end
62
+
63
+ it 'creates ProcessSignal' do
64
+ expect { Execute.new(node, token).execute }.to change { Kuroko2::ProcessSignal.where(pid: execution.pid, hostname: hostname).count }.from(0).to(1)
65
+ end
66
+ end
67
+
68
+ context 'if job passed EXPECTED_TIME' do
69
+ let(:shell) { 'sleep 5 && echo $NAME' }
70
+
71
+ context 'Without EXPECTED_TIME_NOTIFIED_AT' do
72
+ around do |example|
73
+ Execute.new(node, token).execute
74
+ Kuroko2::Execution.of(token).take.update!(pid: 1)
75
+
76
+ Timecop.travel((24.hours + 1.second).since) { example.run }
77
+ end
78
+
79
+ it 'alerts warnings' do
80
+ expect(Kuroko2::Workflow::Notifier).to receive(:notify).with(:long_elapsed_time, token.job_instance)
81
+
82
+ Execute.new(node, token).execute
83
+ is_expected.to eq :pass
84
+ Execute.new(node, token).execute
85
+ expect(token.context['EXPECTED_TIME_NOTIFIED_AT']).to be_present
86
+ end
87
+ end
88
+
89
+ context 'With EXPECTED_TIME_NOTIFIED_AT' do
90
+ around do |example|
91
+ Execute.new(node, token).execute
92
+ Kuroko2::Execution.of(token).take.update!(pid: 1)
93
+
94
+ Timecop.travel((24.hours + 1.second).since) do
95
+ token.context['EXPECTED_TIME_NOTIFIED_AT'] = notified_time
96
+ example.run
97
+ end
98
+ end
99
+
100
+ context 'When EXPECTED_TIME_NOTIFIED_AT is now' do
101
+ let(:notified_time) { Time.now }
102
+
103
+ it 'does not alert warnings' do
104
+ expect(Kuroko2::Workflow::Notifier).not_to receive(:notify)
105
+ Execute.new(node, token).execute
106
+ is_expected.to eq :pass
107
+ Execute.new(node, token).execute
108
+ expect(token.context['EXPECTED_TIME_NOTIFIED_AT']).to be_present
109
+ end
110
+ end
111
+
112
+ context 'When EXPECTED_TIME_NOTIFIED_AT is 1 hours ago' do
113
+ let(:notified_time) { (1.hours + 1.second).ago }
114
+
115
+ it 'alert warnings' do
116
+ expect(Kuroko2::Workflow::Notifier).to receive(:notify).with(:long_elapsed_time, token.job_instance)
117
+
118
+ Execute.new(node, token).execute
119
+ is_expected.to eq :pass
120
+ Execute.new(node, token).execute
121
+ expect(token.context['EXPECTED_TIME_NOTIFIED_AT']).to be_present
122
+ end
123
+ end
124
+ end
125
+ end
126
+ end
127
+ end
@@ -0,0 +1,52 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe ExpectedTime do
5
+ describe '#execute' do
6
+ let(:node) { Kuroko2::Workflow::Node.new('expected_time', '100') }
7
+ let(:token) { build(:token, context: {}) }
8
+
9
+ subject { ExpectedTime.new(node, token).execute }
10
+
11
+ it "sets EXPECTED_TIME context" do
12
+ subject
13
+ expect(token.context).to eq({ 'EXPECTED_TIME' => 100 })
14
+ end
15
+
16
+ context 'With %d+h type' do
17
+ let(:node) { Kuroko2::Workflow::Node.new('expected_time', '10h') }
18
+
19
+ it "sets EXPECTED_TIME context" do
20
+ subject
21
+ expect(token.context).to eq({ 'EXPECTED_TIME' => 10*60 })
22
+ end
23
+ end
24
+
25
+ context 'With %d+m type' do
26
+ let(:node) { Kuroko2::Workflow::Node.new('expected_time', '10m') }
27
+
28
+ it "sets EXPECTED_TIME context" do
29
+ subject
30
+ expect(token.context).to eq({ 'EXPECTED_TIME' => 10 })
31
+ end
32
+ end
33
+
34
+ context 'Without options' do
35
+ let(:node) { Kuroko2::Workflow::Node.new('expected_time') }
36
+
37
+ it "does not sets EXPECTED_TIME context" do
38
+ subject
39
+ expect(token.context).to eq({})
40
+ end
41
+ end
42
+
43
+ context 'With non integer value' do
44
+ let(:node) { Kuroko2::Workflow::Node.new('expected_time', 'abc') }
45
+
46
+ it "raises error" do
47
+ expect { subject }.to raise_error(Kuroko2::Workflow::AssertionError)
48
+ end
49
+ end
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,30 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe Fork do
5
+ let(:node) { Kuroko2::Workflow::ScriptParser.new(token.script).parse.find(token.path) }
6
+ let(:definition) { create(:job_definition) }
7
+ let(:instance) { create(:job_instance, job_definition: definition) }
8
+
9
+ let(:token) do
10
+ Kuroko2::Token.create(uuid: SecureRandom.uuid, path: '/0-fork', job_definition: definition, job_instance: instance, script: <<-EOF)
11
+ fork:
12
+ noop: noop1
13
+ noop: noop2
14
+ EOF
15
+ end
16
+
17
+ describe '#execute' do
18
+ subject(:task) { Fork.new(node, token) }
19
+ let(:children) { token.children }
20
+
21
+ it do
22
+ expect(task.execute).to eq :pass
23
+
24
+ expect(children.size).to eq 2
25
+ expect(children.first.script).to eq "noop: noop1\n"
26
+ expect(children.second.script).to eq "noop: noop2\n"
27
+ end
28
+ end
29
+ end
30
+ end
@@ -0,0 +1,45 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe Queue do
5
+ describe '#execute' do
6
+ subject { token.context['QUEUE'] }
7
+
8
+ let(:token) { build(:token, script: 'queue:') }
9
+ let(:queue) { Kuroko2::Execution::DEFAULT_QUEUE }
10
+
11
+ before do
12
+ Kuroko2::Worker.create!(hostname: 'test', worker_id: 1, queue: queue, working: true)
13
+ end
14
+
15
+ context 'valid case' do
16
+ before { Kuroko2::Workflow::Task::Queue.new(node, token).execute }
17
+
18
+ context 'with default queue' do
19
+ let(:node) { Kuroko2::Workflow::Node.new(:queue) }
20
+
21
+ it { is_expected.to eq queue }
22
+ end
23
+
24
+ context 'with given queue' do
25
+ let(:queue) { 'queue_name' }
26
+ let(:node) { Kuroko2::Workflow::Node.new(:queue, queue) }
27
+
28
+ it { is_expected.to eq queue }
29
+ end
30
+ end
31
+
32
+ context 'with invalid queue' do
33
+ let(:node) { Kuroko2::Workflow::Node.new(:queue, '!invalid!') }
34
+
35
+ it { expect { Kuroko2::Workflow::Task::Queue.new(node, token).validate }.to raise_error(Kuroko2::Workflow::AssertionError) }
36
+ end
37
+
38
+ context 'with no existance queue' do
39
+ let(:node) { Kuroko2::Workflow::Node.new(:queue, 'invalid') }
40
+
41
+ it { expect { Kuroko2::Workflow::Task::Queue.new(node, token).validate }.to raise_error(Kuroko2::Workflow::AssertionError) }
42
+ end
43
+ end
44
+ end
45
+ end
@@ -0,0 +1,30 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe RailsEnv do
5
+ describe '#execute' do
6
+ let(:rails_env) { 'test' }
7
+ let(:node) { Kuroko2::Workflow::Node.new(:rails_env, rails_env) }
8
+ let(:token) { build(:token, context: {}) }
9
+
10
+ subject { RailsEnv.new(node, token).execute }
11
+
12
+ context 'with valid rails_env' do
13
+ it "sets RAILS_ENV" do
14
+ subject
15
+ expect(token.context).to eq({ 'RAILS_ENV' => 'test' })
16
+ end
17
+ end
18
+
19
+ context 'with invalid rails_env' do
20
+ let(:rails_env){ 'production' }
21
+ it { expect{ subject }.to raise_error(Kuroko2::Workflow::AssertionError) }
22
+ end
23
+
24
+ context 'with not exists rails_env' do
25
+ let(:rails_env){ 'xxxxxxx' }
26
+ it { expect{ subject }.to raise_error(Kuroko2::Workflow::AssertionError) }
27
+ end
28
+ end
29
+ end
30
+ end
@@ -0,0 +1,22 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe Queue do
5
+ describe '#execute' do
6
+ around do |example|
7
+ Timecop.freeze(Time.at(time)) do
8
+ example.run
9
+ end
10
+ end
11
+ before { Sleep.new(node, token).execute }
12
+
13
+ let(:token) { build(:token, script: 'sleep:') }
14
+ let(:node) { Kuroko2::Workflow::Node.new(:sleep, seconds.to_s) }
15
+
16
+ let(:seconds) { 10 }
17
+ let(:time) { 1000000000 }
18
+
19
+ it { expect(token.context['SLEEP']).to eq (time + seconds) }
20
+ end
21
+ end
22
+ end
@@ -0,0 +1,32 @@
1
+ require 'rails_helper'
2
+
3
+ module Kuroko2::Workflow::Task
4
+ describe SubProcess do
5
+ let(:definition) { create(:job_definition) }
6
+ let(:sub_definition) { create(:job_definition) }
7
+
8
+ let(:instance) do
9
+ create(:job_instance, job_definition: definition).tap do |instance|
10
+ instance.tokens.first.destroy
11
+ end
12
+ end
13
+
14
+ let(:node) { Kuroko2::Workflow::Node.new(:sub_process, sub_definition.id.to_s) }
15
+ let(:token) { create(:token, job_definition: definition, job_instance: instance) }
16
+
17
+ let(:sub_instance) { sub_definition.job_instances.first }
18
+
19
+ describe '#execute' do
20
+ subject(:task) { SubProcess.new(node, token) }
21
+
22
+ it do
23
+ expect(task.execute).to eq :pass
24
+ expect(Kuroko2::Token.count).to eq 2
25
+ expect(task.execute).to eq :pass
26
+
27
+ sub_instance.touch(:finished_at)
28
+ expect(task.execute).to eq :next
29
+ end
30
+ end
31
+ end
32
+ end