@@ -279,39 +279,56 @@ def orchestrator(ctx: task.OrchestrationContext, _):
279
279
assert state .serialized_output == json .dumps ("some reason for termination" )
280
280
281
281
def test_terminate_recursive ():
282
- def root (ctx : task .OrchestrationContext , _ ):
283
- result = yield ctx .call_sub_orchestrator (child )
284
- return result
285
- def child (ctx : task .OrchestrationContext , _ ):
286
- result = yield ctx .wait_for_external_event ("my_event" )
287
- return result
282
+ thread_lock = threading .Lock ()
283
+ activity_counter = 0
284
+ delay_time = 4 # seconds
288
285
289
- # Start a worker, which will connect to the sidecar in a background thread
290
- with worker . TaskHubGrpcWorker () as w :
291
- w . add_orchestrator ( root )
292
- w . add_orchestrator ( child )
293
- w . start ( )
286
+ def increment ( ctx , _ ):
287
+ with thread_lock :
288
+ nonlocal activity_counter
289
+ activity_counter += 1
290
+ raise Exception ( "Failed: Should not have executed the activity" )
294
291
295
- task_hub_client = client .TaskHubGrpcClient ()
296
- id = task_hub_client .schedule_new_orchestration (root )
297
- state = task_hub_client .wait_for_orchestration_start (id , timeout = 30 )
298
- assert state is not None
299
- assert state .runtime_status == client .OrchestrationStatus .RUNNING
292
+ def orchestrator_child (ctx : task .OrchestrationContext , activity_count : int ):
293
+ due_time = ctx .current_utc_datetime + timedelta (seconds = delay_time )
294
+ yield ctx .create_timer (due_time )
295
+ yield ctx .call_activity (increment )
300
296
301
- # Terminate root orchestration(recursive set to True by default)
302
- task_hub_client . terminate_orchestration ( id , output = "some reason for termination" )
303
- state = task_hub_client . wait_for_orchestration_completion ( id , timeout = 30 )
304
- assert state is not None
305
- assert state . runtime_status == client . OrchestrationStatus . TERMINATED
297
+ def parent_orchestrator ( ctx : task . OrchestrationContext , count : int ):
298
+ tasks = []
299
+ for _ in range ( count ):
300
+ tasks . append ( ctx . call_sub_orchestrator ( orchestrator_child , input = count ))
301
+ yield task . when_all ( tasks )
306
302
307
- # Verify that child orchestration is also terminated
308
- c = task_hub_client .wait_for_orchestration_completion (id , timeout = 30 )
309
- assert state is not None
310
- assert state .runtime_status == client .OrchestrationStatus .TERMINATED
303
+ for recurse in [True , False ]:
304
+ with worker .TaskHubGrpcWorker () as w :
305
+ w .add_activity (increment )
306
+ w .add_orchestrator (orchestrator_child )
307
+ w .add_orchestrator (parent_orchestrator )
308
+ w .start ()
309
+
310
+ task_hub_client = client .TaskHubGrpcClient ()
311
+ instance_id = task_hub_client .schedule_new_orchestration (parent_orchestrator , input = 5 )
312
+
313
+ time .sleep (2 )
314
+
315
+ output = "Recursive termination = {recurse}"
316
+ task_hub_client .terminate_orchestration (instance_id , output = output , recursive = recurse )
317
+
318
+
319
+ metadata = task_hub_client .wait_for_orchestration_completion (instance_id , timeout = 30 )
320
+
321
+ assert metadata is not None
322
+ assert metadata .runtime_status == client .OrchestrationStatus .TERMINATED
323
+ assert metadata .serialized_output == f'"{ output } "'
324
+
325
+ time .sleep (delay_time )
326
+
327
+ if recurse :
328
+ assert activity_counter == 0 , "Activity should not have executed with recursive termination"
329
+ else :
330
+ assert activity_counter == 5 , "Activity should have executed without recursive termination"
311
331
312
- task_hub_client .purge_orchestration (id )
313
- state = task_hub_client .get_orchestration_state (id )
314
- assert state is None
315
332
316
333
317
334
def test_continue_as_new ():
0 commit comments