modal 0.73.95__tar.gz → 0.73.97__tar.gz

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.
Files changed (183) hide show
  1. {modal-0.73.95 → modal-0.73.97}/PKG-INFO +1 -1
  2. {modal-0.73.95 → modal-0.73.97}/modal/_utils/grpc_utils.py +10 -0
  3. {modal-0.73.95 → modal-0.73.97}/modal/client.pyi +2 -2
  4. {modal-0.73.95 → modal-0.73.97}/modal/image.py +5 -2
  5. {modal-0.73.95 → modal-0.73.97}/modal/image.pyi +17 -13
  6. {modal-0.73.95 → modal-0.73.97}/modal/parallel_map.py +19 -21
  7. modal-0.73.97/modal/requirements/PREVIEW.txt +16 -0
  8. {modal-0.73.95 → modal-0.73.97}/modal/requirements/base-images.json +5 -1
  9. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/PKG-INFO +1 -1
  10. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/SOURCES.txt +1 -0
  11. {modal-0.73.95 → modal-0.73.97}/modal_version/_version_generated.py +1 -1
  12. {modal-0.73.95 → modal-0.73.97}/LICENSE +0 -0
  13. {modal-0.73.95 → modal-0.73.97}/README.md +0 -0
  14. {modal-0.73.95 → modal-0.73.97}/modal/__init__.py +0 -0
  15. {modal-0.73.95 → modal-0.73.97}/modal/__main__.py +0 -0
  16. {modal-0.73.95 → modal-0.73.97}/modal/_clustered_functions.py +0 -0
  17. {modal-0.73.95 → modal-0.73.97}/modal/_clustered_functions.pyi +0 -0
  18. {modal-0.73.95 → modal-0.73.97}/modal/_container_entrypoint.py +0 -0
  19. {modal-0.73.95 → modal-0.73.97}/modal/_functions.py +0 -0
  20. {modal-0.73.95 → modal-0.73.97}/modal/_ipython.py +0 -0
  21. {modal-0.73.95 → modal-0.73.97}/modal/_location.py +0 -0
  22. {modal-0.73.95 → modal-0.73.97}/modal/_object.py +0 -0
  23. {modal-0.73.95 → modal-0.73.97}/modal/_output.py +0 -0
  24. {modal-0.73.95 → modal-0.73.97}/modal/_partial_function.py +0 -0
  25. {modal-0.73.95 → modal-0.73.97}/modal/_proxy_tunnel.py +0 -0
  26. {modal-0.73.95 → modal-0.73.97}/modal/_pty.py +0 -0
  27. {modal-0.73.95 → modal-0.73.97}/modal/_resolver.py +0 -0
  28. {modal-0.73.95 → modal-0.73.97}/modal/_resources.py +0 -0
  29. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/__init__.py +0 -0
  30. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/asgi.py +0 -0
  31. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/container_io_manager.py +0 -0
  32. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/container_io_manager.pyi +0 -0
  33. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/execution_context.py +0 -0
  34. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/execution_context.pyi +0 -0
  35. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/gpu_memory_snapshot.py +0 -0
  36. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/telemetry.py +0 -0
  37. {modal-0.73.95 → modal-0.73.97}/modal/_runtime/user_code_imports.py +0 -0
  38. {modal-0.73.95 → modal-0.73.97}/modal/_serialization.py +0 -0
  39. {modal-0.73.95 → modal-0.73.97}/modal/_traceback.py +0 -0
  40. {modal-0.73.95 → modal-0.73.97}/modal/_tunnel.py +0 -0
  41. {modal-0.73.95 → modal-0.73.97}/modal/_tunnel.pyi +0 -0
  42. {modal-0.73.95 → modal-0.73.97}/modal/_utils/__init__.py +0 -0
  43. {modal-0.73.95 → modal-0.73.97}/modal/_utils/app_utils.py +0 -0
  44. {modal-0.73.95 → modal-0.73.97}/modal/_utils/async_utils.py +0 -0
  45. {modal-0.73.95 → modal-0.73.97}/modal/_utils/blob_utils.py +0 -0
  46. {modal-0.73.95 → modal-0.73.97}/modal/_utils/bytes_io_segment_payload.py +0 -0
  47. {modal-0.73.95 → modal-0.73.97}/modal/_utils/deprecation.py +0 -0
  48. {modal-0.73.95 → modal-0.73.97}/modal/_utils/docker_utils.py +0 -0
  49. {modal-0.73.95 → modal-0.73.97}/modal/_utils/function_utils.py +0 -0
  50. {modal-0.73.95 → modal-0.73.97}/modal/_utils/grpc_testing.py +0 -0
  51. {modal-0.73.95 → modal-0.73.97}/modal/_utils/hash_utils.py +0 -0
  52. {modal-0.73.95 → modal-0.73.97}/modal/_utils/http_utils.py +0 -0
  53. {modal-0.73.95 → modal-0.73.97}/modal/_utils/logger.py +0 -0
  54. {modal-0.73.95 → modal-0.73.97}/modal/_utils/mount_utils.py +0 -0
  55. {modal-0.73.95 → modal-0.73.97}/modal/_utils/name_utils.py +0 -0
  56. {modal-0.73.95 → modal-0.73.97}/modal/_utils/package_utils.py +0 -0
  57. {modal-0.73.95 → modal-0.73.97}/modal/_utils/pattern_utils.py +0 -0
  58. {modal-0.73.95 → modal-0.73.97}/modal/_utils/rand_pb_testing.py +0 -0
  59. {modal-0.73.95 → modal-0.73.97}/modal/_utils/shell_utils.py +0 -0
  60. {modal-0.73.95 → modal-0.73.97}/modal/_vendor/__init__.py +0 -0
  61. {modal-0.73.95 → modal-0.73.97}/modal/_vendor/a2wsgi_wsgi.py +0 -0
  62. {modal-0.73.95 → modal-0.73.97}/modal/_vendor/cloudpickle.py +0 -0
  63. {modal-0.73.95 → modal-0.73.97}/modal/_vendor/tblib.py +0 -0
  64. {modal-0.73.95 → modal-0.73.97}/modal/_watcher.py +0 -0
  65. {modal-0.73.95 → modal-0.73.97}/modal/app.py +0 -0
  66. {modal-0.73.95 → modal-0.73.97}/modal/app.pyi +0 -0
  67. {modal-0.73.95 → modal-0.73.97}/modal/call_graph.py +0 -0
  68. {modal-0.73.95 → modal-0.73.97}/modal/cli/__init__.py +0 -0
  69. {modal-0.73.95 → modal-0.73.97}/modal/cli/_download.py +0 -0
  70. {modal-0.73.95 → modal-0.73.97}/modal/cli/_traceback.py +0 -0
  71. {modal-0.73.95 → modal-0.73.97}/modal/cli/app.py +0 -0
  72. {modal-0.73.95 → modal-0.73.97}/modal/cli/config.py +0 -0
  73. {modal-0.73.95 → modal-0.73.97}/modal/cli/container.py +0 -0
  74. {modal-0.73.95 → modal-0.73.97}/modal/cli/dict.py +0 -0
  75. {modal-0.73.95 → modal-0.73.97}/modal/cli/entry_point.py +0 -0
  76. {modal-0.73.95 → modal-0.73.97}/modal/cli/environment.py +0 -0
  77. {modal-0.73.95 → modal-0.73.97}/modal/cli/import_refs.py +0 -0
  78. {modal-0.73.95 → modal-0.73.97}/modal/cli/launch.py +0 -0
  79. {modal-0.73.95 → modal-0.73.97}/modal/cli/network_file_system.py +0 -0
  80. {modal-0.73.95 → modal-0.73.97}/modal/cli/profile.py +0 -0
  81. {modal-0.73.95 → modal-0.73.97}/modal/cli/programs/__init__.py +0 -0
  82. {modal-0.73.95 → modal-0.73.97}/modal/cli/programs/run_jupyter.py +0 -0
  83. {modal-0.73.95 → modal-0.73.97}/modal/cli/programs/vscode.py +0 -0
  84. {modal-0.73.95 → modal-0.73.97}/modal/cli/queues.py +0 -0
  85. {modal-0.73.95 → modal-0.73.97}/modal/cli/run.py +0 -0
  86. {modal-0.73.95 → modal-0.73.97}/modal/cli/secret.py +0 -0
  87. {modal-0.73.95 → modal-0.73.97}/modal/cli/token.py +0 -0
  88. {modal-0.73.95 → modal-0.73.97}/modal/cli/utils.py +0 -0
  89. {modal-0.73.95 → modal-0.73.97}/modal/cli/volume.py +0 -0
  90. {modal-0.73.95 → modal-0.73.97}/modal/client.py +0 -0
  91. {modal-0.73.95 → modal-0.73.97}/modal/cloud_bucket_mount.py +0 -0
  92. {modal-0.73.95 → modal-0.73.97}/modal/cloud_bucket_mount.pyi +0 -0
  93. {modal-0.73.95 → modal-0.73.97}/modal/cls.py +0 -0
  94. {modal-0.73.95 → modal-0.73.97}/modal/cls.pyi +0 -0
  95. {modal-0.73.95 → modal-0.73.97}/modal/config.py +0 -0
  96. {modal-0.73.95 → modal-0.73.97}/modal/container_process.py +0 -0
  97. {modal-0.73.95 → modal-0.73.97}/modal/container_process.pyi +0 -0
  98. {modal-0.73.95 → modal-0.73.97}/modal/dict.py +0 -0
  99. {modal-0.73.95 → modal-0.73.97}/modal/dict.pyi +0 -0
  100. {modal-0.73.95 → modal-0.73.97}/modal/environments.py +0 -0
  101. {modal-0.73.95 → modal-0.73.97}/modal/environments.pyi +0 -0
  102. {modal-0.73.95 → modal-0.73.97}/modal/exception.py +0 -0
  103. {modal-0.73.95 → modal-0.73.97}/modal/experimental.py +0 -0
  104. {modal-0.73.95 → modal-0.73.97}/modal/experimental.pyi +0 -0
  105. {modal-0.73.95 → modal-0.73.97}/modal/extensions/__init__.py +0 -0
  106. {modal-0.73.95 → modal-0.73.97}/modal/extensions/ipython.py +0 -0
  107. {modal-0.73.95 → modal-0.73.97}/modal/file_io.py +0 -0
  108. {modal-0.73.95 → modal-0.73.97}/modal/file_io.pyi +0 -0
  109. {modal-0.73.95 → modal-0.73.97}/modal/file_pattern_matcher.py +0 -0
  110. {modal-0.73.95 → modal-0.73.97}/modal/functions.py +0 -0
  111. {modal-0.73.95 → modal-0.73.97}/modal/functions.pyi +0 -0
  112. {modal-0.73.95 → modal-0.73.97}/modal/gpu.py +0 -0
  113. {modal-0.73.95 → modal-0.73.97}/modal/io_streams.py +0 -0
  114. {modal-0.73.95 → modal-0.73.97}/modal/io_streams.pyi +0 -0
  115. {modal-0.73.95 → modal-0.73.97}/modal/mount.py +0 -0
  116. {modal-0.73.95 → modal-0.73.97}/modal/mount.pyi +0 -0
  117. {modal-0.73.95 → modal-0.73.97}/modal/network_file_system.py +0 -0
  118. {modal-0.73.95 → modal-0.73.97}/modal/network_file_system.pyi +0 -0
  119. {modal-0.73.95 → modal-0.73.97}/modal/object.py +0 -0
  120. {modal-0.73.95 → modal-0.73.97}/modal/object.pyi +0 -0
  121. {modal-0.73.95 → modal-0.73.97}/modal/output.py +0 -0
  122. {modal-0.73.95 → modal-0.73.97}/modal/parallel_map.pyi +0 -0
  123. {modal-0.73.95 → modal-0.73.97}/modal/partial_function.py +0 -0
  124. {modal-0.73.95 → modal-0.73.97}/modal/partial_function.pyi +0 -0
  125. {modal-0.73.95 → modal-0.73.97}/modal/proxy.py +0 -0
  126. {modal-0.73.95 → modal-0.73.97}/modal/proxy.pyi +0 -0
  127. {modal-0.73.95 → modal-0.73.97}/modal/py.typed +0 -0
  128. {modal-0.73.95 → modal-0.73.97}/modal/queue.py +0 -0
  129. {modal-0.73.95 → modal-0.73.97}/modal/queue.pyi +0 -0
  130. {modal-0.73.95 → modal-0.73.97}/modal/requirements/2023.12.312.txt +0 -0
  131. {modal-0.73.95 → modal-0.73.97}/modal/requirements/2023.12.txt +0 -0
  132. {modal-0.73.95 → modal-0.73.97}/modal/requirements/2024.04.txt +0 -0
  133. {modal-0.73.95 → modal-0.73.97}/modal/requirements/2024.10.txt +0 -0
  134. {modal-0.73.95 → modal-0.73.97}/modal/requirements/README.md +0 -0
  135. {modal-0.73.95 → modal-0.73.97}/modal/retries.py +0 -0
  136. {modal-0.73.95 → modal-0.73.97}/modal/runner.py +0 -0
  137. {modal-0.73.95 → modal-0.73.97}/modal/runner.pyi +0 -0
  138. {modal-0.73.95 → modal-0.73.97}/modal/running_app.py +0 -0
  139. {modal-0.73.95 → modal-0.73.97}/modal/sandbox.py +0 -0
  140. {modal-0.73.95 → modal-0.73.97}/modal/sandbox.pyi +0 -0
  141. {modal-0.73.95 → modal-0.73.97}/modal/schedule.py +0 -0
  142. {modal-0.73.95 → modal-0.73.97}/modal/scheduler_placement.py +0 -0
  143. {modal-0.73.95 → modal-0.73.97}/modal/secret.py +0 -0
  144. {modal-0.73.95 → modal-0.73.97}/modal/secret.pyi +0 -0
  145. {modal-0.73.95 → modal-0.73.97}/modal/serving.py +0 -0
  146. {modal-0.73.95 → modal-0.73.97}/modal/serving.pyi +0 -0
  147. {modal-0.73.95 → modal-0.73.97}/modal/snapshot.py +0 -0
  148. {modal-0.73.95 → modal-0.73.97}/modal/snapshot.pyi +0 -0
  149. {modal-0.73.95 → modal-0.73.97}/modal/stream_type.py +0 -0
  150. {modal-0.73.95 → modal-0.73.97}/modal/token_flow.py +0 -0
  151. {modal-0.73.95 → modal-0.73.97}/modal/token_flow.pyi +0 -0
  152. {modal-0.73.95 → modal-0.73.97}/modal/volume.py +0 -0
  153. {modal-0.73.95 → modal-0.73.97}/modal/volume.pyi +0 -0
  154. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/dependency_links.txt +0 -0
  155. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/entry_points.txt +0 -0
  156. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/requires.txt +0 -0
  157. {modal-0.73.95 → modal-0.73.97}/modal.egg-info/top_level.txt +0 -0
  158. {modal-0.73.95 → modal-0.73.97}/modal_docs/__init__.py +0 -0
  159. {modal-0.73.95 → modal-0.73.97}/modal_docs/gen_cli_docs.py +0 -0
  160. {modal-0.73.95 → modal-0.73.97}/modal_docs/gen_reference_docs.py +0 -0
  161. {modal-0.73.95 → modal-0.73.97}/modal_docs/mdmd/__init__.py +0 -0
  162. {modal-0.73.95 → modal-0.73.97}/modal_docs/mdmd/mdmd.py +0 -0
  163. {modal-0.73.95 → modal-0.73.97}/modal_docs/mdmd/signatures.py +0 -0
  164. {modal-0.73.95 → modal-0.73.97}/modal_proto/__init__.py +0 -0
  165. {modal-0.73.95 → modal-0.73.97}/modal_proto/api.proto +0 -0
  166. {modal-0.73.95 → modal-0.73.97}/modal_proto/api_grpc.py +0 -0
  167. {modal-0.73.95 → modal-0.73.97}/modal_proto/api_pb2.py +0 -0
  168. {modal-0.73.95 → modal-0.73.97}/modal_proto/api_pb2.pyi +0 -0
  169. {modal-0.73.95 → modal-0.73.97}/modal_proto/api_pb2_grpc.py +0 -0
  170. {modal-0.73.95 → modal-0.73.97}/modal_proto/api_pb2_grpc.pyi +0 -0
  171. {modal-0.73.95 → modal-0.73.97}/modal_proto/modal_api_grpc.py +0 -0
  172. {modal-0.73.95 → modal-0.73.97}/modal_proto/modal_options_grpc.py +0 -0
  173. {modal-0.73.95 → modal-0.73.97}/modal_proto/options.proto +0 -0
  174. {modal-0.73.95 → modal-0.73.97}/modal_proto/options_grpc.py +0 -0
  175. {modal-0.73.95 → modal-0.73.97}/modal_proto/options_pb2.py +0 -0
  176. {modal-0.73.95 → modal-0.73.97}/modal_proto/options_pb2.pyi +0 -0
  177. {modal-0.73.95 → modal-0.73.97}/modal_proto/options_pb2_grpc.py +0 -0
  178. {modal-0.73.95 → modal-0.73.97}/modal_proto/options_pb2_grpc.pyi +0 -0
  179. {modal-0.73.95 → modal-0.73.97}/modal_proto/py.typed +0 -0
  180. {modal-0.73.95 → modal-0.73.97}/modal_version/__init__.py +0 -0
  181. {modal-0.73.95 → modal-0.73.97}/modal_version/__main__.py +0 -0
  182. {modal-0.73.95 → modal-0.73.97}/pyproject.toml +0 -0
  183. {modal-0.73.95 → modal-0.73.97}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.2
