temporalio 0.0.0 → 0.0.2

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 (327) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +301 -0
  3. data/bridge/Cargo.lock +2888 -0
  4. data/bridge/Cargo.toml +27 -0
  5. data/bridge/sdk-core/ARCHITECTURE.md +76 -0
  6. data/bridge/sdk-core/Cargo.lock +2606 -0
  7. data/bridge/sdk-core/Cargo.toml +2 -0
  8. data/bridge/sdk-core/LICENSE.txt +23 -0
  9. data/bridge/sdk-core/README.md +104 -0
  10. data/bridge/sdk-core/arch_docs/diagrams/README.md +10 -0
  11. data/bridge/sdk-core/arch_docs/diagrams/sticky_queues.puml +40 -0
  12. data/bridge/sdk-core/arch_docs/diagrams/workflow_internals.svg +1 -0
  13. data/bridge/sdk-core/arch_docs/sticky_queues.md +51 -0
  14. data/bridge/sdk-core/client/Cargo.toml +40 -0
  15. data/bridge/sdk-core/client/LICENSE.txt +23 -0
  16. data/bridge/sdk-core/client/src/lib.rs +1286 -0
  17. data/bridge/sdk-core/client/src/metrics.rs +165 -0
  18. data/bridge/sdk-core/client/src/raw.rs +932 -0
  19. data/bridge/sdk-core/client/src/retry.rs +751 -0
  20. data/bridge/sdk-core/client/src/workflow_handle/mod.rs +185 -0
  21. data/bridge/sdk-core/core/Cargo.toml +116 -0
  22. data/bridge/sdk-core/core/LICENSE.txt +23 -0
  23. data/bridge/sdk-core/core/benches/workflow_replay.rs +76 -0
  24. data/bridge/sdk-core/core/src/abstractions.rs +166 -0
  25. data/bridge/sdk-core/core/src/core_tests/activity_tasks.rs +1014 -0
  26. data/bridge/sdk-core/core/src/core_tests/child_workflows.rs +221 -0
  27. data/bridge/sdk-core/core/src/core_tests/determinism.rs +107 -0
  28. data/bridge/sdk-core/core/src/core_tests/local_activities.rs +925 -0
  29. data/bridge/sdk-core/core/src/core_tests/mod.rs +100 -0
  30. data/bridge/sdk-core/core/src/core_tests/queries.rs +894 -0
  31. data/bridge/sdk-core/core/src/core_tests/replay_flag.rs +65 -0
  32. data/bridge/sdk-core/core/src/core_tests/workers.rs +259 -0
  33. data/bridge/sdk-core/core/src/core_tests/workflow_cancels.rs +124 -0
  34. data/bridge/sdk-core/core/src/core_tests/workflow_tasks.rs +2090 -0
  35. data/bridge/sdk-core/core/src/ephemeral_server/mod.rs +515 -0
  36. data/bridge/sdk-core/core/src/lib.rs +282 -0
  37. data/bridge/sdk-core/core/src/pollers/mod.rs +54 -0
  38. data/bridge/sdk-core/core/src/pollers/poll_buffer.rs +297 -0
  39. data/bridge/sdk-core/core/src/protosext/mod.rs +428 -0
  40. data/bridge/sdk-core/core/src/replay/mod.rs +215 -0
  41. data/bridge/sdk-core/core/src/retry_logic.rs +202 -0
  42. data/bridge/sdk-core/core/src/telemetry/log_export.rs +190 -0
  43. data/bridge/sdk-core/core/src/telemetry/metrics.rs +428 -0
  44. data/bridge/sdk-core/core/src/telemetry/mod.rs +407 -0
  45. data/bridge/sdk-core/core/src/telemetry/prometheus_server.rs +78 -0
  46. data/bridge/sdk-core/core/src/test_help/mod.rs +889 -0
  47. data/bridge/sdk-core/core/src/worker/activities/activity_heartbeat_manager.rs +580 -0
  48. data/bridge/sdk-core/core/src/worker/activities/local_activities.rs +1048 -0
  49. data/bridge/sdk-core/core/src/worker/activities.rs +481 -0
  50. data/bridge/sdk-core/core/src/worker/client/mocks.rs +87 -0
  51. data/bridge/sdk-core/core/src/worker/client.rs +373 -0
  52. data/bridge/sdk-core/core/src/worker/mod.rs +570 -0
  53. data/bridge/sdk-core/core/src/worker/workflow/bridge.rs +37 -0
  54. data/bridge/sdk-core/core/src/worker/workflow/driven_workflow.rs +101 -0
  55. data/bridge/sdk-core/core/src/worker/workflow/history_update.rs +532 -0
  56. data/bridge/sdk-core/core/src/worker/workflow/machines/activity_state_machine.rs +907 -0
  57. data/bridge/sdk-core/core/src/worker/workflow/machines/cancel_external_state_machine.rs +294 -0
  58. data/bridge/sdk-core/core/src/worker/workflow/machines/cancel_workflow_state_machine.rs +167 -0
  59. data/bridge/sdk-core/core/src/worker/workflow/machines/child_workflow_state_machine.rs +858 -0
  60. data/bridge/sdk-core/core/src/worker/workflow/machines/complete_workflow_state_machine.rs +136 -0
  61. data/bridge/sdk-core/core/src/worker/workflow/machines/continue_as_new_workflow_state_machine.rs +157 -0
  62. data/bridge/sdk-core/core/src/worker/workflow/machines/fail_workflow_state_machine.rs +129 -0
  63. data/bridge/sdk-core/core/src/worker/workflow/machines/local_activity_state_machine.rs +1450 -0
  64. data/bridge/sdk-core/core/src/worker/workflow/machines/mod.rs +316 -0
  65. data/bridge/sdk-core/core/src/worker/workflow/machines/modify_workflow_properties_state_machine.rs +178 -0
  66. data/bridge/sdk-core/core/src/worker/workflow/machines/patch_state_machine.rs +708 -0
  67. data/bridge/sdk-core/core/src/worker/workflow/machines/signal_external_state_machine.rs +439 -0
  68. data/bridge/sdk-core/core/src/worker/workflow/machines/timer_state_machine.rs +435 -0
  69. data/bridge/sdk-core/core/src/worker/workflow/machines/transition_coverage.rs +175 -0
  70. data/bridge/sdk-core/core/src/worker/workflow/machines/upsert_search_attributes_state_machine.rs +242 -0
  71. data/bridge/sdk-core/core/src/worker/workflow/machines/workflow_machines/local_acts.rs +96 -0
  72. data/bridge/sdk-core/core/src/worker/workflow/machines/workflow_machines.rs +1200 -0
  73. data/bridge/sdk-core/core/src/worker/workflow/machines/workflow_task_state_machine.rs +272 -0
  74. data/bridge/sdk-core/core/src/worker/workflow/managed_run/managed_wf_test.rs +198 -0
  75. data/bridge/sdk-core/core/src/worker/workflow/managed_run.rs +655 -0
  76. data/bridge/sdk-core/core/src/worker/workflow/mod.rs +1200 -0
  77. data/bridge/sdk-core/core/src/worker/workflow/run_cache.rs +145 -0
  78. data/bridge/sdk-core/core/src/worker/workflow/wft_poller.rs +88 -0
  79. data/bridge/sdk-core/core/src/worker/workflow/workflow_stream.rs +985 -0
  80. data/bridge/sdk-core/core-api/Cargo.toml +32 -0
  81. data/bridge/sdk-core/core-api/LICENSE.txt +23 -0
  82. data/bridge/sdk-core/core-api/src/errors.rs +95 -0
  83. data/bridge/sdk-core/core-api/src/lib.rs +109 -0
  84. data/bridge/sdk-core/core-api/src/telemetry.rs +147 -0
  85. data/bridge/sdk-core/core-api/src/worker.rs +148 -0
  86. data/bridge/sdk-core/etc/deps.svg +162 -0
  87. data/bridge/sdk-core/etc/dynamic-config.yaml +2 -0
  88. data/bridge/sdk-core/etc/otel-collector-config.yaml +36 -0
  89. data/bridge/sdk-core/etc/prometheus.yaml +6 -0
  90. data/bridge/sdk-core/etc/regen-depgraph.sh +5 -0
  91. data/bridge/sdk-core/fsm/Cargo.toml +18 -0
  92. data/bridge/sdk-core/fsm/LICENSE.txt +23 -0
  93. data/bridge/sdk-core/fsm/README.md +3 -0
  94. data/bridge/sdk-core/fsm/rustfsm_procmacro/Cargo.toml +27 -0
  95. data/bridge/sdk-core/fsm/rustfsm_procmacro/LICENSE.txt +23 -0
  96. data/bridge/sdk-core/fsm/rustfsm_procmacro/src/lib.rs +647 -0
  97. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/progress.rs +8 -0
  98. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/dupe_transitions_fail.rs +18 -0
  99. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/dupe_transitions_fail.stderr +12 -0
  100. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/dynamic_dest_pass.rs +41 -0
  101. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/forgot_name_fail.rs +14 -0
  102. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/forgot_name_fail.stderr +11 -0
  103. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/handler_arg_pass.rs +32 -0
  104. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/handler_pass.rs +31 -0
  105. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/medium_complex_pass.rs +46 -0
  106. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/no_handle_conversions_require_into_fail.rs +29 -0
  107. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/no_handle_conversions_require_into_fail.stderr +12 -0
  108. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/simple_pass.rs +32 -0
  109. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/struct_event_variant_fail.rs +18 -0
  110. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/struct_event_variant_fail.stderr +5 -0
  111. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/tuple_more_item_event_variant_fail.rs +11 -0
  112. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/tuple_more_item_event_variant_fail.stderr +5 -0
  113. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/tuple_zero_item_event_variant_fail.rs +11 -0
  114. data/bridge/sdk-core/fsm/rustfsm_procmacro/tests/trybuild/tuple_zero_item_event_variant_fail.stderr +5 -0
  115. data/bridge/sdk-core/fsm/rustfsm_trait/Cargo.toml +14 -0
  116. data/bridge/sdk-core/fsm/rustfsm_trait/LICENSE.txt +23 -0
  117. data/bridge/sdk-core/fsm/rustfsm_trait/src/lib.rs +249 -0
  118. data/bridge/sdk-core/fsm/src/lib.rs +2 -0
  119. data/bridge/sdk-core/histories/evict_while_la_running_no_interference-23_history.bin +0 -0
  120. data/bridge/sdk-core/histories/evict_while_la_running_no_interference-85_history.bin +0 -0
  121. data/bridge/sdk-core/histories/fail_wf_task.bin +0 -0
  122. data/bridge/sdk-core/histories/timer_workflow_history.bin +0 -0
  123. data/bridge/sdk-core/integ-with-otel.sh +7 -0
  124. data/bridge/sdk-core/protos/api_upstream/README.md +9 -0
  125. data/bridge/sdk-core/protos/api_upstream/api-linter.yaml +40 -0
  126. data/bridge/sdk-core/protos/api_upstream/buf.yaml +9 -0
  127. data/bridge/sdk-core/protos/api_upstream/build/go.mod +7 -0
  128. data/bridge/sdk-core/protos/api_upstream/build/go.sum +5 -0
  129. data/bridge/sdk-core/protos/api_upstream/build/tools.go +29 -0
  130. data/bridge/sdk-core/protos/api_upstream/dependencies/gogoproto/gogo.proto +141 -0
  131. data/bridge/sdk-core/protos/api_upstream/go.mod +6 -0
  132. data/bridge/sdk-core/protos/api_upstream/temporal/api/batch/v1/message.proto +89 -0
  133. data/bridge/sdk-core/protos/api_upstream/temporal/api/command/v1/message.proto +260 -0
  134. data/bridge/sdk-core/protos/api_upstream/temporal/api/common/v1/message.proto +112 -0
  135. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/batch_operation.proto +47 -0
  136. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/command_type.proto +57 -0
  137. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/common.proto +56 -0
  138. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/event_type.proto +170 -0
  139. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/failed_cause.proto +118 -0
  140. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/interaction_type.proto +39 -0
  141. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/namespace.proto +51 -0
  142. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/query.proto +50 -0
  143. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/reset.proto +41 -0
  144. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/schedule.proto +60 -0
  145. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/task_queue.proto +59 -0
  146. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/update.proto +40 -0
  147. data/bridge/sdk-core/protos/api_upstream/temporal/api/enums/v1/workflow.proto +122 -0
  148. data/bridge/sdk-core/protos/api_upstream/temporal/api/errordetails/v1/message.proto +108 -0
  149. data/bridge/sdk-core/protos/api_upstream/temporal/api/failure/v1/message.proto +114 -0
  150. data/bridge/sdk-core/protos/api_upstream/temporal/api/filter/v1/message.proto +56 -0
  151. data/bridge/sdk-core/protos/api_upstream/temporal/api/history/v1/message.proto +758 -0
  152. data/bridge/sdk-core/protos/api_upstream/temporal/api/interaction/v1/message.proto +87 -0
  153. data/bridge/sdk-core/protos/api_upstream/temporal/api/namespace/v1/message.proto +97 -0
  154. data/bridge/sdk-core/protos/api_upstream/temporal/api/operatorservice/v1/request_response.proto +121 -0
  155. data/bridge/sdk-core/protos/api_upstream/temporal/api/operatorservice/v1/service.proto +80 -0
  156. data/bridge/sdk-core/protos/api_upstream/temporal/api/query/v1/message.proto +61 -0
  157. data/bridge/sdk-core/protos/api_upstream/temporal/api/replication/v1/message.proto +55 -0
  158. data/bridge/sdk-core/protos/api_upstream/temporal/api/schedule/v1/message.proto +379 -0
  159. data/bridge/sdk-core/protos/api_upstream/temporal/api/taskqueue/v1/message.proto +108 -0
  160. data/bridge/sdk-core/protos/api_upstream/temporal/api/version/v1/message.proto +59 -0
  161. data/bridge/sdk-core/protos/api_upstream/temporal/api/workflow/v1/message.proto +146 -0
  162. data/bridge/sdk-core/protos/api_upstream/temporal/api/workflowservice/v1/request_response.proto +1168 -0
  163. data/bridge/sdk-core/protos/api_upstream/temporal/api/workflowservice/v1/service.proto +415 -0
  164. data/bridge/sdk-core/protos/grpc/health/v1/health.proto +63 -0
  165. data/bridge/sdk-core/protos/local/temporal/sdk/core/activity_result/activity_result.proto +78 -0
  166. data/bridge/sdk-core/protos/local/temporal/sdk/core/activity_task/activity_task.proto +79 -0
  167. data/bridge/sdk-core/protos/local/temporal/sdk/core/child_workflow/child_workflow.proto +77 -0
  168. data/bridge/sdk-core/protos/local/temporal/sdk/core/common/common.proto +15 -0
  169. data/bridge/sdk-core/protos/local/temporal/sdk/core/core_interface.proto +30 -0
  170. data/bridge/sdk-core/protos/local/temporal/sdk/core/external_data/external_data.proto +30 -0
  171. data/bridge/sdk-core/protos/local/temporal/sdk/core/workflow_activation/workflow_activation.proto +263 -0
  172. data/bridge/sdk-core/protos/local/temporal/sdk/core/workflow_commands/workflow_commands.proto +304 -0
  173. data/bridge/sdk-core/protos/local/temporal/sdk/core/workflow_completion/workflow_completion.proto +29 -0
  174. data/bridge/sdk-core/protos/testsrv_upstream/api-linter.yaml +38 -0
  175. data/bridge/sdk-core/protos/testsrv_upstream/buf.yaml +13 -0
  176. data/bridge/sdk-core/protos/testsrv_upstream/dependencies/gogoproto/gogo.proto +141 -0
  177. data/bridge/sdk-core/protos/testsrv_upstream/temporal/api/testservice/v1/request_response.proto +63 -0
  178. data/bridge/sdk-core/protos/testsrv_upstream/temporal/api/testservice/v1/service.proto +90 -0
  179. data/bridge/sdk-core/rustfmt.toml +1 -0
  180. data/bridge/sdk-core/sdk/Cargo.toml +47 -0
  181. data/bridge/sdk-core/sdk/LICENSE.txt +23 -0
  182. data/bridge/sdk-core/sdk/src/activity_context.rs +230 -0
  183. data/bridge/sdk-core/sdk/src/app_data.rs +37 -0
  184. data/bridge/sdk-core/sdk/src/interceptors.rs +50 -0
  185. data/bridge/sdk-core/sdk/src/lib.rs +794 -0
  186. data/bridge/sdk-core/sdk/src/payload_converter.rs +11 -0
  187. data/bridge/sdk-core/sdk/src/workflow_context/options.rs +295 -0
  188. data/bridge/sdk-core/sdk/src/workflow_context.rs +694 -0
  189. data/bridge/sdk-core/sdk/src/workflow_future.rs +499 -0
  190. data/bridge/sdk-core/sdk-core-protos/Cargo.toml +30 -0
  191. data/bridge/sdk-core/sdk-core-protos/LICENSE.txt +23 -0
  192. data/bridge/sdk-core/sdk-core-protos/build.rs +107 -0
  193. data/bridge/sdk-core/sdk-core-protos/src/constants.rs +7 -0
  194. data/bridge/sdk-core/sdk-core-protos/src/history_builder.rs +544 -0
  195. data/bridge/sdk-core/sdk-core-protos/src/history_info.rs +230 -0
  196. data/bridge/sdk-core/sdk-core-protos/src/lib.rs +1970 -0
  197. data/bridge/sdk-core/sdk-core-protos/src/task_token.rs +38 -0
  198. data/bridge/sdk-core/sdk-core-protos/src/utilities.rs +14 -0
  199. data/bridge/sdk-core/test-utils/Cargo.toml +36 -0
  200. data/bridge/sdk-core/test-utils/src/canned_histories.rs +1579 -0
  201. data/bridge/sdk-core/test-utils/src/histfetch.rs +28 -0
  202. data/bridge/sdk-core/test-utils/src/lib.rs +650 -0
  203. data/bridge/sdk-core/tests/integ_tests/client_tests.rs +36 -0
  204. data/bridge/sdk-core/tests/integ_tests/ephemeral_server_tests.rs +128 -0
  205. data/bridge/sdk-core/tests/integ_tests/heartbeat_tests.rs +221 -0
  206. data/bridge/sdk-core/tests/integ_tests/metrics_tests.rs +37 -0
  207. data/bridge/sdk-core/tests/integ_tests/polling_tests.rs +133 -0
  208. data/bridge/sdk-core/tests/integ_tests/queries_tests.rs +437 -0
  209. data/bridge/sdk-core/tests/integ_tests/visibility_tests.rs +93 -0
  210. data/bridge/sdk-core/tests/integ_tests/workflow_tests/activities.rs +878 -0
  211. data/bridge/sdk-core/tests/integ_tests/workflow_tests/appdata_propagation.rs +61 -0
  212. data/bridge/sdk-core/tests/integ_tests/workflow_tests/cancel_external.rs +59 -0
  213. data/bridge/sdk-core/tests/integ_tests/workflow_tests/cancel_wf.rs +58 -0
  214. data/bridge/sdk-core/tests/integ_tests/workflow_tests/child_workflows.rs +50 -0
  215. data/bridge/sdk-core/tests/integ_tests/workflow_tests/continue_as_new.rs +60 -0
  216. data/bridge/sdk-core/tests/integ_tests/workflow_tests/determinism.rs +54 -0
  217. data/bridge/sdk-core/tests/integ_tests/workflow_tests/local_activities.rs +788 -0
  218. data/bridge/sdk-core/tests/integ_tests/workflow_tests/modify_wf_properties.rs +53 -0
  219. data/bridge/sdk-core/tests/integ_tests/workflow_tests/patches.rs +113 -0
  220. data/bridge/sdk-core/tests/integ_tests/workflow_tests/replay.rs +223 -0
  221. data/bridge/sdk-core/tests/integ_tests/workflow_tests/resets.rs +93 -0
  222. data/bridge/sdk-core/tests/integ_tests/workflow_tests/signals.rs +167 -0
  223. data/bridge/sdk-core/tests/integ_tests/workflow_tests/stickyness.rs +99 -0
  224. data/bridge/sdk-core/tests/integ_tests/workflow_tests/timers.rs +131 -0
  225. data/bridge/sdk-core/tests/integ_tests/workflow_tests/upsert_search_attrs.rs +75 -0
  226. data/bridge/sdk-core/tests/integ_tests/workflow_tests.rs +597 -0
  227. data/bridge/sdk-core/tests/load_tests.rs +191 -0
  228. data/bridge/sdk-core/tests/main.rs +113 -0
  229. data/bridge/sdk-core/tests/runner.rs +93 -0
  230. data/bridge/src/connection.rs +186 -0
  231. data/bridge/src/lib.rs +239 -0
  232. data/bridge/src/runtime.rs +54 -0
  233. data/bridge/src/worker.rs +124 -0
  234. data/ext/Rakefile +9 -0
  235. data/lib/bridge.so +0 -0
  236. data/lib/gen/dependencies/gogoproto/gogo_pb.rb +14 -0
  237. data/lib/gen/temporal/api/batch/v1/message_pb.rb +50 -0
  238. data/lib/gen/temporal/api/command/v1/message_pb.rb +174 -0
  239. data/lib/gen/temporal/api/common/v1/message_pb.rb +69 -0
  240. data/lib/gen/temporal/api/enums/v1/batch_operation_pb.rb +33 -0
  241. data/lib/gen/temporal/api/enums/v1/command_type_pb.rb +39 -0
  242. data/lib/gen/temporal/api/enums/v1/common_pb.rb +42 -0
  243. data/lib/gen/temporal/api/enums/v1/event_type_pb.rb +68 -0
  244. data/lib/gen/temporal/api/enums/v1/failed_cause_pb.rb +77 -0
  245. data/lib/gen/temporal/api/enums/v1/interaction_type_pb.rb +25 -0
  246. data/lib/gen/temporal/api/enums/v1/namespace_pb.rb +37 -0
  247. data/lib/gen/temporal/api/enums/v1/query_pb.rb +31 -0
  248. data/lib/gen/temporal/api/enums/v1/reset_pb.rb +24 -0
  249. data/lib/gen/temporal/api/enums/v1/schedule_pb.rb +28 -0
  250. data/lib/gen/temporal/api/enums/v1/task_queue_pb.rb +30 -0
  251. data/lib/gen/temporal/api/enums/v1/update_pb.rb +23 -0
  252. data/lib/gen/temporal/api/enums/v1/workflow_pb.rb +89 -0
  253. data/lib/gen/temporal/api/errordetails/v1/message_pb.rb +84 -0
  254. data/lib/gen/temporal/api/failure/v1/message_pb.rb +83 -0
  255. data/lib/gen/temporal/api/filter/v1/message_pb.rb +40 -0
  256. data/lib/gen/temporal/api/history/v1/message_pb.rb +490 -0
  257. data/lib/gen/temporal/api/interaction/v1/message_pb.rb +49 -0
  258. data/lib/gen/temporal/api/namespace/v1/message_pb.rb +63 -0
  259. data/lib/gen/temporal/api/operatorservice/v1/request_response_pb.rb +85 -0
  260. data/lib/gen/temporal/api/operatorservice/v1/service_pb.rb +20 -0
  261. data/lib/gen/temporal/api/query/v1/message_pb.rb +38 -0
  262. data/lib/gen/temporal/api/replication/v1/message_pb.rb +37 -0
  263. data/lib/gen/temporal/api/schedule/v1/message_pb.rb +149 -0
  264. data/lib/gen/temporal/api/taskqueue/v1/message_pb.rb +73 -0
  265. data/lib/gen/temporal/api/version/v1/message_pb.rb +41 -0
  266. data/lib/gen/temporal/api/workflow/v1/message_pb.rb +111 -0
  267. data/lib/gen/temporal/api/workflowservice/v1/request_response_pb.rb +788 -0
  268. data/lib/gen/temporal/api/workflowservice/v1/service_pb.rb +20 -0
  269. data/lib/gen/temporal/sdk/core/activity_result/activity_result_pb.rb +58 -0
  270. data/lib/gen/temporal/sdk/core/activity_task/activity_task_pb.rb +57 -0
  271. data/lib/gen/temporal/sdk/core/bridge/bridge_pb.rb +222 -0
  272. data/lib/gen/temporal/sdk/core/child_workflow/child_workflow_pb.rb +57 -0
  273. data/lib/gen/temporal/sdk/core/common/common_pb.rb +22 -0
  274. data/lib/gen/temporal/sdk/core/core_interface_pb.rb +34 -0
  275. data/lib/gen/temporal/sdk/core/external_data/external_data_pb.rb +27 -0
  276. data/lib/gen/temporal/sdk/core/workflow_activation/workflow_activation_pb.rb +165 -0
  277. data/lib/gen/temporal/sdk/core/workflow_commands/workflow_commands_pb.rb +196 -0
  278. data/lib/gen/temporal/sdk/core/workflow_completion/workflow_completion_pb.rb +34 -0
  279. data/lib/temporalio/activity/context.rb +97 -0
  280. data/lib/temporalio/activity/info.rb +67 -0
  281. data/lib/temporalio/activity.rb +85 -0
  282. data/lib/temporalio/bridge/error.rb +8 -0
  283. data/lib/temporalio/bridge.rb +14 -0
  284. data/lib/temporalio/client/implementation.rb +340 -0
  285. data/lib/temporalio/client/workflow_handle.rb +243 -0
  286. data/lib/temporalio/client.rb +131 -0
  287. data/lib/temporalio/connection.rb +751 -0
  288. data/lib/temporalio/data_converter.rb +191 -0
  289. data/lib/temporalio/error/failure.rb +194 -0
  290. data/lib/temporalio/error/workflow_failure.rb +19 -0
  291. data/lib/temporalio/errors.rb +40 -0
  292. data/lib/temporalio/failure_converter/base.rb +26 -0
  293. data/lib/temporalio/failure_converter/basic.rb +319 -0
  294. data/lib/temporalio/failure_converter.rb +7 -0
  295. data/lib/temporalio/interceptor/chain.rb +28 -0
  296. data/lib/temporalio/interceptor/client.rb +123 -0
  297. data/lib/temporalio/payload_codec/base.rb +32 -0
  298. data/lib/temporalio/payload_converter/base.rb +24 -0
  299. data/lib/temporalio/payload_converter/bytes.rb +27 -0
  300. data/lib/temporalio/payload_converter/composite.rb +49 -0
  301. data/lib/temporalio/payload_converter/encoding_base.rb +35 -0
  302. data/lib/temporalio/payload_converter/json.rb +26 -0
  303. data/lib/temporalio/payload_converter/nil.rb +26 -0
  304. data/lib/temporalio/payload_converter.rb +14 -0
  305. data/lib/temporalio/retry_policy.rb +82 -0
  306. data/lib/temporalio/retry_state.rb +35 -0
  307. data/lib/temporalio/runtime.rb +25 -0
  308. data/lib/temporalio/timeout_type.rb +29 -0
  309. data/lib/temporalio/version.rb +3 -0
  310. data/lib/temporalio/worker/activity_runner.rb +92 -0
  311. data/lib/temporalio/worker/activity_worker.rb +138 -0
  312. data/lib/temporalio/worker/reactor.rb +46 -0
  313. data/lib/temporalio/worker/runner.rb +63 -0
  314. data/lib/temporalio/worker/sync_worker.rb +88 -0
  315. data/lib/temporalio/worker/thread_pool_executor.rb +51 -0
  316. data/lib/temporalio/worker.rb +198 -0
  317. data/lib/temporalio/workflow/execution_info.rb +54 -0
  318. data/lib/temporalio/workflow/execution_status.rb +36 -0
  319. data/lib/temporalio/workflow/id_reuse_policy.rb +36 -0
  320. data/lib/temporalio/workflow/query_reject_condition.rb +33 -0
  321. data/lib/temporalio.rb +12 -1
  322. data/lib/thermite_patch.rb +23 -0
  323. data/temporalio.gemspec +45 -0
  324. metadata +566 -9
  325. data/lib/temporal/version.rb +0 -3
  326. data/lib/temporal.rb +0 -4
  327. data/temporal.gemspec +0 -20
