modal 0.74.18__py3-none-any.whl → 0.74.19__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.
- modal/_runtime/container_io_manager.py +26 -10
- modal/client.pyi +2 -2
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/METADATA +1 -1
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/RECORD +9 -9
- modal_version/_version_generated.py +1 -1
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/WHEEL +0 -0
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/entry_points.txt +0 -0
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/licenses/LICENSE +0 -0
- {modal-0.74.18.dist-info → modal-0.74.19.dist-info}/top_level.txt +0 -0
@@ -352,15 +352,19 @@ class _ContainerIOManager:
|
|
352
352
|
await self._client.stub.ContainerHello(Empty())
|
353
353
|
|
354
354
|
async def _run_heartbeat_loop(self):
|
355
|
+
t_last_success = time.monotonic()
|
355
356
|
while 1:
|
356
357
|
t0 = time.monotonic()
|
357
358
|
try:
|
358
|
-
|
359
|
-
|
360
|
-
|
361
|
-
#
|
362
|
-
#
|
363
|
-
#
|
359
|
+
got_cancellation = await self._heartbeat_handle_cancellations()
|
360
|
+
t_last_success = time.monotonic()
|
361
|
+
if got_cancellation:
|
362
|
+
# Got a cancellation event, so it is fine to start another heartbeat immediately
|
363
|
+
# since the cancellation queue should be empty on the worker server.
|
364
|
+
# However, we wait at least 1s to prevent short-circuiting the heartbeat loop
|
365
|
+
# in case there is ever a bug.
|
366
|
+
# This means it will take at least 1s between two subsequent cancellations on the
|
367
|
+
# same task at the moment.
|
364
368
|
await asyncio.sleep(1.0)
|
365
369
|
continue
|
366
370
|
except ClientClosed:
|
@@ -368,13 +372,25 @@ class _ContainerIOManager:
|
|
368
372
|
break
|
369
373
|
except Exception as exc:
|
370
374
|
# don't stop heartbeat loop if there are transient exceptions!
|
371
|
-
|
375
|
+
attempt_dur = time.monotonic() - t0
|
376
|
+
time_since_heartbeat_success = time.monotonic() - t_last_success
|
372
377
|
error = exc
|
373
|
-
logger.warning(
|
378
|
+
logger.warning(
|
379
|
+
f"Modal Client → Modal Worker Heartbeat attempt failed "
|
380
|
+
f"({attempt_dur=:.2f}, {time_since_heartbeat_success=:.2f}, {error=})"
|
381
|
+
)
|
382
|
+
if time_since_heartbeat_success > HEARTBEAT_INTERVAL * 50:
|
383
|
+
trouble_mins = time_since_heartbeat_success / 60
|
384
|
+
logger.warning(
|
385
|
+
"Modal Client → Modal Worker heartbeat attempts have been failing for "
|
386
|
+
f"over {trouble_mins:.2f} minutes. "
|
387
|
+
"Container will eventually be marked unhealthy. "
|
388
|
+
"See https://modal.com/docs/guide/troubleshooting#heartbeat-timeout. "
|
389
|
+
)
|
374
390
|
|
375
391
|
heartbeat_duration = time.monotonic() - t0
|
376
|
-
|
377
|
-
await asyncio.sleep(
|
392
|
+
time_until_next_heartbeat = max(0.0, HEARTBEAT_INTERVAL - heartbeat_duration)
|
393
|
+
await asyncio.sleep(time_until_next_heartbeat)
|
378
394
|
|
379
395
|
async def _heartbeat_handle_cancellations(self) -> bool:
|
380
396
|
# Return True if a cancellation event was received, in that case
|
modal/client.pyi
CHANGED
@@ -27,7 +27,7 @@ class _Client:
|
|
27
27
|
_snapshotted: bool
|
28
28
|
|
29
29
|
def __init__(
|
30
|
-
self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.74.
|
30
|
+
self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.74.19"
|
31
31
|
): ...
|
32
32
|
def is_closed(self) -> bool: ...
|
33
33
|
@property
|
@@ -85,7 +85,7 @@ class Client:
|
|
85
85
|
_snapshotted: bool
|
86
86
|
|
87
87
|
def __init__(
|
88
|
-
self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.74.
|
88
|
+
self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.74.19"
|
89
89
|
): ...
|
90
90
|
def is_closed(self) -> bool: ...
|
91
91
|
@property
|
@@ -22,7 +22,7 @@ modal/app.py,sha256=TnXw6EaejqwyJNawbtp177mG87ufJZClPoiIxO41Ioc,50664
|
|
22
22
|
modal/app.pyi,sha256=8oj9DpTHySHymjy6aSat6IG-z2FF77mx_ls1dGbt3Qg,28222
|
23
23
|
modal/call_graph.py,sha256=1g2DGcMIJvRy-xKicuf63IVE98gJSnQsr8R_NVMptNc,2581
|
24
24
|
modal/client.py,sha256=U-YKSw0n7J1ZLREt9cbEJCtmHe5YoPKFxl0xlkan2yc,15565
|
25
|
-
modal/client.pyi,sha256=
|
25
|
+
modal/client.pyi,sha256=wSvN4PKoC4BazCyPhTWaAjvyruNM7vlVoKhuICNBBIE,7593
|
26
26
|
modal/cloud_bucket_mount.py,sha256=YOe9nnvSr4ZbeCn587d7_VhE9IioZYRvF9VYQTQux08,5914
|
27
27
|
modal/cloud_bucket_mount.pyi,sha256=30T3K1a89l6wzmEJ_J9iWv9SknoGqaZDx59Xs-ZQcmk,1607
|
28
28
|
modal/cls.py,sha256=GvaNl8R5UsH7Vg88WEOyerdjvZEPK7xxi3nqHlyOW_c,33497
|
@@ -82,7 +82,7 @@ modal/volume.py,sha256=3c5_aJNJtgpsFRZWBjc0jwn8Zs0jo9V6UDmh6ifrbdA,40145
|
|
82
82
|
modal/volume.pyi,sha256=juOVWGlgz7IeOY4M7jBhbeNRPA9xdGUwvA3AzlZUscQ,17958
|
83
83
|
modal/_runtime/__init__.py,sha256=MIEP8jhXUeGq_eCjYFcqN5b1bxBM4fdk0VESpjWR0fc,28
|
84
84
|
modal/_runtime/asgi.py,sha256=KNarxvZI9z8fnmZl2vbkWTjnoLXs9kqOahkrbsTLkyc,22429
|
85
|
-
modal/_runtime/container_io_manager.py,sha256=
|
85
|
+
modal/_runtime/container_io_manager.py,sha256=6j0jO2-s9ShckM4SK45OapoQxWW9HQwQjFaBkXPJPwU,44763
|
86
86
|
modal/_runtime/container_io_manager.pyi,sha256=3N9TOYa5hfufhJV5IiIhpvJYCeLZe-ne76-XuxcFLW0,16147
|
87
87
|
modal/_runtime/execution_context.py,sha256=73Y5zH_o-MhVCrkJXakYVlFkKqCa2CWvqoHjOfJrJGg,3034
|
88
88
|
modal/_runtime/execution_context.pyi,sha256=TAxQq7uLj7i9r9XbXgFZiSVBWxObFWN-rkssS0I7Vkk,661
|
@@ -145,7 +145,7 @@ modal/requirements/2024.10.txt,sha256=qD-5cVIVM9wXesJ6JC89Ew-3m2KjEElUz3jaw_MddR
|
|
145
145
|
modal/requirements/PREVIEW.txt,sha256=qD-5cVIVM9wXesJ6JC89Ew-3m2KjEElUz3jaw_MddRo,296
|
146
146
|
modal/requirements/README.md,sha256=9tK76KP0Uph7O0M5oUgsSwEZDj5y-dcUPsnpR0Sc-Ik,854
|
147
147
|
modal/requirements/base-images.json,sha256=57vMSqzMbLBxw5tFWSaMiIkkVEps4JfX5PAtXGnkS4U,740
|
148
|
-
modal-0.74.
|
148
|
+
modal-0.74.19.dist-info/licenses/LICENSE,sha256=psuoW8kuDP96RQsdhzwOqi6fyWv0ct8CR6Jr7He_P_k,10173
|
149
149
|
modal_docs/__init__.py,sha256=svYKtV8HDwDCN86zbdWqyq5T8sMdGDj0PVlzc2tIxDM,28
|
150
150
|
modal_docs/gen_cli_docs.py,sha256=c1yfBS_x--gL5bs0N4ihMwqwX8l3IBWSkBAKNNIi6bQ,3801
|
151
151
|
modal_docs/gen_reference_docs.py,sha256=d_CQUGQ0rfw28u75I2mov9AlS773z9rG40-yq5o7g2U,6359
|
@@ -170,9 +170,9 @@ modal_proto/options_pb2_grpc.pyi,sha256=CImmhxHsYnF09iENPoe8S4J-n93jtgUYD2JPAc0y
|
|
170
170
|
modal_proto/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
171
171
|
modal_version/__init__.py,sha256=m94xZNWIjH8oUtJk4l9xfovzDJede2o7X-q0MHVECtM,470
|
172
172
|
modal_version/__main__.py,sha256=2FO0yYQQwDTh6udt1h-cBnGd1c4ZyHnHSI4BksxzVac,105
|
173
|
-
modal_version/_version_generated.py,sha256=
|
174
|
-
modal-0.74.
|
175
|
-
modal-0.74.
|
176
|
-
modal-0.74.
|
177
|
-
modal-0.74.
|
178
|
-
modal-0.74.
|
173
|
+
modal_version/_version_generated.py,sha256=gnhxZOCXj_bRya3jnYgGg-_unrUkIoDxJUMPO5LnVz0,149
|
174
|
+
modal-0.74.19.dist-info/METADATA,sha256=xfMm5VTZynErYWFfwZ0xu5TyBDzl6PxByuJll7Raras,2474
|
175
|
+
modal-0.74.19.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
|
176
|
+
modal-0.74.19.dist-info/entry_points.txt,sha256=An-wYgeEUnm6xzrAP9_NTSTSciYvvEWsMZILtYrvpAI,46
|
177
|
+
modal-0.74.19.dist-info/top_level.txt,sha256=4BWzoKYREKUZ5iyPzZpjqx4G8uB5TWxXPDwibLcVa7k,43
|
178
|
+
modal-0.74.19.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|