2
2
  Name: modal
3
- Version: 0.73.95
3
+ Version: 0.73.97
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -8,6 +8,7 @@ import typing
8
8
  import urllib.parse
9
9
  import uuid
10
10
  from collections.abc import AsyncIterator
11
+ from dataclasses import dataclass
11
12
  from typing import (
12
13
  Any,
13
14
  Optional,
@@ -68,6 +69,11 @@ RETRYABLE_GRPC_STATUS_CODES = [
68
69
  Status.INTERNAL,
69
70
  ]
70
71
 
72
+ @dataclass
73
+ class RetryWarningMessage:
74
+ message: str
75
+ warning_interval: int
76
+ errors_to_warn_for: typing.List[Status]
71
77
 
72
78
  def create_channel(
73
79
  server_url: str,
@@ -144,6 +150,7 @@ async def retry_transient_errors(
144
150
  attempt_timeout: Optional[float] = None, # timeout for each attempt
145
151
  total_timeout: Optional[float] = None, # timeout for the entire function call
146
152
  attempt_timeout_floor=2.0, # always have at least this much timeout (only for total_timeout)
153
+ retry_warning_message: Optional[RetryWarningMessage] = None
147
154
  ) -> ResponseType:
148
155
  """Retry on transient gRPC failures with back-off until max_retries is reached.
149
156
  If max_retries is None, retry forever."""
@@ -212,6 +219,9 @@ async def retry_transient_errors(
212
219
 
213
220
  n_retries += 1
214
221
 
222
+ if retry_warning_message and n_retries % retry_warning_message.warning_interval == 0:
223
+ logger.warning(retry_warning_message.message)
224
+
215
225
  await asyncio.sleep(delay)
216
226
  delay = min(delay * delay_factor, max_delay)
217
227
 
@@ -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.73.95"
30
+ self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.73.97"
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.73.95"
88
+ self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.73.97"
89
89
  ): ...
90
90
  def is_closed(self) -> bool: ...
91
91
  @property
@@ -56,13 +56,14 @@ if typing.TYPE_CHECKING:
56
56
  import modal._functions
57
57
 
58
58
  # This is used for both type checking and runtime validation
59
- ImageBuilderVersion = Literal["2023.12", "2024.04", "2024.10"]
59
+ ImageBuilderVersion = Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]
60
60
 
61
61
  # Note: we also define supported Python versions via logic at the top of the package __init__.py
62
62
  # so that we fail fast / clearly in unsupported containers. Additionally, we enumerate the supported
63
63
  # Python versions in mount.py where we specify the "standalone Python versions" we create mounts for.
64
64
  # Consider consolidating these multiple sources of truth?
65
65
  SUPPORTED_PYTHON_SERIES: dict[ImageBuilderVersion, list[str]] = {
66
+ "PREVIEW": ["3.9", "3.10", "3.11", "3.12", "3.13"],
66
67
  "2024.10": ["3.9", "3.10", "3.11", "3.12", "3.13"],
67
68
  "2024.04": ["3.9", "3.10", "3.11", "3.12"],
68
69
  "2023.12": ["3.9", "3.10", "3.11", "3.12"],
@@ -248,9 +249,11 @@ def _get_image_builder_version(server_version: ImageBuilderVersion) -> ImageBuil
248
249
  update_suggestion = "your image builder version using the Modal dashboard"
249
250
  else:
250
251
  update_suggestion = "your client library (pip install --upgrade modal)"
252
+ preview_versions: set[ImageBuilderVersion] = {"PREVIEW"}
253
+ suggested_versions = supported_versions - preview_versions
251
254
  raise VersionError(
252
255
  "This version of the modal client supports the following image builder versions:"
253
- f" {supported_versions!r}."
256
+ f" {suggested_versions!r}."
254
257
  f"\n\nYou are using {version!r}{version_source}."
255
258
  f" Please update {update_suggestion}."
256
259
  )
@@ -16,7 +16,7 @@ import pathlib
16
16
  import typing
17
17
  import typing_extensions
18
18
 
19
- ImageBuilderVersion = typing.Literal["2023.12", "2024.04", "2024.10"]
19
+ ImageBuilderVersion = typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]
20
20
 
21
21
  class _AutoDockerIgnoreSentinel:
22
22
  def __repr__(self) -> str: ...
@@ -26,17 +26,21 @@ AUTO_DOCKERIGNORE: _AutoDockerIgnoreSentinel
26
26
 
27
27
  def _validate_python_version(
28
28
  python_version: typing.Optional[str],
29
- builder_version: typing.Literal["2023.12", "2024.04", "2024.10"],
29
+ builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
30
30
  allow_micro_granularity: bool = True,
31
31
  ) -> str: ...
32
32
  def _dockerhub_python_version(
33
- builder_version: typing.Literal["2023.12", "2024.04", "2024.10"], python_version: typing.Optional[str] = None
33
+ builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
34
+ python_version: typing.Optional[str] = None,
34
35
  ) -> str: ...
35
- def _base_image_config(group: str, builder_version: typing.Literal["2023.12", "2024.04", "2024.10"]) -> typing.Any: ...
36
+ def _base_image_config(
37
+ group: str, builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]
38
+ ) -> typing.Any: ...
36
39
  def _get_modal_requirements_path(
37
- builder_version: typing.Literal["2023.12", "2024.04", "2024.10"], python_version: typing.Optional[str] = None
40
+ builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
41
+ python_version: typing.Optional[str] = None,
38
42
  ) -> str: ...
