clara-temporalio 0.4.3-x86_64-darwin

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 (190) hide show
  1. checksums.yaml +7 -0
  2. data/.yardopts +2 -0
  3. data/Gemfile +27 -0
  4. data/Rakefile +101 -0
  5. data/lib/temporalio/activity/cancellation_details.rb +58 -0
  6. data/lib/temporalio/activity/complete_async_error.rb +11 -0
  7. data/lib/temporalio/activity/context.rb +131 -0
  8. data/lib/temporalio/activity/definition.rb +197 -0
  9. data/lib/temporalio/activity/info.rb +70 -0
  10. data/lib/temporalio/activity.rb +14 -0
  11. data/lib/temporalio/api/activity/v1/message.rb +25 -0
  12. data/lib/temporalio/api/batch/v1/message.rb +38 -0
  13. data/lib/temporalio/api/cloud/account/v1/message.rb +28 -0
  14. data/lib/temporalio/api/cloud/cloudservice/v1/request_response.rb +135 -0
  15. data/lib/temporalio/api/cloud/cloudservice/v1/service.rb +25 -0
  16. data/lib/temporalio/api/cloud/cloudservice.rb +3 -0
  17. data/lib/temporalio/api/cloud/identity/v1/message.rb +46 -0
  18. data/lib/temporalio/api/cloud/namespace/v1/message.rb +46 -0
  19. data/lib/temporalio/api/cloud/nexus/v1/message.rb +32 -0
  20. data/lib/temporalio/api/cloud/operation/v1/message.rb +28 -0
  21. data/lib/temporalio/api/cloud/region/v1/message.rb +24 -0
  22. data/lib/temporalio/api/cloud/resource/v1/message.rb +23 -0
  23. data/lib/temporalio/api/cloud/sink/v1/message.rb +24 -0
  24. data/lib/temporalio/api/cloud/usage/v1/message.rb +31 -0
  25. data/lib/temporalio/api/command/v1/message.rb +46 -0
  26. data/lib/temporalio/api/common/v1/grpc_status.rb +23 -0
  27. data/lib/temporalio/api/common/v1/message.rb +49 -0
  28. data/lib/temporalio/api/deployment/v1/message.rb +39 -0
  29. data/lib/temporalio/api/enums/v1/batch_operation.rb +22 -0
  30. data/lib/temporalio/api/enums/v1/command_type.rb +21 -0
  31. data/lib/temporalio/api/enums/v1/common.rb +28 -0
  32. data/lib/temporalio/api/enums/v1/deployment.rb +23 -0
  33. data/lib/temporalio/api/enums/v1/event_type.rb +21 -0
  34. data/lib/temporalio/api/enums/v1/failed_cause.rb +26 -0
  35. data/lib/temporalio/api/enums/v1/namespace.rb +23 -0
  36. data/lib/temporalio/api/enums/v1/nexus.rb +21 -0
  37. data/lib/temporalio/api/enums/v1/query.rb +22 -0
  38. data/lib/temporalio/api/enums/v1/reset.rb +23 -0
  39. data/lib/temporalio/api/enums/v1/schedule.rb +21 -0
  40. data/lib/temporalio/api/enums/v1/task_queue.rb +25 -0
  41. data/lib/temporalio/api/enums/v1/update.rb +22 -0
  42. data/lib/temporalio/api/enums/v1/workflow.rb +31 -0
  43. data/lib/temporalio/api/errordetails/v1/message.rb +44 -0
  44. data/lib/temporalio/api/export/v1/message.rb +24 -0
  45. data/lib/temporalio/api/failure/v1/message.rb +38 -0
  46. data/lib/temporalio/api/filter/v1/message.rb +27 -0
  47. data/lib/temporalio/api/history/v1/message.rb +94 -0
  48. data/lib/temporalio/api/namespace/v1/message.rb +31 -0
  49. data/lib/temporalio/api/nexus/v1/message.rb +41 -0
  50. data/lib/temporalio/api/operatorservice/v1/request_response.rb +49 -0
  51. data/lib/temporalio/api/operatorservice/v1/service.rb +23 -0
  52. data/lib/temporalio/api/operatorservice.rb +3 -0
  53. data/lib/temporalio/api/payload_visitor.rb +1668 -0
  54. data/lib/temporalio/api/protocol/v1/message.rb +23 -0
  55. data/lib/temporalio/api/query/v1/message.rb +28 -0
  56. data/lib/temporalio/api/replication/v1/message.rb +26 -0
  57. data/lib/temporalio/api/rules/v1/message.rb +27 -0
  58. data/lib/temporalio/api/schedule/v1/message.rb +43 -0
  59. data/lib/temporalio/api/sdk/v1/enhanced_stack_trace.rb +25 -0
  60. data/lib/temporalio/api/sdk/v1/task_complete_metadata.rb +21 -0
  61. data/lib/temporalio/api/sdk/v1/user_metadata.rb +23 -0
  62. data/lib/temporalio/api/sdk/v1/workflow_metadata.rb +23 -0
  63. data/lib/temporalio/api/taskqueue/v1/message.rb +48 -0
  64. data/lib/temporalio/api/testservice/v1/request_response.rb +31 -0
  65. data/lib/temporalio/api/testservice/v1/service.rb +23 -0
  66. data/lib/temporalio/api/update/v1/message.rb +33 -0
  67. data/lib/temporalio/api/version/v1/message.rb +26 -0
  68. data/lib/temporalio/api/workflow/v1/message.rb +63 -0
  69. data/lib/temporalio/api/workflowservice/v1/request_response.rb +244 -0
  70. data/lib/temporalio/api/workflowservice/v1/service.rb +23 -0
  71. data/lib/temporalio/api/workflowservice.rb +3 -0
  72. data/lib/temporalio/api.rb +15 -0
  73. data/lib/temporalio/cancellation.rb +170 -0
  74. data/lib/temporalio/client/activity_id_reference.rb +32 -0
  75. data/lib/temporalio/client/async_activity_handle.rb +85 -0
  76. data/lib/temporalio/client/connection/cloud_service.rb +786 -0
  77. data/lib/temporalio/client/connection/operator_service.rb +201 -0
  78. data/lib/temporalio/client/connection/service.rb +42 -0
  79. data/lib/temporalio/client/connection/test_service.rb +111 -0
  80. data/lib/temporalio/client/connection/workflow_service.rb +1326 -0
  81. data/lib/temporalio/client/connection.rb +316 -0
  82. data/lib/temporalio/client/interceptor.rb +457 -0
  83. data/lib/temporalio/client/schedule.rb +991 -0
  84. data/lib/temporalio/client/schedule_handle.rb +126 -0
  85. data/lib/temporalio/client/with_start_workflow_operation.rb +115 -0
  86. data/lib/temporalio/client/workflow_execution.rb +119 -0
  87. data/lib/temporalio/client/workflow_execution_count.rb +36 -0
  88. data/lib/temporalio/client/workflow_execution_status.rb +18 -0
  89. data/lib/temporalio/client/workflow_handle.rb +389 -0
  90. data/lib/temporalio/client/workflow_query_reject_condition.rb +14 -0
  91. data/lib/temporalio/client/workflow_update_handle.rb +65 -0
  92. data/lib/temporalio/client/workflow_update_wait_stage.rb +17 -0
  93. data/lib/temporalio/client.rb +625 -0
  94. data/lib/temporalio/common_enums.rb +55 -0
  95. data/lib/temporalio/contrib/open_telemetry.rb +469 -0
  96. data/lib/temporalio/converters/data_converter.rb +99 -0
  97. data/lib/temporalio/converters/failure_converter.rb +205 -0
  98. data/lib/temporalio/converters/payload_codec.rb +26 -0
  99. data/lib/temporalio/converters/payload_converter/binary_null.rb +34 -0
  100. data/lib/temporalio/converters/payload_converter/binary_plain.rb +35 -0
  101. data/lib/temporalio/converters/payload_converter/binary_protobuf.rb +42 -0
  102. data/lib/temporalio/converters/payload_converter/composite.rb +66 -0
  103. data/lib/temporalio/converters/payload_converter/encoding.rb +35 -0
  104. data/lib/temporalio/converters/payload_converter/json_plain.rb +44 -0
  105. data/lib/temporalio/converters/payload_converter/json_protobuf.rb +41 -0
  106. data/lib/temporalio/converters/payload_converter.rb +71 -0
  107. data/lib/temporalio/converters/raw_value.rb +20 -0
  108. data/lib/temporalio/converters.rb +9 -0
  109. data/lib/temporalio/error/failure.rb +237 -0
  110. data/lib/temporalio/error.rb +156 -0
  111. data/lib/temporalio/internal/bridge/3.2/temporalio_bridge.bundle +0 -0
  112. data/lib/temporalio/internal/bridge/3.3/temporalio_bridge.bundle +0 -0
  113. data/lib/temporalio/internal/bridge/3.4/temporalio_bridge.bundle +0 -0
  114. data/lib/temporalio/internal/bridge/api/activity_result/activity_result.rb +34 -0
  115. data/lib/temporalio/internal/bridge/api/activity_task/activity_task.rb +32 -0
  116. data/lib/temporalio/internal/bridge/api/child_workflow/child_workflow.rb +33 -0
  117. data/lib/temporalio/internal/bridge/api/common/common.rb +27 -0
  118. data/lib/temporalio/internal/bridge/api/core_interface.rb +40 -0
  119. data/lib/temporalio/internal/bridge/api/external_data/external_data.rb +27 -0
  120. data/lib/temporalio/internal/bridge/api/nexus/nexus.rb +34 -0
  121. data/lib/temporalio/internal/bridge/api/workflow_activation/workflow_activation.rb +56 -0
  122. data/lib/temporalio/internal/bridge/api/workflow_commands/workflow_commands.rb +58 -0
  123. data/lib/temporalio/internal/bridge/api/workflow_completion/workflow_completion.rb +31 -0
  124. data/lib/temporalio/internal/bridge/api.rb +3 -0
  125. data/lib/temporalio/internal/bridge/client.rb +95 -0
  126. data/lib/temporalio/internal/bridge/runtime.rb +56 -0
  127. data/lib/temporalio/internal/bridge/testing.rb +69 -0
  128. data/lib/temporalio/internal/bridge/worker.rb +109 -0
  129. data/lib/temporalio/internal/bridge.rb +36 -0
  130. data/lib/temporalio/internal/client/implementation.rb +926 -0
  131. data/lib/temporalio/internal/metric.rb +122 -0
  132. data/lib/temporalio/internal/proto_utils.rb +165 -0
  133. data/lib/temporalio/internal/worker/activity_worker.rb +448 -0
  134. data/lib/temporalio/internal/worker/multi_runner.rb +213 -0
  135. data/lib/temporalio/internal/worker/workflow_instance/child_workflow_handle.rb +54 -0
  136. data/lib/temporalio/internal/worker/workflow_instance/context.rb +391 -0
  137. data/lib/temporalio/internal/worker/workflow_instance/details.rb +49 -0
  138. data/lib/temporalio/internal/worker/workflow_instance/external_workflow_handle.rb +32 -0
  139. data/lib/temporalio/internal/worker/workflow_instance/externally_immutable_hash.rb +22 -0
  140. data/lib/temporalio/internal/worker/workflow_instance/handler_execution.rb +25 -0
  141. data/lib/temporalio/internal/worker/workflow_instance/handler_hash.rb +41 -0
  142. data/lib/temporalio/internal/worker/workflow_instance/illegal_call_tracer.rb +97 -0
  143. data/lib/temporalio/internal/worker/workflow_instance/inbound_implementation.rb +62 -0
  144. data/lib/temporalio/internal/worker/workflow_instance/outbound_implementation.rb +404 -0
  145. data/lib/temporalio/internal/worker/workflow_instance/replay_safe_logger.rb +37 -0
  146. data/lib/temporalio/internal/worker/workflow_instance/replay_safe_metric.rb +40 -0
  147. data/lib/temporalio/internal/worker/workflow_instance/scheduler.rb +183 -0
  148. data/lib/temporalio/internal/worker/workflow_instance.rb +800 -0
  149. data/lib/temporalio/internal/worker/workflow_worker.rb +249 -0
  150. data/lib/temporalio/internal.rb +7 -0
  151. data/lib/temporalio/metric.rb +109 -0
  152. data/lib/temporalio/priority.rb +59 -0
  153. data/lib/temporalio/retry_policy.rb +74 -0
  154. data/lib/temporalio/runtime/metric_buffer.rb +94 -0
  155. data/lib/temporalio/runtime.rb +352 -0
  156. data/lib/temporalio/scoped_logger.rb +96 -0
  157. data/lib/temporalio/search_attributes.rb +356 -0
  158. data/lib/temporalio/testing/activity_environment.rb +175 -0
  159. data/lib/temporalio/testing/workflow_environment.rb +406 -0
  160. data/lib/temporalio/testing.rb +10 -0
  161. data/lib/temporalio/version.rb +5 -0
  162. data/lib/temporalio/versioning_override.rb +55 -0
  163. data/lib/temporalio/worker/activity_executor/fiber.rb +49 -0
  164. data/lib/temporalio/worker/activity_executor/thread_pool.rb +46 -0
  165. data/lib/temporalio/worker/activity_executor.rb +55 -0
  166. data/lib/temporalio/worker/deployment_options.rb +45 -0
  167. data/lib/temporalio/worker/interceptor.rb +367 -0
  168. data/lib/temporalio/worker/poller_behavior.rb +61 -0
  169. data/lib/temporalio/worker/thread_pool.rb +237 -0
  170. data/lib/temporalio/worker/tuner.rb +189 -0
  171. data/lib/temporalio/worker/workflow_executor/thread_pool.rb +236 -0
  172. data/lib/temporalio/worker/workflow_executor.rb +26 -0
  173. data/lib/temporalio/worker/workflow_replayer.rb +349 -0
  174. data/lib/temporalio/worker.rb +633 -0
  175. data/lib/temporalio/worker_deployment_version.rb +67 -0
  176. data/lib/temporalio/workflow/activity_cancellation_type.rb +20 -0
  177. data/lib/temporalio/workflow/child_workflow_cancellation_type.rb +21 -0
  178. data/lib/temporalio/workflow/child_workflow_handle.rb +43 -0
  179. data/lib/temporalio/workflow/definition.rb +680 -0
  180. data/lib/temporalio/workflow/external_workflow_handle.rb +41 -0
  181. data/lib/temporalio/workflow/future.rb +151 -0
  182. data/lib/temporalio/workflow/handler_unfinished_policy.rb +13 -0
  183. data/lib/temporalio/workflow/info.rb +107 -0
  184. data/lib/temporalio/workflow/parent_close_policy.rb +19 -0
  185. data/lib/temporalio/workflow/update_info.rb +20 -0
  186. data/lib/temporalio/workflow.rb +594 -0
  187. data/lib/temporalio/workflow_history.rb +47 -0
  188. data/lib/temporalio.rb +12 -0
  189. data/temporalio.gemspec +31 -0
  190. metadata +267 -0
