forked from temporalio/samples-python
-
Notifications
You must be signed in to change notification settings - Fork 0
/
activity_utils.py
39 lines (32 loc) · 1.29 KB
/
activity_utils.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
import asyncio
from datetime import datetime
from functools import wraps
from typing import Any, Awaitable, Callable, TypeVar, cast
from temporalio import activity
F = TypeVar("F", bound=Callable[..., Awaitable[Any]])
def auto_heartbeater(fn: F) -> F:
# We want to ensure that the type hints from the original callable are
# available via our wrapper, so we use the functools wraps decorator
@wraps(fn)
async def wrapper(*args, **kwargs):
heartbeat_timeout = activity.info().heartbeat_timeout
heartbeat_task = None
if heartbeat_timeout:
# Heartbeat twice as often as the timeout
heartbeat_task = asyncio.create_task(
heartbeat_every(heartbeat_timeout.total_seconds() / 2)
)
try:
return await fn(*args, **kwargs)
finally:
if heartbeat_task:
heartbeat_task.cancel()
# Wait for heartbeat cancellation to complete
await asyncio.wait([heartbeat_task])
return cast(F, wrapper)
async def heartbeat_every(delay: float, *details: Any) -> None:
# Heartbeat every so often while not cancelled
while True:
await asyncio.sleep(delay)
print(f"Heartbeating at {datetime.now()}")
activity.heartbeat(*details)