azure-functions-durable 1.2.9__py3-none-any.whl → 1.3.0__py3-none-any.whl
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.
- azure/durable_functions/__init__.py +81 -81
- azure/durable_functions/constants.py +9 -9
- azure/durable_functions/decorators/__init__.py +3 -3
- azure/durable_functions/decorators/durable_app.py +260 -249
- azure/durable_functions/decorators/metadata.py +109 -109
- azure/durable_functions/entity.py +129 -125
- azure/durable_functions/models/DurableEntityContext.py +201 -201
- azure/durable_functions/models/DurableHttpRequest.py +58 -58
- azure/durable_functions/models/DurableOrchestrationBindings.py +66 -66
- azure/durable_functions/models/DurableOrchestrationClient.py +812 -781
- azure/durable_functions/models/DurableOrchestrationContext.py +761 -707
- azure/durable_functions/models/DurableOrchestrationStatus.py +156 -156
- azure/durable_functions/models/EntityStateResponse.py +23 -23
- azure/durable_functions/models/FunctionContext.py +7 -7
- azure/durable_functions/models/OrchestrationRuntimeStatus.py +32 -32
- azure/durable_functions/models/OrchestratorState.py +117 -116
- azure/durable_functions/models/PurgeHistoryResult.py +33 -33
- azure/durable_functions/models/ReplaySchema.py +9 -8
- azure/durable_functions/models/RetryOptions.py +69 -69
- azure/durable_functions/models/RpcManagementOptions.py +86 -86
- azure/durable_functions/models/Task.py +540 -426
- azure/durable_functions/models/TaskOrchestrationExecutor.py +352 -336
- azure/durable_functions/models/TokenSource.py +56 -56
- azure/durable_functions/models/__init__.py +26 -24
- azure/durable_functions/models/actions/Action.py +23 -23
- azure/durable_functions/models/actions/ActionType.py +18 -18
- azure/durable_functions/models/actions/CallActivityAction.py +41 -41
- azure/durable_functions/models/actions/CallActivityWithRetryAction.py +45 -45
- azure/durable_functions/models/actions/CallEntityAction.py +46 -46
- azure/durable_functions/models/actions/CallHttpAction.py +35 -35
- azure/durable_functions/models/actions/CallSubOrchestratorAction.py +40 -40
- azure/durable_functions/models/actions/CallSubOrchestratorWithRetryAction.py +44 -44
- azure/durable_functions/models/actions/CompoundAction.py +35 -35
- azure/durable_functions/models/actions/ContinueAsNewAction.py +36 -36
- azure/durable_functions/models/actions/CreateTimerAction.py +48 -48
- azure/durable_functions/models/actions/NoOpAction.py +35 -35
- azure/durable_functions/models/actions/SignalEntityAction.py +47 -47
- azure/durable_functions/models/actions/WaitForExternalEventAction.py +63 -63
- azure/durable_functions/models/actions/WhenAllAction.py +14 -14
- azure/durable_functions/models/actions/WhenAnyAction.py +14 -14
- azure/durable_functions/models/actions/__init__.py +24 -24
- azure/durable_functions/models/entities/EntityState.py +74 -74
- azure/durable_functions/models/entities/OperationResult.py +94 -76
- azure/durable_functions/models/entities/RequestMessage.py +53 -53
- azure/durable_functions/models/entities/ResponseMessage.py +48 -48
- azure/durable_functions/models/entities/Signal.py +62 -62
- azure/durable_functions/models/entities/__init__.py +17 -17
- azure/durable_functions/models/history/HistoryEvent.py +92 -92
- azure/durable_functions/models/history/HistoryEventType.py +27 -27
- azure/durable_functions/models/history/__init__.py +8 -8
- azure/durable_functions/models/utils/__init__.py +7 -7
- azure/durable_functions/models/utils/entity_utils.py +103 -91
- azure/durable_functions/models/utils/http_utils.py +80 -69
- azure/durable_functions/models/utils/json_utils.py +96 -56
- azure/durable_functions/orchestrator.py +73 -71
- azure/durable_functions/testing/OrchestratorGeneratorWrapper.py +42 -0
- azure/durable_functions/testing/__init__.py +6 -0
- {azure_functions_durable-1.2.9.dist-info → azure_functions_durable-1.3.0.dist-info}/LICENSE +21 -21
- {azure_functions_durable-1.2.9.dist-info → azure_functions_durable-1.3.0.dist-info}/METADATA +59 -58
- azure_functions_durable-1.3.0.dist-info/RECORD +103 -0
- {azure_functions_durable-1.2.9.dist-info → azure_functions_durable-1.3.0.dist-info}/WHEEL +1 -1
- tests/models/test_DecoratorMetadata.py +135 -135
- tests/models/test_Decorators.py +107 -107
- tests/models/test_DurableOrchestrationBindings.py +68 -68
- tests/models/test_DurableOrchestrationClient.py +730 -730
- tests/models/test_DurableOrchestrationContext.py +102 -102
- tests/models/test_DurableOrchestrationStatus.py +59 -59
- tests/models/test_OrchestrationState.py +28 -28
- tests/models/test_RpcManagementOptions.py +79 -79
- tests/models/test_TokenSource.py +10 -10
- tests/orchestrator/models/OrchestrationInstance.py +18 -18
- tests/orchestrator/orchestrator_test_utils.py +130 -130
- tests/orchestrator/schemas/OrchetrationStateSchema.py +66 -66
- tests/orchestrator/test_call_http.py +235 -176
- tests/orchestrator/test_continue_as_new.py +67 -67
- tests/orchestrator/test_create_timer.py +126 -126
- tests/orchestrator/test_entity.py +397 -395
- tests/orchestrator/test_external_event.py +53 -53
- tests/orchestrator/test_fan_out_fan_in.py +175 -175
- tests/orchestrator/test_is_replaying_flag.py +101 -101
- tests/orchestrator/test_retries.py +308 -308
- tests/orchestrator/test_sequential_orchestrator.py +841 -841
- tests/orchestrator/test_sequential_orchestrator_custom_status.py +119 -119
- tests/orchestrator/test_sequential_orchestrator_with_retry.py +465 -465
- tests/orchestrator/test_serialization.py +30 -30
- tests/orchestrator/test_sub_orchestrator.py +95 -95
- tests/orchestrator/test_sub_orchestrator_with_retry.py +129 -129
- tests/orchestrator/test_task_any.py +60 -60
- tests/tasks/tasks_test_utils.py +17 -17
- tests/tasks/test_long_timers.py +70 -0
- tests/tasks/test_new_uuid.py +34 -34
- tests/test_utils/ContextBuilder.py +174 -174
- tests/test_utils/EntityContextBuilder.py +56 -56
- tests/test_utils/constants.py +1 -1
- tests/test_utils/json_utils.py +30 -30
- tests/test_utils/testClasses.py +56 -56
- tests/utils/__init__.py +1 -0
- tests/utils/test_entity_utils.py +24 -0
- azure_functions_durable-1.2.9.data/data/_manifest/bsi.json +0 -1
- azure_functions_durable-1.2.9.data/data/_manifest/manifest.cat +0 -0
- azure_functions_durable-1.2.9.data/data/_manifest/manifest.spdx.json +0 -11985
- azure_functions_durable-1.2.9.data/data/_manifest/manifest.spdx.json.sha256 +0 -1
- azure_functions_durable-1.2.9.dist-info/RECORD +0 -102
- {azure_functions_durable-1.2.9.dist-info → azure_functions_durable-1.3.0.dist-info}/top_level.txt +0 -0
|
@@ -1,101 +1,101 @@
|
|
|
1
|
-
from azure.durable_functions.models.ReplaySchema import ReplaySchema
|
|
2
|
-
from tests.test_utils.ContextBuilder import ContextBuilder
|
|
3
|
-
from .orchestrator_test_utils \
|
|
4
|
-
import get_orchestration_property, assert_orchestration_state_equals, assert_valid_schema
|
|
5
|
-
from azure.durable_functions.models.actions.CreateTimerAction import CreateTimerAction
|
|
6
|
-
from azure.durable_functions.models.OrchestratorState import OrchestratorState
|
|
7
|
-
from azure.durable_functions.constants import DATETIME_STRING_FORMAT
|
|
8
|
-
from datetime import datetime, timedelta, timezone
|
|
9
|
-
|
|
10
|
-
def generator_function(context):
|
|
11
|
-
# Create a timezone aware datetime object, just like a normal
|
|
12
|
-
# call to `context.current_utc_datetime` would create
|
|
13
|
-
timestamp = "2020-07-23T21:56:54.936700Z"
|
|
14
|
-
deadline = datetime.strptime(timestamp, DATETIME_STRING_FORMAT)
|
|
15
|
-
deadline = deadline.replace(tzinfo=timezone.utc)
|
|
16
|
-
|
|
17
|
-
for _ in range(0, 3):
|
|
18
|
-
deadline = deadline + timedelta(seconds=30)
|
|
19
|
-
yield context.create_timer(deadline)
|
|
20
|
-
|
|
21
|
-
def generator_function_compound_task(context):
|
|
22
|
-
# Create a timezone aware datetime object, just like a normal
|
|
23
|
-
# call to `context.current_utc_datetime` would create
|
|
24
|
-
timestamp = "2020-07-23T21:56:54.936700Z"
|
|
25
|
-
deadline = datetime.strptime(timestamp, DATETIME_STRING_FORMAT)
|
|
26
|
-
deadline = deadline.replace(tzinfo=timezone.utc)
|
|
27
|
-
|
|
28
|
-
tasks = []
|
|
29
|
-
for _ in range(0, 3):
|
|
30
|
-
deadline = deadline + timedelta(seconds=30)
|
|
31
|
-
tasks.append(context.create_timer(deadline))
|
|
32
|
-
yield context.task_any(tasks)
|
|
33
|
-
|
|
34
|
-
def base_expected_state(output=None, replay_schema: ReplaySchema = ReplaySchema.V1) -> OrchestratorState:
|
|
35
|
-
return OrchestratorState(is_done=False, actions=[], output=output, replay_schema=replay_schema.value)
|
|
36
|
-
|
|
37
|
-
def add_timer_fired_events(context_builder: ContextBuilder, id_: int, timestamp: str,
|
|
38
|
-
is_played: bool = True):
|
|
39
|
-
fire_at: str = context_builder.add_timer_created_event(id_, timestamp)
|
|
40
|
-
context_builder.add_orchestrator_completed_event()
|
|
41
|
-
context_builder.add_orchestrator_started_event()
|
|
42
|
-
context_builder.add_timer_fired_event(id_=id_, fire_at=fire_at, is_played=is_played)
|
|
43
|
-
|
|
44
|
-
def add_timer_action(state: OrchestratorState, fire_at: datetime):
|
|
45
|
-
action = CreateTimerAction(fire_at=fire_at)
|
|
46
|
-
state._actions.append([action])
|
|
47
|
-
|
|
48
|
-
def test_is_replaying_initial_value():
|
|
49
|
-
|
|
50
|
-
context_builder = ContextBuilder("", is_replaying=False)
|
|
51
|
-
result = get_orchestration_property(
|
|
52
|
-
context_builder, generator_function, "durable_context")
|
|
53
|
-
|
|
54
|
-
assert result.is_replaying == False
|
|
55
|
-
|
|
56
|
-
def test_is_replaying_one_replayed_event():
|
|
57
|
-
|
|
58
|
-
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
59
|
-
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
60
|
-
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
61
|
-
|
|
62
|
-
context_builder = ContextBuilder("")
|
|
63
|
-
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
64
|
-
|
|
65
|
-
result = get_orchestration_property(
|
|
66
|
-
context_builder, generator_function, "durable_context")
|
|
67
|
-
|
|
68
|
-
assert result.is_replaying == True
|
|
69
|
-
|
|
70
|
-
def test_is_replaying_one_replayed_one_not():
|
|
71
|
-
|
|
72
|
-
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
73
|
-
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
74
|
-
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
75
|
-
fire_at2 = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=60)
|
|
76
|
-
fire_at_str2 = fire_at2.strftime(DATETIME_STRING_FORMAT)
|
|
77
|
-
|
|
78
|
-
context_builder = ContextBuilder("")
|
|
79
|
-
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
80
|
-
add_timer_fired_events(context_builder, 1, fire_at_str2, is_played=False)
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
result = get_orchestration_property(
|
|
84
|
-
context_builder, generator_function, "durable_context")
|
|
85
|
-
|
|
86
|
-
assert result.is_replaying == False
|
|
87
|
-
|
|
88
|
-
def test_is_replaying_propagates_in_compound_task():
|
|
89
|
-
|
|
90
|
-
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
91
|
-
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
92
|
-
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
93
|
-
|
|
94
|
-
context_builder = ContextBuilder("")
|
|
95
|
-
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
96
|
-
|
|
97
|
-
result = get_orchestration_property(
|
|
98
|
-
context_builder, generator_function_compound_task, "durable_context")
|
|
99
|
-
|
|
100
|
-
assert result.is_replaying == True
|
|
101
|
-
|
|
1
|
+
from azure.durable_functions.models.ReplaySchema import ReplaySchema
|
|
2
|
+
from tests.test_utils.ContextBuilder import ContextBuilder
|
|
3
|
+
from .orchestrator_test_utils \
|
|
4
|
+
import get_orchestration_property, assert_orchestration_state_equals, assert_valid_schema
|
|
5
|
+
from azure.durable_functions.models.actions.CreateTimerAction import CreateTimerAction
|
|
6
|
+
from azure.durable_functions.models.OrchestratorState import OrchestratorState
|
|
7
|
+
from azure.durable_functions.constants import DATETIME_STRING_FORMAT
|
|
8
|
+
from datetime import datetime, timedelta, timezone
|
|
9
|
+
|
|
10
|
+
def generator_function(context):
|
|
11
|
+
# Create a timezone aware datetime object, just like a normal
|
|
12
|
+
# call to `context.current_utc_datetime` would create
|
|
13
|
+
timestamp = "2020-07-23T21:56:54.936700Z"
|
|
14
|
+
deadline = datetime.strptime(timestamp, DATETIME_STRING_FORMAT)
|
|
15
|
+
deadline = deadline.replace(tzinfo=timezone.utc)
|
|
16
|
+
|
|
17
|
+
for _ in range(0, 3):
|
|
18
|
+
deadline = deadline + timedelta(seconds=30)
|
|
19
|
+
yield context.create_timer(deadline)
|
|
20
|
+
|
|
21
|
+
def generator_function_compound_task(context):
|
|
22
|
+
# Create a timezone aware datetime object, just like a normal
|
|
23
|
+
# call to `context.current_utc_datetime` would create
|
|
24
|
+
timestamp = "2020-07-23T21:56:54.936700Z"
|
|
25
|
+
deadline = datetime.strptime(timestamp, DATETIME_STRING_FORMAT)
|
|
26
|
+
deadline = deadline.replace(tzinfo=timezone.utc)
|
|
27
|
+
|
|
28
|
+
tasks = []
|
|
29
|
+
for _ in range(0, 3):
|
|
30
|
+
deadline = deadline + timedelta(seconds=30)
|
|
31
|
+
tasks.append(context.create_timer(deadline))
|
|
32
|
+
yield context.task_any(tasks)
|
|
33
|
+
|
|
34
|
+
def base_expected_state(output=None, replay_schema: ReplaySchema = ReplaySchema.V1) -> OrchestratorState:
|
|
35
|
+
return OrchestratorState(is_done=False, actions=[], output=output, replay_schema=replay_schema.value)
|
|
36
|
+
|
|
37
|
+
def add_timer_fired_events(context_builder: ContextBuilder, id_: int, timestamp: str,
|
|
38
|
+
is_played: bool = True):
|
|
39
|
+
fire_at: str = context_builder.add_timer_created_event(id_, timestamp)
|
|
40
|
+
context_builder.add_orchestrator_completed_event()
|
|
41
|
+
context_builder.add_orchestrator_started_event()
|
|
42
|
+
context_builder.add_timer_fired_event(id_=id_, fire_at=fire_at, is_played=is_played)
|
|
43
|
+
|
|
44
|
+
def add_timer_action(state: OrchestratorState, fire_at: datetime):
|
|
45
|
+
action = CreateTimerAction(fire_at=fire_at)
|
|
46
|
+
state._actions.append([action])
|
|
47
|
+
|
|
48
|
+
def test_is_replaying_initial_value():
|
|
49
|
+
|
|
50
|
+
context_builder = ContextBuilder("", is_replaying=False)
|
|
51
|
+
result = get_orchestration_property(
|
|
52
|
+
context_builder, generator_function, "durable_context")
|
|
53
|
+
|
|
54
|
+
assert result.is_replaying == False
|
|
55
|
+
|
|
56
|
+
def test_is_replaying_one_replayed_event():
|
|
57
|
+
|
|
58
|
+
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
59
|
+
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
60
|
+
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
61
|
+
|
|
62
|
+
context_builder = ContextBuilder("")
|
|
63
|
+
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
64
|
+
|
|
65
|
+
result = get_orchestration_property(
|
|
66
|
+
context_builder, generator_function, "durable_context")
|
|
67
|
+
|
|
68
|
+
assert result.is_replaying == True
|
|
69
|
+
|
|
70
|
+
def test_is_replaying_one_replayed_one_not():
|
|
71
|
+
|
|
72
|
+
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
73
|
+
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
74
|
+
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
75
|
+
fire_at2 = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=60)
|
|
76
|
+
fire_at_str2 = fire_at2.strftime(DATETIME_STRING_FORMAT)
|
|
77
|
+
|
|
78
|
+
context_builder = ContextBuilder("")
|
|
79
|
+
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
80
|
+
add_timer_fired_events(context_builder, 1, fire_at_str2, is_played=False)
|
|
81
|
+
|
|
82
|
+
|
|
83
|
+
result = get_orchestration_property(
|
|
84
|
+
context_builder, generator_function, "durable_context")
|
|
85
|
+
|
|
86
|
+
assert result.is_replaying == False
|
|
87
|
+
|
|
88
|
+
def test_is_replaying_propagates_in_compound_task():
|
|
89
|
+
|
|
90
|
+
timestamp = "2020-07-23T21:56:54.9367Z"
|
|
91
|
+
fire_at = datetime.strptime(timestamp, DATETIME_STRING_FORMAT) + timedelta(seconds=30)
|
|
92
|
+
fire_at_str = fire_at.strftime(DATETIME_STRING_FORMAT)
|
|
93
|
+
|
|
94
|
+
context_builder = ContextBuilder("")
|
|
95
|
+
add_timer_fired_events(context_builder, 0, fire_at_str, is_played=True)
|
|
96
|
+
|
|
97
|
+
result = get_orchestration_property(
|
|
98
|
+
context_builder, generator_function_compound_task, "durable_context")
|
|
99
|
+
|
|
100
|
+
assert result.is_replaying == True
|
|
101
|
+
|