39
- def _get_modal_requirements_command(version: typing.Literal["2023.12", "2024.04", "2024.10"]) -> str: ...
43
+ def _get_modal_requirements_command(version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]) -> str: ...
40
44
  def _flatten_str_args(
41
45
  function_name: str, arg_name: str, args: collections.abc.Sequence[typing.Union[str, list[str]]]
42
46
  ) -> list[str]: ...
@@ -50,8 +54,8 @@ def _make_pip_install_args(
50
54
  extra_options: str = "",
51
55
  ) -> str: ...
52
56
  def _get_image_builder_version(
53
- server_version: typing.Literal["2023.12", "2024.04", "2024.10"],
54
- ) -> typing.Literal["2023.12", "2024.04", "2024.10"]: ...
57
+ server_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
58
+ ) -> typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]: ...
55
59
  def _create_context_mount(
56
60
  docker_commands: collections.abc.Sequence[str],
57
61
  ignore_fn: collections.abc.Callable[[pathlib.Path], bool],
@@ -100,7 +104,7 @@ class _Image(modal._object._Object):
100
104
  *,
101
105
  base_images: typing.Optional[dict[str, _Image]] = None,
102
106
  dockerfile_function: typing.Optional[
103
- collections.abc.Callable[[typing.Literal["2023.12", "2024.04", "2024.10"]], DockerfileSpec]
107
+ collections.abc.Callable[[typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]], DockerfileSpec]
104
108
  ] = None,
