Skip to content

Commit 61a8492

Browse files
authoredFeb 12, 2025
Merge pull request #10 from elena-kolevska/fix/terminate-recursive-test
Extend the recursive termination test to match the test in durabletask-go
2 parents a47e15b + 16ca420 commit 61a8492

File tree

2 files changed

+48
-30
lines changed

2 files changed

+48
-30
lines changed
 

‎README.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -178,10 +178,11 @@ make test-unit
178178

179179
### Running E2E tests
180180

181-
The E2E (end-to-end) tests require a sidecar process to be running. You can use the Dapr sidecar for this or run a Durable Task test sidecar using the following `docker` command:
181+
The E2E (end-to-end) tests require a sidecar process to be running. You can use the Dapr sidecar for this or run a Durable Task test sidecar using the following command:
182182

183183
```sh
184-
docker run --name durabletask-sidecar -p 4001:4001 --env 'DURABLETASK_SIDECAR_LOGLEVEL=Debug' --rm cgillum/durabletask-sidecar:latest start --backend Emulator
184+
go install github.com/dapr/durabletask-go@main
185+
durabletask-go --port 4001
185186
```
186187

187188
To run the E2E tests, run the following command from the project root:

‎tests/test_orchestration_e2e.py

+45-28
Original file line numberDiff line numberDiff line change
@@ -279,39 +279,56 @@ def orchestrator(ctx: task.OrchestrationContext, _):
279279
assert state.serialized_output == json.dumps("some reason for termination")
280280

281281
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
288285

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")
294291

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)
300296

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)
306302

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"
311331

312-
task_hub_client.purge_orchestration(id)
313-
state = task_hub_client.get_orchestration_state(id)
314-
assert state is None
315332

316333

317334
def test_continue_as_new():

0 commit comments

Comments
 (0)