@@ -0,0 +1,367 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Temporalio
4
+ class Worker
5
+ module Interceptor
6
+ # Mixin for intercepting activity worker work. Clases that `include` may implement their own {intercept_activity}
7
+ # that returns their own instance of {Inbound}.
8
+ #
9
+ # @note Input classes herein may get new required fields added and therefore the constructors of the Input classes
10
+ # may change in backwards incompatible ways. Users should not try to construct Input classes themselves.
11
+ module Activity
12
+ # Method called when intercepting an activity. This is called when starting an activity attempt.
13
+ #
14
+ # @param next_interceptor [Inbound] Next interceptor in the chain that should be called. This is usually passed
15
+ # to {Inbound} constructor.
16
+ # @return [Inbound] Interceptor to be called for activity calls.
17
+ def intercept_activity(next_interceptor)
18
+ next_interceptor
19
+ end
20
+
21
+ # Input for {Inbound.execute}.
22
+ ExecuteInput = Data.define(
23
+ :proc,
24
+ :args,
25
+ :headers
26
+ )
27
+
28
+ # Inbound interceptor for intercepting inbound activity calls. This should be extended by users needing to
29
+ # intercept activities.
30
+ class Inbound
31
+ # @return [Inbound] Next interceptor in the chain.
32
+ attr_reader :next_interceptor
33
+
34
+ # Initialize inbound with the next interceptor in the chain.
35
+ #
36
+ # @param next_interceptor [Inbound] Next interceptor in the chain.
37
+ def initialize(next_interceptor)
38
+ @next_interceptor = next_interceptor
39
+ end
40
+
41
+ # Initialize the outbound interceptor. This should be extended by users to return their own {Outbound}
42
+ # implementation that wraps the parameter here.
43
+ #
44
+ # @param outbound [Outbound] Next outbound interceptor in the chain.
45
+ # @return [Outbound] Outbound activity interceptor.
46
+ def init(outbound)
47
+ @next_interceptor.init(outbound)
48
+ end
49
+
50
+ # Execute an activity and return result or raise exception. Next interceptor in chain (i.e. `super`) will
51
+ # perform the execution.
52
+ #
53
+ # @param input [ExecuteInput] Input information.
54
+ # @return [Object] Activity result.
55
+ def execute(input)
56
+ @next_interceptor.execute(input)
57
+ end
58
+ end
59
+
60
+ # Input for {Outbound.heartbeat}.
61
+ HeartbeatInput = Data.define(
62
+ :details
63
+ )
64
+
65
+ # Outbound interceptor for intercepting outbound activity calls. This should be extended by users needing to
66
+ # intercept activity calls.
67
+ class Outbound
68
+ # @return [Outbound] Next interceptor in the chain.
69
+ attr_reader :next_interceptor
70
+
71
+ # Initialize outbound with the next interceptor in the chain.
72
+ #
73
+ # @param next_interceptor [Outbound] Next interceptor in the chain.
74
+ def initialize(next_interceptor)
75
+ @next_interceptor = next_interceptor
76
+ end
77
+
78
+ # Issue a heartbeat.
79
+ #
80
+ # @param input [HeartbeatInput] Input information.
81
+ def heartbeat(input)
82
+ @next_interceptor.heartbeat(input)
83
+ end
84
+ end
85
+ end
86
+
87
+ # Mixin for intercepting workflow worker work. Classes that `include` may implement their own {intercept_workflow}
88
+ # that returns their own instance of {Inbound}.
89
+ #
90
+ # @note Input classes herein may get new required fields added and therefore the constructors of the Input classes
91
+ # may change in backwards incompatible ways. Users should not try to construct Input classes themselves.
92
+ module Workflow
93
+ # Method called when intercepting a workflow. This is called when creating a workflow instance.
94
+ #
95
+ # @param next_interceptor [Inbound] Next interceptor in the chain that should be called. This is usually passed
96
+ # to {Inbound} constructor.
97
+ # @return [Inbound] Interceptor to be called for workflow calls.
98
+ def intercept_workflow(next_interceptor)
99
+ next_interceptor
100
+ end
101
+
102
+ # Input for {Inbound.execute}.
103
+ ExecuteInput = Data.define(
104
+ :args,
105
+ :headers
106
+ )
107
+
108
+ # Input for {Inbound.handle_signal}.
109
+ HandleSignalInput = Data.define(
110
+ :signal,
111
+ :args,
112
+ :definition,
113
+ :headers
114
+ )
115
+
116
+ # Input for {Inbound.handle_query}.
117
+ HandleQueryInput = Data.define(
118
+ :id,
119
+ :query,
120
+ :args,
121
+ :definition,
122
+ :headers
123
+ )
124
+
125
+ # Input for {Inbound.validate_update} and {Inbound.handle_update}.
126
+ HandleUpdateInput = Data.define(
127
+ :id,
128
+ :update,
129
+ :args,
130
+ :definition,
131
+ :headers
132
+ )
133
+
134
+ # Inbound interceptor for intercepting inbound workflow calls. This should be extended by users needing to
135
+ # intercept workflows.
136
+ class Inbound
137
+ # @return [Inbound] Next interceptor in the chain.
138
+ attr_reader :next_interceptor
139
+
140
+ # Initialize inbound with the next interceptor in the chain.
141
+ #
142
+ # @param next_interceptor [Inbound] Next interceptor in the chain.
143
+ def initialize(next_interceptor)
144
+ @next_interceptor = next_interceptor
145
+ end
146
+
147
+ # Initialize the outbound interceptor. This should be extended by users to return their own {Outbound}
148
+ # implementation that wraps the parameter here.
149
+ #
150
+ # @param outbound [Outbound] Next outbound interceptor in the chain.
151
+ # @return [Outbound] Outbound workflow interceptor.
152
+ def init(outbound)
153
+ @next_interceptor.init(outbound)
154
+ end
155
+
156
+ # Execute a workflow and return result or raise exception. Next interceptor in chain (i.e. `super`) will
157
+ # perform the execution.
158
+ #
159
+ # @param input [ExecuteInput] Input information.
160
+ # @return [Object] Workflow result.
161
+ def execute(input)
162
+ @next_interceptor.execute(input)
163
+ end
164
+
165
+ # Handle a workflow signal. Next interceptor in chain (i.e. `super`) will perform the handling.
166
+ #
167
+ # @param input [HandleSignalInput] Input information.
168
+ def handle_signal(input)
169
+ @next_interceptor.handle_signal(input)
170
+ end
171
+
172
+ # Handle a workflow query and return result or raise exception. Next interceptor in chain (i.e. `super`) will
173
+ # perform the handling.
174
+ #
175
+ # @param input [HandleQueryInput] Input information.
176
+ # @return [Object] Query result.
177
+ def handle_query(input)
178
+ @next_interceptor.handle_query(input)
179
+ end
180
+
181
+ # Validate a workflow update. Next interceptor in chain (i.e. `super`) will perform the validation.
182
+ #
183
+ # @param input [HandleUpdateInput] Input information.
184
+ def validate_update(input)
185
+ @next_interceptor.validate_update(input)
186
+ end
187
+
188
+ # Handle a workflow update and return result or raise exception. Next interceptor in chain (i.e. `super`) will
189
+ # perform the handling.
190
+ #
191
+ # @param input [HandleUpdateInput] Input information.
192
+ # @return [Object] Update result.
193
+ def handle_update(input)
194
+ @next_interceptor.handle_update(input)
195
+ end
196
+ end
197
+
198
+ # Input for {Outbound.cancel_external_workflow}.
199
+ CancelExternalWorkflowInput = Data.define(
200
+ :id,
201
+ :run_id
202
+ )
203
+
204
+ # Input for {Outbound.execute_activity}.
205
+ ExecuteActivityInput = Data.define(
206
+ :activity,
207
+ :args,
208
+ :task_queue,
209
+ :summary,
210
+ :schedule_to_close_timeout,
211
+ :schedule_to_start_timeout,
212
+ :start_to_close_timeout,
213
+ :heartbeat_timeout,
214
+ :retry_policy,
215
+ :cancellation,
216
+ :cancellation_type,
217
+ :activity_id,
218
+ :disable_eager_execution,
219
+ :headers,
220
+ :priority
221
+ )
222
+
223
+ # Input for {Outbound.execute_local_activity}.
224
+ ExecuteLocalActivityInput = Data.define(
225
+ :activity,
226
+ :args,
227
+ :schedule_to_close_timeout,
228
+ :schedule_to_start_timeout,
229
+ :start_to_close_timeout,
230
+ :retry_policy,
231
+ :local_retry_threshold,
232
+ :cancellation,
233
+ :cancellation_type,
234
+ :activity_id,
235
+ :headers
236
+ )
237
+
238
+ # Input for {Outbound.initialize_continue_as_new_error}.
239
+ InitializeContinueAsNewErrorInput = Data.define(
240
+ :error
241
+ )
242
+
243
+ # Input for {Outbound.signal_child_workflow}.
244
+ SignalChildWorkflowInput = Data.define(
245
+ :id,
246
+ :signal,
247
+ :args,
248
+ :cancellation,
249
+ :headers
250
+ )
251
+
252
+ # Input for {Outbound.signal_external_workflow}.
253
+ SignalExternalWorkflowInput = Data.define(
254
+ :id,
255
+ :run_id,
256
+ :signal,
257
+ :args,
258
+ :cancellation,
259
+ :headers
260
+ )
261
+
262
+ # Input for {Outbound.sleep}.
263
+ SleepInput = Data.define(
264
+ :duration,
265
+ :summary,
266
+ :cancellation
267
+ )
268
+
269
+ # Input for {Outbound.start_child_workflow}.
270
+ StartChildWorkflowInput = Data.define(
271
+ :workflow,
272
+ :args,
273
+ :id,
274
+ :task_queue,
275
+ :static_summary,
276
+ :static_details,
277
+ :cancellation,
278
+ :cancellation_type,
279
+ :parent_close_policy,
280
+ :execution_timeout,
281
+ :run_timeout,
282
+ :task_timeout,
283
+ :id_reuse_policy,
284
+ :retry_policy,
285
+ :cron_schedule,
286
+ :memo,
287
+ :search_attributes,
288
+ :headers,
289
+ :priority
290
+ )
291
+
292
+ # Outbound interceptor for intercepting outbound workflow calls. This should be extended by users needing to
293
+ # intercept workflow calls.
294
+ class Outbound
295
+ # @return [Outbound] Next interceptor in the chain.
296
+ attr_reader :next_interceptor
297
+
298
+ # Initialize outbound with the next interceptor in the chain.
299
+ #
300
+ # @param next_interceptor [Outbound] Next interceptor in the chain.
301
+ def initialize(next_interceptor)
302
+ @next_interceptor = next_interceptor
303
+ end
304
+
305
+ # Cancel external workflow.
306
+ #
307
+ # @param input [CancelExternalWorkflowInput] Input.
308
+ def cancel_external_workflow(input)
309
+ @next_interceptor.cancel_external_workflow(input)
310
+ end
311
+
312
+ # Execute activity.
313
+ #
314
+ # @param input [ExecuteActivityInput] Input.
315
+ # @return [Object] Activity result.
316
+ def execute_activity(input)
317
+ @next_interceptor.execute_activity(input)
318
+ end
319
+
320
+ # Execute local activity.
321
+ #
322
+ # @param input [ExecuteLocalActivityInput] Input.
323
+ # @return [Object] Activity result.
324
+ def execute_local_activity(input)
325
+ @next_interceptor.execute_local_activity(input)
326
+ end
327
+
328
+ # Initialize continue as new error.
329
+ #
330
+ # @param input [InitializeContinueAsNewErrorInput] Input.
331
+ def initialize_continue_as_new_error(input)
332
+ @next_interceptor.initialize_continue_as_new_error(input)
333
+ end
334
+
335
+ # Signal child workflow.
336
+ #
337
+ # @param input [SignalChildWorkflowInput] Input.
338
+ def signal_child_workflow(input)
339
+ @next_interceptor.signal_child_workflow(input)
340
+ end
341
+
342
+ # Signal external workflow.
343
+ #
344
+ # @param input [SignalExternalWorkflowInput] Input.
345
+ def signal_external_workflow(input)
346
+ @next_interceptor.signal_external_workflow(input)
347
+ end
348
+
349
+ # Sleep.
350
+ #
351
+ # @param input [SleepInput] Input.
352
+ def sleep(input)
353
+ @next_interceptor.sleep(input)
354
+ end
355
+
356
+ # Start child workflow.
357
+ #
358
+ # @param input [StartChildWorkflowInput] Input.
359
+ # @return [Workflow::ChildWorkflowHandle] Child workflow handle.
360
+ def start_child_workflow(input)
361
+ @next_interceptor.start_child_workflow(input)
362
+ end
363
+ end
364
+ end
365
+ end
366
+ end
367
+ end
@@ -0,0 +1,61 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Temporalio
4
+ class Worker
5
+ # Base class for poller behaviors that control how polling scales.
6
+ class PollerBehavior
7
+ # @!visibility private
8
+ def _to_bridge_options
9
+ raise NotImplementedError, 'Subclasses must implement this method'
10
+ end
11
+
12
+ # A poller behavior that attempts to poll as long as a slot is available, up to the
13
+ # provided maximum. Cannot be less than two for workflow tasks, or one for other tasks.
14
+ class SimpleMaximum < PollerBehavior
15
+ # @return [Integer] Maximum number of concurrent poll requests.
16
+ attr_reader :maximum
17
+
18
+ # @param maximum [Integer] Maximum number of concurrent poll requests.
19
+ def initialize(maximum)
20
+ super()
21
+ @maximum = maximum
22
+ end
23
+
24
+ # @!visibility private
25
+ def _to_bridge_options
26
+ Internal::Bridge::Worker::PollerBehaviorSimpleMaximum.new(simple_maximum: @maximum)
27
+ end
28
+ end
29
+
30
+ # A poller behavior that automatically scales the number of pollers based on feedback
31
+ # from the server. A slot must be available before beginning polling.
32
+ class Autoscaling < PollerBehavior
33
+ # @return [Integer] Minimum number of poll calls (assuming slots are available).
34
+ attr_reader :minimum
35
+ # @return [Integer] Maximum number of poll calls that will ever be open at once.
36
+ attr_reader :maximum
37
+ # @return [Integer] Number of polls attempted initially before scaling kicks in.
38
+ attr_reader :initial
39
+
40
+ # @param minimum [Integer] Minimum number of poll calls (assuming slots are available).
41
+ # @param maximum [Integer] Maximum number of poll calls that will ever be open at once.
42
+ # @param initial [Integer] Number of polls attempted initially before scaling kicks in.
43
+ def initialize(minimum: 1, maximum: 100, initial: 5)
44
+ super()
45
+ @minimum = minimum
46
+ @maximum = maximum
47
+ @initial = initial
48
+ end
49
+
50
+ # @!visibility private
51
+ def _to_bridge_options
52
+ Internal::Bridge::Worker::PollerBehaviorAutoscaling.new(
53
+ minimum: @minimum,
54
+ maximum: @maximum,
55
+ initial: @initial
56
+ )
57
+ end
58
+ end
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,237 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Temporalio
4
+ class Worker
5
+ # Implementation of a thread pool. This implementation is a stripped down form of Concurrent Ruby's
6
+ # `CachedThreadPool`.
7
+ class ThreadPool
8
+ # Much of this logic taken from
9
+ # https://github.com/ruby-concurrency/concurrent-ruby/blob/044020f44b36930b863b930f3ee8fa1e9f750469/lib/concurrent-ruby/concurrent/executor/ruby_thread_pool_executor.rb,
10
+ # see MIT license at
11
+ # https://github.com/ruby-concurrency/concurrent-ruby/blob/044020f44b36930b863b930f3ee8fa1e9f750469/LICENSE.txt
12
+
13
+ # @return [ThreadPool] Default/shared thread pool instance with unlimited max threads.
14
+ def self.default
15
+ @default ||= new
16
+ end
17
+
18
+ # @!visibility private
19
+ def self._monotonic_time
20
+ Process.clock_gettime(Process::CLOCK_MONOTONIC)
21
+ end
22
+
23
+ # Create a new thread pool that creates threads as needed.
24
+ #
25
+ # @param max_threads [Integer, nil] Maximum number of thread workers to create, or nil for unlimited max.
26
+ # @param idle_timeout [Float] Number of seconds before a thread worker with no work should be stopped. Note,
27
+ # the check of whether a thread worker is idle is only done on each new {execute} call.
28
+ def initialize(max_threads: nil, idle_timeout: 20)
29
+ @max_threads = max_threads
30
+ @idle_timeout = idle_timeout
31
+
32
+ @mutex = Mutex.new
33
+ @pool = []
34
+ @ready = []
35
+ @queue = []
36
+ @scheduled_task_count = 0
37
+ @completed_task_count = 0
38
+ @largest_length = 0
39
+ @workers_counter = 0
40
+ @prune_interval = @idle_timeout / 2
41
+ @next_prune_time = ThreadPool._monotonic_time + @prune_interval
42
+ end
43
+
44
+ # Execute the given block in a thread. The block should be built to never raise and need no arguments.
45
+ #
46
+ # @yield Block to execute.
47
+ def execute(&block)
48
+ @mutex.synchronize do
49
+ locked_assign_worker(&block) || locked_enqueue(&block)
50
+ @scheduled_task_count += 1
51
+ locked_prune_pool if @next_prune_time < ThreadPool._monotonic_time
52
+ end
53
+ end
54
+
55
+ # @return [Integer] The largest number of threads that have been created in the pool since construction.
56
+ def largest_length
57
+ @mutex.synchronize { @largest_length }
58
+ end
59
+
60
+ # @return [Integer] The number of tasks that have been scheduled for execution on the pool since construction.
61
+ def scheduled_task_count
62
+ @mutex.synchronize { @scheduled_task_count }
63
+ end
64
+
65
+ # @return [Integer] The number of tasks that have been completed by the pool since construction.
66
+ def completed_task_count
67
+ @mutex.synchronize { @completed_task_count }
68
+ end
69
+
70
+ # @return [Integer] The number of threads that are actively executing tasks.
71
+ def active_count
72
+ @mutex.synchronize { @pool.length - @ready.length }
73
+ end
74
+
75
+ # @return [Integer] The number of threads currently in the pool.
76
+ def length
77
+ @mutex.synchronize { @pool.length }
78
+ end
79
+
80
+ # @return [Integer] The number of tasks in the queue awaiting execution.
81
+ def queue_length
82
+ @mutex.synchronize { @queue.length }
83
+ end
84
+
85
+ # Gracefully shutdown each thread when it is done with its current task. This should not be called until all
86
+ # workers using this executor are complete. This does not need to be called at all on program exit (e.g. for the
87
+ # global default).
88
+ def shutdown
89
+ @mutex.synchronize do
90
+ # Stop all workers
91
+ @pool.each(&:stop)
92
+ end
93
+ end
94
+
95
+ # Kill each thread. This should not be called until all workers using this executor are complete. This does not
96
+ # need to be called at all on program exit (e.g. for the global default).
97
+ def kill
98
+ @mutex.synchronize do
99
+ # Kill all workers
100
+ @pool.each(&:kill)
101
+ @pool.clear
102
+ @ready.clear
103
+ end
104
+ end
105
+
106
+ # @!visibility private
107
+ def _remove_busy_worker(worker)
108
+ @mutex.synchronize { locked_remove_busy_worker(worker) }
109
+ end
110
+
111
+ # @!visibility private
112
+ def _ready_worker(worker, last_message)
113
+ @mutex.synchronize { locked_ready_worker(worker, last_message) }
114
+ end
115
+
116
+ # @!visibility private
117
+ def _worker_died(worker)
118
+ @mutex.synchronize { locked_worker_died(worker) }
119
+ end
120
+
121
+ # @!visibility private
122
+ def _worker_task_completed
123
+ @mutex.synchronize { @completed_task_count += 1 }
124
+ end
125
+
126
+ private
127
+
128
+ def locked_assign_worker(&block) # rubocop:disable Naming/PredicateMethod
129
+ # keep growing if the pool is not at the minimum yet
130
+ worker, = @ready.pop || locked_add_busy_worker
131
+ if worker
132
+ worker << block
133
+ true
134
+ else
135
+ false
136
+ end
137
+ end
138
+
139
+ def locked_enqueue(&block)
140
+ @queue << block
141
+ end
142
+
143
+ def locked_add_busy_worker
144
+ return if @max_threads && @pool.size >= @max_threads
145
+
146
+ @workers_counter += 1
147
+ @pool << (worker = Worker.new(self, @workers_counter))
148
+ @largest_length = @pool.length if @pool.length > @largest_length
149
+ worker
150
+ end
151
+
152
+ def locked_prune_pool
153
+ now = ThreadPool._monotonic_time
154
+ stopped_workers = 0
155
+ while !@ready.empty? && (@pool.size - stopped_workers).positive?
156
+ worker, last_message = @ready.first
157
+ break unless now - last_message > @idle_timeout
158
+
159
+ stopped_workers += 1
160
+ @ready.shift
161
+ worker << :stop
162
+
163
+ end
164
+
165
+ @next_prune_time = ThreadPool._monotonic_time + @prune_interval
166
+ end
167
+
168
+ def locked_remove_busy_worker(worker)
169
+ @pool.delete(worker)
170
+ end
171
+
172
+ def locked_ready_worker(worker, last_message)
173
+ block = @queue.shift
174
+ if block
175
+ worker << block
176
+ else
177
+ @ready.push([worker, last_message])
178
+ end
179
+ end
180
+
181
+ def locked_worker_died(worker)
182
+ locked_remove_busy_worker(worker)
183
+ replacement_worker = locked_add_busy_worker
184
+ locked_ready_worker(replacement_worker, ThreadPool._monotonic_time) if replacement_worker
185
+ end
186
+
187
+ # @!visibility private
188
+ class Worker
189
+ def initialize(pool, id)
190
+ @queue = Queue.new
191
+ @thread = Thread.new(@queue, pool) do |my_queue, my_pool|
192
+ catch(:stop) do
193
+ loop do
194
+ case block = my_queue.pop
195
+ when :stop
196
+ pool._remove_busy_worker(self)
197
+ throw :stop
198
+ else
199
+ begin
200
+ block.call
201
+ my_pool._worker_task_completed
202
+ my_pool._ready_worker(self, ThreadPool._monotonic_time)
203
+ rescue StandardError => e
204
+ # Ignore
205
+ warn("Unexpected execute block error: #{e.full_message}")
206
+ rescue Exception => e # rubocop:disable Lint/RescueException
207
+ warn("Unexpected execute block exception: #{e.full_message}")
208
+ my_pool._worker_died(self)
209
+ throw :stop
210
+ end
211
+ end
212
+ end
213
+ end
214
+ end
215
+ @thread.name = "temporal-thread-#{id}"
216
+ end
217
+
218
+ # @!visibility private
219
+ def <<(block)
220
+ @queue << block
221
+ end
222
+
223
+ # @!visibility private
224
+ def stop
225
+ @queue << :stop
226
+ end
227
+
228
+ # @!visibility private
229
+ def kill
230
+ @thread.kill
231
+ end
232
+ end
233
+
234
+ private_constant :Worker
235
+ end
236
+ end
237
+ end