105
109
  secrets: typing.Optional[collections.abc.Sequence[modal.secret._Secret]] = None,
106
110
  gpu_config: typing.Optional[modal_proto.api_pb2.GPUConfig] = None,
@@ -246,7 +250,7 @@ class _Image(modal._object._Object):
246
250
  @staticmethod
247
251
  def _registry_setup_commands(
248
252
  tag: str,
249
- builder_version: typing.Literal["2023.12", "2024.04", "2024.10"],
253
+ builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
250
254
  setup_commands: list[str],
251
255
  add_python: typing.Optional[str] = None,
252
256
  ) -> list[str]: ...
@@ -354,7 +358,7 @@ class Image(modal.object.Object):
354
358
  *,
355
359
  base_images: typing.Optional[dict[str, Image]] = None,
356
360
  dockerfile_function: typing.Optional[
357
- collections.abc.Callable[[typing.Literal["2023.12", "2024.04", "2024.10"]], DockerfileSpec]
361
+ collections.abc.Callable[[typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"]], DockerfileSpec]
358
362
  ] = None,
359
363
  secrets: typing.Optional[collections.abc.Sequence[modal.secret.Secret]] = None,
360
364
  gpu_config: typing.Optional[modal_proto.api_pb2.GPUConfig] = None,
@@ -505,7 +509,7 @@ class Image(modal.object.Object):
505
509
  @staticmethod
506
510
  def _registry_setup_commands(
507
511
  tag: str,
508
- builder_version: typing.Literal["2023.12", "2024.04", "2024.10"],
512
+ builder_version: typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"],
509
513
  setup_commands: list[str],
510
514
  add_python: typing.Optional[str] = None,
511
515
  ) -> list[str]: ...
@@ -595,4 +599,4 @@ class Image(modal.object.Object):
595
599
 
596
600
  _logs: ___logs_spec[typing_extensions.Self]
597
601
 
598
- SUPPORTED_PYTHON_SERIES: dict[typing.Literal["2023.12", "2024.04", "2024.10"], list[str]]
602
+ SUPPORTED_PYTHON_SERIES: dict[typing.Literal["2023.12", "2024.04", "2024.10", "PREVIEW"], list[str]]
@@ -5,7 +5,7 @@ import typing
5
5
  from dataclasses import dataclass
6
6
  from typing import Any, Callable, Optional
7
7
 
8
- from grpclib import GRPCError, Status
8
+ from grpclib import Status
9
9
 
10
10
  from modal._runtime.execution_context import current_input_id
11
11
  from modal._utils.async_utils import (
@@ -27,13 +27,17 @@ from modal._utils.function_utils import (
27
27
  _create_input,
28
28
  _process_result,
29
29
  )
30
- from modal._utils.grpc_utils import retry_transient_errors
30
+ from modal._utils.grpc_utils import RETRYABLE_GRPC_STATUS_CODES, RetryWarningMessage, retry_transient_errors
31
31
  from modal.config import logger
32
32
  from modal_proto import api_pb2
33
33
 
34
34
  if typing.TYPE_CHECKING:
35
35
  import modal.client
36
36
 
37
+ # pump_inputs should retry if it receives any of the standard retryable codes plus RESOURCE_EXHAUSTED.
38
+ PUMP_INPUTS_RETRYABLE_GRPC_STATUS_CODES = RETRYABLE_GRPC_STATUS_CODES + [Status.RESOURCE_EXHAUSTED]
39
+ PUMP_INPUTS_MAX_RETRIES = 8
40
+ PUMP_INPUTS_MAX_RETRY_DELAY=15
37
41
 
38
42
  class _SynchronizedQueue:
39
43
  """mdmd:hidden"""
@@ -136,25 +140,19 @@ async def _map_invocation(
136
140
  logger.debug(
137
141
  f"Pushing {len(items)} inputs to server. Num queued inputs awaiting push is {input_queue.qsize()}."
138
142
  )
139
- while True:
140
- try:
141
- resp = await retry_transient_errors(
142
- client.stub.FunctionPutInputs,
143
- request,
144
- # with 8 retries we log the warning below about every 30 secondswhich isn't too spammy.
145
- max_retries=8,
146
- max_delay=15,
147
- additional_status_codes=[Status.RESOURCE_EXHAUSTED],
148
- )
149
- break
150
- except GRPCError as err:
151
- if err.status != Status.RESOURCE_EXHAUSTED:
152
- raise err
153
- logger.warning(
154
- f"Warning: map progress for function {function._function_name} is limited."
155
- " Common bottlenecks include slow iteration over results, or function backlogs."
156
- )
157
-
143
+ # with 8 retries we log the warning below about every 30 seconds which isn't too spammy.
144
+ retry_warning_message = RetryWarningMessage(
145
+ message=f"Warning: map progress for function {function._function_name} is limited."
146
+ " Common bottlenecks include slow iteration over results, or function backlogs.",
147
+ warning_interval=8,
148
+ errors_to_warn_for=[Status.RESOURCE_EXHAUSTED])
149
+ resp = await retry_transient_errors(
150
+ client.stub.FunctionPutInputs,
151
+ request,
152
+ max_retries=None,
153
+ max_delay=PUMP_INPUTS_MAX_RETRY_DELAY,
154
+ additional_status_codes=[Status.RESOURCE_EXHAUSTED],
155
+ retry_warning_message=retry_warning_message)
158
156
  count_update()
159
157
  for item in resp.inputs:
160
158
  pending_outputs.setdefault(item.input_id, 0)
@@ -0,0 +1,16 @@
1
+ aiohappyeyeballs==2.4.3
2
+ aiohttp==3.10.8
3
+ aiosignal==1.3.1
4
+ async-timeout==4.0.3 ; python_version < "3.11"
5
+ attrs==24.2.0
6
+ certifi==2024.8.30
7
+ frozenlist==1.4.1
8
+ grpclib==0.4.7
9
+ h2==4.1.0
10
+ hpack==4.0.0
11
+ hyperframe==6.0.1
12
+ idna==3.10
13
+ multidict==6.1.0
14
+ protobuf>=3.20,<6
15
+ typing_extensions==4.12.2
16
+ yarl==1.13.1
@@ -1,22 +1,26 @@
1
1
  {
2
2
  "debian": {
3
+ "PREVIEW": "bookworm",
3
4
  "2024.10": "bookworm",
4
5
  "2024.04": "bookworm",
5
6
  "2023.12": "bullseye"
6
7
  },
7
8
  "python": {
9
+ "PREVIEW": ["3.9.20", "3.10.15", "3.11.10", "3.12.6", "3.13.0"],
8
10
  "2024.10": ["3.9.20", "3.10.15", "3.11.10", "3.12.6", "3.13.0"],
9
11
  "2024.04": ["3.9.19", "3.10.14", "3.11.8", "3.12.2"],
10
12
  "2023.12": ["3.9.15", "3.10.8", "3.11.0", "3.12.1"]
11
13
  },
12
14
  "micromamba": {
15
+ "PREVIEW": "1.5.10",
13
16
  "2024.10": "1.5.10",
14
17
  "2024.04": "1.5.8",
15
18
  "2023.12": "1.3.1"
16
19
  },
17
20
  "package_tools": {
21
+ "PREVIEW": "pip wheel uv",
18
22
  "2024.10": "pip wheel uv",
19
23
  "2024.04": "pip wheel uv",
20
24
  "2023.12": "pip"
21
25
  }
22
- }
26
+ }
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.2
2
2
  Name: modal
3
- Version: 0.73.95
3
+ Version: 0.73.97
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -151,6 +151,7 @@ modal/requirements/2023.12.312.txt
151
151
  modal/requirements/2023.12.txt
152
152
  modal/requirements/2024.04.txt
153
153
  modal/requirements/2024.10.txt
154
+ modal/requirements/PREVIEW.txt
154
155
  modal/requirements/README.md
155
156
  modal/requirements/base-images.json
156
157
  modal_docs/__init__.py
@@ -1,4 +1,4 @@
1
1
  # Copyright Modal Labs 2025
2
2
 
3
3
  # Note: Reset this value to -1 whenever you make a minor `0.X` release of the client.
4
- build_number = 95 # git: 4e11a66
4
+ build_number = 97 # git: 20eb8cc
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes