forked from temporalio/samples-python
-
Notifications
You must be signed in to change notification settings - Fork 0
/
tasks.py
150 lines (118 loc) · 4.94 KB
/
tasks.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
import asyncio
from dataclasses import dataclass
from datetime import timedelta
from hashlib import sha256
from pathlib import Path
from temporalio import activity, workflow
from temporalio.common import RetryPolicy
def _get_delay_secs() -> float:
return 3
def _get_local_path() -> Path:
return Path(__file__).parent / "demo_fs"
def write_file(path: Path, body: str) -> None:
"""Convenience write wrapper for mocking FS"""
with open(path, "w") as handle:
handle.write(body)
def read_file(path) -> bytes:
"""Convenience read wrapper for mocking FS"""
with open(path, "rb") as handle:
return handle.read()
def delete_file(path) -> None:
"""Convenience delete wrapper for mocking FS"""
Path(path).unlink()
def create_filepath(unique_worker_id: str, workflow_uuid: str) -> Path:
"""Creates required folders and builds filepath"""
directory = _get_local_path() / unique_worker_id
directory.mkdir(parents=True, exist_ok=True)
filepath = directory / workflow_uuid
return filepath
def process_file_contents(file_content: bytes) -> str:
"""Returns hash of file string"""
return sha256(file_content).hexdigest()
@dataclass
class DownloadObj:
url: str
unique_worker_id: str
workflow_uuid: str
@activity.defn
async def get_available_task_queue() -> str:
"""Just a stub for typedworkflow invocation."""
raise NotImplementedError
@activity.defn
async def download_file_to_worker_filesystem(details: DownloadObj) -> str:
"""Simulates downloading a file to a local filesystem"""
# FS ops
print(details.unique_worker_id, details.workflow_uuid)
path = create_filepath(details.unique_worker_id, details.workflow_uuid)
activity.logger.info(f"Downloading ${details.url} and saving to ${path}")
# Here is where the real download code goes. Developers should be careful
# not to block an async activity. If there are concerns about blocking download
# or disk IO, developers should use loop.run_in_executor or change this activity
# to be synchronous. Also like for all non-immediate activities, be sure to
# heartbeat during download.
await asyncio.sleep(_get_delay_secs())
body = "downloaded body"
write_file(path, body)
return str(path)
@activity.defn
async def work_on_file_in_worker_filesystem(path: str) -> str:
"""Processing the file, in this case identical MD5 hashes"""
content = read_file(path)
checksum = process_file_contents(content)
await asyncio.sleep(_get_delay_secs())
activity.logger.info(f"Did some work on {path}, checksum {checksum}")
return checksum
@activity.defn
async def clean_up_file_from_worker_filesystem(path: str) -> None:
"""Deletes the file created in the first activity, but leaves the folder"""
await asyncio.sleep(_get_delay_secs())
activity.logger.info(f"Removing {path}")
delete_file(path)
@workflow.defn
class FileProcessing:
@workflow.run
async def run(self) -> str:
"""Workflow implementing the basic file processing example.
First, a task queue is selected randomly. A single worker is listening on
this queue, so when we execute all the file processing activities on this
queue, they will all be run on the same worker, and all be able to access
the same file on disk. The activities download the file, do some processing
task on the file, and clean up the file.
"""
workflow.logger.info("Searching for available worker")
unique_worker_task_queue = await workflow.execute_activity(
activity=get_available_task_queue,
start_to_close_timeout=timedelta(seconds=10),
)
workflow.logger.info(f"Matching workflow to worker {unique_worker_task_queue}")
download_params = DownloadObj(
url="http://temporal.io",
unique_worker_id=unique_worker_task_queue,
workflow_uuid=str(workflow.uuid4()),
)
download_path = await workflow.execute_activity(
download_file_to_worker_filesystem,
download_params,
start_to_close_timeout=timedelta(seconds=10),
task_queue=unique_worker_task_queue,
)
checksum = "failed execution" # Sentinel value
try:
checksum = await workflow.execute_activity(
work_on_file_in_worker_filesystem,
download_path,
start_to_close_timeout=timedelta(seconds=10),
retry_policy=RetryPolicy(
maximum_attempts=1,
# maximum_interval=timedelta(milliseconds=500),
),
task_queue=unique_worker_task_queue,
)
finally:
await workflow.execute_activity(
clean_up_file_from_worker_filesystem,
download_path,
start_to_close_timeout=timedelta(seconds=10),
task_queue=unique_worker_task_queue,
)
return checksum