@@ -0,0 +1,282 @@
1
+ #![warn(missing_docs)] // error if there are missing docs
2
+ #![allow(clippy::upper_case_acronyms)]
3
+
4
+ //! This crate provides a basis for creating new Temporal SDKs without completely starting from
5
+ //! scratch
6
+
7
+ #[cfg(test)]
8
+ #[macro_use]
9
+ pub extern crate assert_matches;
10
+ #[macro_use]
11
+ extern crate tracing;
12
+ extern crate core;
13
+
14
+ mod abstractions;
15
+ pub mod ephemeral_server;
16
+ mod pollers;
17
+ mod protosext;
18
+ pub mod replay;
19
+ pub(crate) mod retry_logic;
20
+ pub mod telemetry;
21
+ mod worker;
22
+
23
+ #[cfg(test)]
24
+ mod core_tests;
25
+ #[cfg(test)]
26
+ #[macro_use]
27
+ mod test_help;
28
+
29
+ pub(crate) use temporal_sdk_core_api::errors;
30
+
31
+ pub use pollers::{
32
+ Client, ClientOptions, ClientOptionsBuilder, ClientTlsConfig, RetryClient, RetryConfig,
33
+ TlsConfig, WorkflowClientTrait,
34
+ };
35
+ pub use temporal_sdk_core_api as api;
36
+ pub use temporal_sdk_core_protos as protos;
37
+ pub use temporal_sdk_core_protos::TaskToken;
38
+ pub use url::Url;
39
+ pub use worker::{Worker, WorkerConfig, WorkerConfigBuilder};
40
+
41
+ use crate::{
42
+ replay::{mock_client_from_histories, Historator, HistoryForReplay},
43
+ telemetry::{
44
+ metrics::MetricsContext, remove_trace_subscriber_for_current_thread,
45
+ set_trace_subscriber_for_current_thread, telemetry_init, TelemetryInstance,
46
+ },
47
+ worker::client::WorkerClientBag,
48
+ };
49
+ use futures::Stream;
50
+ use std::sync::Arc;
51
+ use temporal_client::{ConfiguredClient, TemporalServiceClientWithMetrics};
52
+ use temporal_sdk_core_api::{
53
+ errors::{CompleteActivityError, PollActivityError, PollWfError},
54
+ telemetry::{CoreTelemetry, TelemetryOptions},
55
+ Worker as WorkerTrait,
56
+ };
57
+ use temporal_sdk_core_protos::coresdk::ActivityHeartbeat;
58
+
59
+ /// Initialize a worker bound to a task queue.
60
+ ///
61
+ /// You will need to have already initialized a [CoreRuntime] which will be used for this worker.
62
+ /// After the worker is initialized, you should use [CoreRuntime::tokio_handle] to run the worker's
63
+ /// async functions.
64
+ ///
65
+ /// Lang implementations may pass in a [temporal_client::ConfiguredClient] directly (or a
66
+ /// [RetryClient] wrapping one, or a handful of other variants of the same idea). When they do so,
67
+ /// this function will always overwrite the client retry configuration, force the client to use the
68
+ /// namespace defined in the worker config, and set the client identity appropriately. IE: Use
69
+ /// [ClientOptions::connect_no_namespace], not [ClientOptions::connect].
70
+ pub fn init_worker<CT>(
71
+ runtime: &CoreRuntime,
72
+ worker_config: WorkerConfig,
73
+ client: CT,
74
+ ) -> Result<Worker, anyhow::Error>
75
+ where
76
+ CT: Into<sealed::AnyClient>,
77
+ {
78
+ let client = {
79
+ let ll = client.into().into_inner();
80
+ let mut client = Client::new(*ll, worker_config.namespace.clone());
81
+ client.set_worker_build_id(worker_config.worker_build_id.clone());
82
+ if let Some(ref id_override) = worker_config.client_identity_override {
83
+ client.options_mut().identity = id_override.clone();
84
+ }
85
+ RetryClient::new(client, RetryConfig::default())
86
+ };
87
+ if client.namespace() != worker_config.namespace {
88
+ panic!("Passed in client is not bound to the same namespace as the worker");
89
+ }
90
+ let client_ident = client.get_options().identity.clone();
91
+ let sticky_q = sticky_q_name_for_worker(&client_ident, &worker_config);
92
+ let client_bag = Arc::new(WorkerClientBag::new(
93
+ client,
94
+ worker_config.namespace.clone(),
95
+ client_ident,
96
+ worker_config.worker_build_id.clone(),
97
+ worker_config.use_worker_versioning,
98
+ ));
99
+
100
+ let metrics = MetricsContext::top_level(worker_config.namespace.clone(), &runtime.telemetry)
101
+ .with_task_q(worker_config.task_queue.clone());
102
+ Ok(Worker::new(worker_config, sticky_q, client_bag, metrics))
103
+ }
104
+
105
+ /// Create a worker for replaying a specific history. It will auto-shutdown as soon as the history
106
+ /// has finished being replayed.
107
+ ///
108
+ /// You do not necessarily need a [CoreRuntime] for replay workers, but it's advisable to create
109
+ /// one and use it to run the replay worker's async functions the same way you would for a normal
110
+ /// worker.
111
+ pub fn init_replay_worker<I>(
112
+ mut config: WorkerConfig,
113
+ histories: I,
114
+ ) -> Result<Worker, anyhow::Error>
115
+ where
116
+ I: Stream<Item = HistoryForReplay> + Send + 'static,
117
+ {
118
+ info!(
119
+ task_queue = config.task_queue.as_str(),
120
+ "Registering replay worker"
121
+ );
122
+ config.max_cached_workflows = 1;
123
+ config.max_concurrent_wft_polls = 1;
124
+ config.no_remote_activities = true;
125
+ let historator = Historator::new(histories);
126
+ let post_activate = historator.get_post_activate_hook();
127
+ let shutdown_tok = historator.get_shutdown_setter();
128
+ let client = mock_client_from_histories(historator);
129
+ let mut worker = Worker::new(config, None, Arc::new(client), MetricsContext::no_op());
130
+ worker.set_post_activate_hook(post_activate);
131
+ shutdown_tok(worker.shutdown_token());
132
+ Ok(worker)
133
+ }
134
+
135
+ /// Creates a unique sticky queue name for a worker, iff the config allows for 1 or more cached
136
+ /// workflows.
137
+ pub(crate) fn sticky_q_name_for_worker(
138
+ process_identity: &str,
139
+ config: &WorkerConfig,
140
+ ) -> Option<String> {
141
+ if config.max_cached_workflows > 0 {
142
+ Some(format!(
143
+ "{}-{}-{}",
144
+ &process_identity,
145
+ &config.task_queue,
146
+ uuid::Uuid::new_v4().simple()
147
+ ))
148
+ } else {
149
+ None
150
+ }
151
+ }
152
+
153
+ mod sealed {
154
+ use super::*;
155
+
156
+ /// Allows passing different kinds of clients into things that want to be flexible. Motivating
157
+ /// use-case was worker initialization.
158
+ ///
159
+ /// Needs to exist in this crate to avoid blanket impl conflicts.
160
+ pub struct AnyClient(Box<ConfiguredClient<TemporalServiceClientWithMetrics>>);
161
+ impl AnyClient {
162
+ pub(crate) fn into_inner(self) -> Box<ConfiguredClient<TemporalServiceClientWithMetrics>> {
163
+ self.0
164
+ }
165
+ }
166
+
167
+ impl From<RetryClient<ConfiguredClient<TemporalServiceClientWithMetrics>>> for AnyClient {
168
+ fn from(c: RetryClient<ConfiguredClient<TemporalServiceClientWithMetrics>>) -> Self {
169
+ Self(Box::new(c.into_inner()))
170
+ }
171
+ }
172
+ impl From<RetryClient<Client>> for AnyClient {
173
+ fn from(c: RetryClient<Client>) -> Self {
174
+ Self(Box::new(c.into_inner().into_inner()))
175
+ }
176
+ }
177
+ impl From<Arc<RetryClient<Client>>> for AnyClient {
178
+ fn from(c: Arc<RetryClient<Client>>) -> Self {
179
+ Self(Box::new(c.get_client().inner().clone()))
180
+ }
181
+ }
182
+ impl From<ConfiguredClient<TemporalServiceClientWithMetrics>> for AnyClient {
183
+ fn from(c: ConfiguredClient<TemporalServiceClientWithMetrics>) -> Self {
184
+ Self(Box::new(c))
185
+ }
186
+ }
187
+ }
188
+
189
+ /// Holds shared state/components needed to back instances of workers and clients. More than one
190
+ /// may be instantiated, but typically only one is needed. More than one runtime instance may be
191
+ /// useful if multiple different telemetry settings are required.
192
+ pub struct CoreRuntime {
193
+ telemetry: TelemetryInstance,
194
+ runtime: Option<tokio::runtime::Runtime>,
195
+ runtime_handle: tokio::runtime::Handle,
196
+ }
197
+
198
+ impl CoreRuntime {
199
+ /// Create a new core runtime with the provided telemetry options and tokio runtime builder.
200
+ /// Also initialize telemetry for the thread this is being called on.
201
+ ///
202
+ /// Note that this function will call the [tokio::runtime::Builder::enable_all] builder option
203
+ /// on the Tokio runtime builder, and will call [tokio::runtime::Builder::on_thread_start] to
204
+ /// ensure telemetry subscribers are set on every tokio thread.
205
+ ///
206
+ /// **Important**: You need to call this *before* calling any async functions on workers or
207
+ /// clients, otherwise the tracing subscribers will not be properly attached.
208
+ ///
209
+ /// # Panics
210
+ /// If a tokio runtime has already been initialized. To re-use an existing runtime, call
211
+ /// [CoreRuntime::new_assume_tokio].
212
+ pub fn new(
213
+ telemetry_options: TelemetryOptions,
214
+ mut tokio_builder: tokio::runtime::Builder,
215
+ ) -> Result<Self, anyhow::Error> {
216
+ let telemetry = telemetry_init(telemetry_options)?;
217
+ let subscriber = telemetry.trace_subscriber();
218
+ let runtime = tokio_builder
219
+ .enable_all()
220
+ .on_thread_start(move || {
221
+ set_trace_subscriber_for_current_thread(subscriber.clone());
222
+ })
223
+ .build()?;
224
+ let _rg = runtime.enter();
225
+ let mut me = Self::new_assume_tokio_initialized_telem(telemetry);
226
+ me.runtime = Some(runtime);
227
+ Ok(me)
228
+ }
229
+
230
+ /// Initialize telemetry for the thread this is being called on, assuming a tokio runtime is
231
+ /// already active and this call exists in its context. See [Self::new] for more.
232
+ ///
233
+ /// # Panics
234
+ /// If there is no currently active Tokio runtime
235
+ pub fn new_assume_tokio(telemetry_options: TelemetryOptions) -> Result<Self, anyhow::Error> {
236
+ let telemetry = telemetry_init(telemetry_options)?;
237
+ Ok(Self::new_assume_tokio_initialized_telem(telemetry))
238
+ }
239
+
240
+ /// Construct a runtime from an already-initialized telemetry instance, assuming a tokio runtime
241
+ /// is already active and this call exists in its context. See [Self::new] for more.
242
+ ///
243
+ /// # Panics
244
+ /// If there is no currently active Tokio runtime
245
+ pub fn new_assume_tokio_initialized_telem(telemetry: TelemetryInstance) -> Self {
246
+ let runtime_handle = tokio::runtime::Handle::current();
247
+ set_trace_subscriber_for_current_thread(telemetry.trace_subscriber());
248
+ Self {
249
+ telemetry,
250
+ runtime: None,
251
+ runtime_handle,
252
+ }
253
+ }
254
+
255
+ /// Get a handle to the tokio runtime used by this Core runtime.
256
+ pub fn tokio_handle(&self) -> tokio::runtime::Handle {
257
+ self.runtime_handle.clone()
258
+ }
259
+
260
+ /// Returns the metric meter used for recording metrics, if they were enabled.
261
+ pub fn metric_meter(&self) -> Option<&opentelemetry::metrics::Meter> {
262
+ self.telemetry.get_metric_meter()
263
+ }
264
+
265
+ /// Return the trace subscriber associated with the telemetry options/instance. Can be used
266
+ /// to manually set the default for a thread or globally using the `tracing` crate, or with
267
+ /// [set_trace_subscriber_for_current_thread]
268
+ pub fn trace_subscriber(&self) -> Arc<dyn tracing::Subscriber + Send + Sync> {
269
+ self.telemetry.trace_subscriber()
270
+ }
271
+
272
+ /// Return a reference to the owned [TelemetryInstance]
273
+ pub fn telemetry(&self) -> &TelemetryInstance {
274
+ &self.telemetry
275
+ }
276
+ }
277
+
278
+ impl Drop for CoreRuntime {
279
+ fn drop(&mut self) {
280
+ remove_trace_subscriber_for_current_thread();
281
+ }
282
+ }
@@ -0,0 +1,54 @@
1
+ mod poll_buffer;
2
+
3
+ pub(crate) use poll_buffer::{
4
+ new_activity_task_buffer, new_workflow_task_buffer, WorkflowTaskPoller,
5
+ };
6
+ pub use temporal_client::{
7
+ Client, ClientOptions, ClientOptionsBuilder, ClientTlsConfig, RetryClient, RetryConfig,
8
+ TlsConfig, WorkflowClientTrait,
9
+ };
10
+ use temporal_sdk_core_protos::temporal::api::workflowservice::v1::{
11
+ PollActivityTaskQueueResponse, PollWorkflowTaskQueueResponse,
12
+ };
13
+
14
+ #[cfg(test)]
15
+ use futures::Future;
16
+
17
+ pub type Result<T, E = tonic::Status> = std::result::Result<T, E>;
18
+
19
+ /// A trait for things that poll the server. Hides complexity of concurrent polling or polling
20
+ /// on sticky/nonsticky queues simultaneously.
21
+ #[cfg_attr(test, mockall::automock)]
22
+ #[cfg_attr(test, allow(unused))]
23
+ #[async_trait::async_trait]
24
+ pub trait Poller<PollResult>
25
+ where
26
+ PollResult: Send + Sync + 'static,
27
+ {
28
+ async fn poll(&self) -> Option<Result<PollResult>>;
29
+ fn notify_shutdown(&self);
30
+ async fn shutdown(self);
31
+ /// Need a separate shutdown to be able to consume boxes :(
32
+ async fn shutdown_box(self: Box<Self>);
33
+ }
34
+ pub type BoxedPoller<T> = Box<dyn Poller<T> + Send + Sync + 'static>;
35
+ pub type BoxedWFPoller = BoxedPoller<PollWorkflowTaskQueueResponse>;
36
+ pub type BoxedActPoller = BoxedPoller<PollActivityTaskQueueResponse>;
37
+
38
+ #[cfg(test)]
39
+ mockall::mock! {
40
+ pub ManualPoller<T: Send + Sync + 'static> {}
41
+ #[allow(unused)]
42
+ impl<T: Send + Sync + 'static> Poller<T> for ManualPoller<T> {
43
+ fn poll<'a, 'b>(&self)
44
+ -> impl Future<Output = Option<Result<T>>> + Send + 'b
45
+ where 'a: 'b, Self: 'b;
46
+ fn notify_shutdown(&self);
47
+ fn shutdown<'a>(self)
48
+ -> impl Future<Output = ()> + Send + 'a
49
+ where Self: 'a;
50
+ fn shutdown_box<'a>(self: Box<Self>)
51
+ -> impl Future<Output = ()> + Send + 'a
52
+ where Self: 'a;
53
+ }
54
+ }
@@ -0,0 +1,297 @@
1
+ use crate::{
2
+ pollers::{self, Poller},
3
+ worker::client::WorkerClient,
4
+ };
5
+ use futures::{prelude::stream::FuturesUnordered, StreamExt};
6
+ use std::{
7
+ fmt::Debug,
8
+ future::Future,
9
+ sync::{
10
+ atomic::{AtomicUsize, Ordering},
11
+ Arc,
12
+ },
13
+ };
14
+ use temporal_sdk_core_protos::temporal::api::workflowservice::v1::{
15
+ PollActivityTaskQueueResponse, PollWorkflowTaskQueueResponse,
16
+ };
17
+ use tokio::{
18
+ sync::{
19
+ mpsc::{channel, Receiver},
20
+ Mutex, Semaphore,
21
+ },
22
+ task::JoinHandle,
23
+ };
24
+ use tokio_util::sync::CancellationToken;
25
+
26
+ pub struct LongPollBuffer<T> {
27
+ buffered_polls: Mutex<Receiver<pollers::Result<T>>>,
28
+ shutdown: CancellationToken,
29
+ /// This semaphore exists to ensure that we only poll server as many times as core actually
30
+ /// *asked* it to be polled - otherwise we might spin and buffer polls constantly. This also
31
+ /// means unit tests can continue to function in a predictable manner when calling mocks.
32
+ polls_requested: Arc<Semaphore>,
33
+ join_handles: FuturesUnordered<JoinHandle<()>>,
34
+ /// Called every time the number of pollers is changed
35
+ num_pollers_changed: Option<Box<dyn Fn(usize) + Send + Sync>>,
36
+ active_pollers: Arc<AtomicUsize>,
37
+ }
38
+
39
+ struct ActiveCounter<'a>(&'a AtomicUsize);
40
+ impl<'a> ActiveCounter<'a> {
41
+ fn new(a: &'a AtomicUsize) -> Self {
42
+ a.fetch_add(1, Ordering::Relaxed);
43
+ Self(a)
44
+ }
45
+ }
46
+ impl Drop for ActiveCounter<'_> {
47
+ fn drop(&mut self) {
48
+ self.0.fetch_sub(1, Ordering::Relaxed);
49
+ }
50
+ }
51
+
52
+ impl<T> LongPollBuffer<T>
53
+ where
54
+ T: Send + Debug + 'static,
55
+ {
56
+ pub fn new<FT>(
57
+ poll_fn: impl Fn() -> FT + Send + Sync + 'static,
58
+ max_pollers: usize,
59
+ buffer_size: usize,
60
+ shutdown: CancellationToken,
61
+ ) -> Self
62
+ where
63
+ FT: Future<Output = pollers::Result<T>> + Send,
64
+ {
65
+ let (tx, rx) = channel(buffer_size);
66
+ let polls_requested = Arc::new(Semaphore::new(0));
67
+ let active_pollers = Arc::new(AtomicUsize::new(0));
68
+ let join_handles = FuturesUnordered::new();
69
+ let pf = Arc::new(poll_fn);
70
+ for _ in 0..max_pollers {
71
+ let tx = tx.clone();
72
+ let pf = pf.clone();
73
+ let shutdown = shutdown.clone();
74
+ let polls_requested = polls_requested.clone();
75
+ let ap = active_pollers.clone();
76
+ let jh = tokio::spawn(async move {
77
+ loop {
78
+ if shutdown.is_cancelled() {
79
+ break;
80
+ }
81
+ let sp = tokio::select! {
82
+ sp = polls_requested.acquire() => sp.expect("Polls semaphore not dropped"),
83
+ _ = shutdown.cancelled() => continue,
84
+ };
85
+ let _active_guard = ActiveCounter::new(ap.as_ref());
86
+ let r = tokio::select! {
87
+ r = pf() => r,
88
+ _ = shutdown.cancelled() => continue,
89
+ };
90
+ sp.forget();
91
+ let _ = tx.send(r).await;
92
+ }
93
+ });
94
+ join_handles.push(jh);
95
+ }
96
+ Self {
97
+ buffered_polls: Mutex::new(rx),
98
+ shutdown,
99
+ polls_requested,
100
+ join_handles,
101
+ num_pollers_changed: None,
102
+ active_pollers,
103
+ }
104
+ }
105
+
106
+ /// Set a function that will be called every time the number of pollers changes.
107
+ /// TODO: Currently a bit weird, will make more sense once we implement dynamic poller scaling.
108
+ pub fn set_num_pollers_handler(&mut self, handler: impl Fn(usize) + Send + Sync + 'static) {
109
+ self.num_pollers_changed = Some(Box::new(handler));
110
+ }
111
+ }
112
+
113
+ #[async_trait::async_trait]
114
+ impl<T> Poller<T> for LongPollBuffer<T>
115
+ where
116
+ T: Send + Sync + Debug + 'static,
117
+ {
118
+ /// Poll the buffer. Adds one permit to the polling pool - the point of this being that the
119
+ /// buffer may support many concurrent pollers, but there is no reason to have them poll unless
120
+ /// enough polls have actually been requested. Calling this function adds a permit that any
121
+ /// concurrent poller may fulfill.
122
+ ///
123
+ /// EX: If this function is only ever called serially and always `await`ed, there will be no
124
+ /// concurrent polling. If it is called many times and the futures are awaited concurrently,
125
+ /// then polling will happen concurrently.
126
+ ///
127
+ /// Returns `None` if the poll buffer has been shut down
128
+ #[instrument(name = "long_poll", level = "trace", skip(self))]
129
+ async fn poll(&self) -> Option<pollers::Result<T>> {
130
+ self.polls_requested.add_permits(1);
131
+ if let Some(fun) = self.num_pollers_changed.as_ref() {
132
+ fun(self.active_pollers.load(Ordering::Relaxed));
133
+ }
134
+
135
+ let mut locked = self.buffered_polls.lock().await;
136
+ let res = (*locked).recv().await;
137
+
138
+ if let Some(fun) = self.num_pollers_changed.as_ref() {
139
+ fun(self.active_pollers.load(Ordering::Relaxed));
140
+ }
141
+
142
+ res
143
+ }
144
+
145
+ fn notify_shutdown(&self) {
146
+ self.shutdown.cancel();
147
+ }
148
+
149
+ async fn shutdown(mut self) {
150
+ self.notify_shutdown();
151
+ while self.join_handles.next().await.is_some() {}
152
+ }
153
+
154
+ async fn shutdown_box(self: Box<Self>) {
155
+ let this = *self;
156
+ this.shutdown().await;
157
+ }
158
+ }
159
+
160
+ /// A poller capable of polling on a sticky and a nonsticky queue simultaneously for workflow tasks.
161
+ #[derive(derive_more::Constructor)]
162
+ pub struct WorkflowTaskPoller {
163
+ normal_poller: PollWorkflowTaskBuffer,
164
+ sticky_poller: Option<PollWorkflowTaskBuffer>,
165
+ }
166
+
167
+ #[async_trait::async_trait]
168
+ impl Poller<PollWorkflowTaskQueueResponse> for WorkflowTaskPoller {
169
+ async fn poll(&self) -> Option<pollers::Result<PollWorkflowTaskQueueResponse>> {
170
+ if let Some(sq) = self.sticky_poller.as_ref() {
171
+ tokio::select! {
172
+ r = self.normal_poller.poll() => r,
173
+ r = sq.poll() => r,
174
+ }
175
+ } else {
176
+ self.normal_poller.poll().await
177
+ }
178
+ }
179
+
180
+ fn notify_shutdown(&self) {
181
+ self.normal_poller.notify_shutdown();
182
+ if let Some(sq) = self.sticky_poller.as_ref() {
183
+ sq.notify_shutdown();
184
+ }
185
+ }
186
+
187
+ async fn shutdown(mut self) {
188
+ self.normal_poller.shutdown().await;
189
+ if let Some(sq) = self.sticky_poller {
190
+ sq.shutdown().await;
191
+ }
192
+ }
193
+
194
+ async fn shutdown_box(self: Box<Self>) {
195
+ let this = *self;
196
+ this.shutdown().await;
197
+ }
198
+ }
199
+
200
+ pub type PollWorkflowTaskBuffer = LongPollBuffer<PollWorkflowTaskQueueResponse>;
201
+ pub(crate) fn new_workflow_task_buffer(
202
+ client: Arc<dyn WorkerClient>,
203
+ task_queue: String,
204
+ is_sticky: bool,
205
+ concurrent_pollers: usize,
206
+ buffer_size: usize,
207
+ shutdown: CancellationToken,
208
+ ) -> PollWorkflowTaskBuffer {
209
+ LongPollBuffer::new(
210
+ move || {
211
+ let client = client.clone();
212
+ let task_queue = task_queue.clone();
213
+ async move { client.poll_workflow_task(task_queue, is_sticky).await }
214
+ },
215
+ concurrent_pollers,
216
+ buffer_size,
217
+ shutdown,
218
+ )
219
+ }
220
+
221
+ pub type PollActivityTaskBuffer = LongPollBuffer<PollActivityTaskQueueResponse>;
222
+ pub(crate) fn new_activity_task_buffer(
223
+ client: Arc<dyn WorkerClient>,
224
+ task_queue: String,
225
+ concurrent_pollers: usize,
226
+ buffer_size: usize,
227
+ max_tps: Option<f64>,
228
+ shutdown: CancellationToken,
229
+ ) -> PollActivityTaskBuffer {
230
+ LongPollBuffer::new(
231
+ move || {
232
+ let client = client.clone();
233
+ let task_queue = task_queue.clone();
234
+ async move { client.poll_activity_task(task_queue, max_tps).await }
235
+ },
236
+ concurrent_pollers,
237
+ buffer_size,
238
+ shutdown,
239
+ )
240
+ }
241
+
242
+ #[cfg(test)]
243
+ mod tests {
244
+ use super::*;
245
+ use crate::worker::client::mocks::mock_manual_workflow_client;
246
+ use futures::FutureExt;
247
+ use std::time::Duration;
248
+ use tokio::{select, sync::mpsc::channel};
249
+
250
+ #[tokio::test]
251
+ async fn only_polls_once_with_1_poller() {
252
+ let mut mock_client = mock_manual_workflow_client();
253
+ mock_client
254
+ .expect_poll_workflow_task()
255
+ .times(2)
256
+ .returning(move |_, _| {
257
+ async {
258
+ tokio::time::sleep(Duration::from_millis(100)).await;
259
+ Ok(Default::default())
260
+ }
261
+ .boxed()
262
+ });
263
+
264
+ let pb = new_workflow_task_buffer(
265
+ Arc::new(mock_client),
266
+ "someq".to_string(),
267
+ false,
268
+ 1,
269
+ 1,
270
+ CancellationToken::new(),
271
+ );
272
+
273
+ // Poll a bunch of times, "interrupting" it each time, we should only actually have polled
274
+ // once since the poll takes a while
275
+ let (interrupter_tx, mut interrupter_rx) = channel(50);
276
+ for _ in 0..10 {
277
+ interrupter_tx.send(()).await.unwrap();
278
+ }
279
+
280
+ // We should never get anything out since we interrupted 100% of polls
281
+ let mut last_val = false;
282
+ for _ in 0..10 {
283
+ select! {
284
+ _ = interrupter_rx.recv() => {
285
+ last_val = true;
286
+ }
287
+ _ = pb.poll() => {
288
+ }
289
+ }
290
+ }
291
+ assert!(last_val);
292
+ // Now we grab the buffered poll response, the poll task will go again but we don't grab it,
293
+ // therefore we will have only polled twice.
294
+ pb.poll().await.unwrap().unwrap();
295
+ pb.shutdown().await;
296
+ }
297
+ }