modal 0.75.1__py3-none-any.whl → 0.75.3__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/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.75.1"
30
+ self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.75.3"
31
31
  ): ...
32
32
  def is_closed(self) -> bool: ...
33
33
  @property
@@ -86,7 +86,7 @@ class Client:
86
86
  _snapshotted: bool
87
87
 
88
88
  def __init__(
89
- self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.75.1"
89
+ self, server_url: str, client_type: int, credentials: typing.Optional[tuple[str, str]], version: str = "0.75.3"
90
90
  ): ...
91
91
  def is_closed(self) -> bool: ...
92
92
  @property
modal/dict.py CHANGED
@@ -35,10 +35,12 @@ class _Dict(_Object, type_prefix="di"):
35
35
 
36
36
  **Lifetime of a Dict and its items**
37
37
 
38
- An individual dict entry will expire 30 days after it was last added to its Dict object.
39
- Additionally, data are stored in memory on the Modal server and could be lost due to
40
- unexpected server restarts. Because of this, `Dict` is best suited for storing short-term
41
- state and is not recommended for durable storage.
38
+ An individual Dict entry will expire after 7 days of inactivity (no reads or writes). The
39
+ Dict entries are written to durable storage.
40
+
41
+ Legacy Dicts (created before 2025-05-20) will still have entries expire 30 days after being
42
+ last added. Additionally, data are stored in memory on the Modal server and could be lost due to
43
+ unexpected server restarts. Eventually, these Dicts will be fully sunset.
42
44
 
43
45
  **Usage**
44
46
 
@@ -275,13 +277,18 @@ class _Dict(_Object, type_prefix="di"):
275
277
  raise exc
276
278
 
277
279
  @live_method
278
- async def put(self, key: Any, value: Any) -> None:
279
- """Add a specific key-value pair to the dictionary."""
280
+ async def put(self, key: Any, value: Any, *, skip_if_exists: bool = False) -> bool:
281
+ """Add a specific key-value pair to the dictionary.
282
+
283
+ Returns True if the key-value pair was added and False if it wasn't because the key already existed and
284
+ `skip_if_exists` was set.
285
+ """
280
286
  updates = {key: value}
281
287
  serialized = _serialize_dict(updates)
282
- req = api_pb2.DictUpdateRequest(dict_id=self.object_id, updates=serialized)
288
+ req = api_pb2.DictUpdateRequest(dict_id=self.object_id, updates=serialized, if_not_exists=skip_if_exists)
283
289
  try:
284
- await retry_transient_errors(self._client.stub.DictUpdate, req)
290
+ resp = await retry_transient_errors(self._client.stub.DictUpdate, req)
291
+ return resp.created
285
292
  except GRPCError as exc:
286
293
  if "status = '413'" in exc.message:
287
294
  raise RequestSizeError("Dict.put request is too large") from exc
modal/dict.pyi CHANGED
@@ -49,7 +49,7 @@ class _Dict(modal._object._Object):
49
49
  async def len(self) -> int: ...
50
50
  async def __getitem__(self, key: typing.Any) -> typing.Any: ...
51
51
  async def update(self, other: typing.Optional[collections.abc.Mapping] = None, /, **kwargs) -> None: ...
52
- async def put(self, key: typing.Any, value: typing.Any) -> None: ...
52
+ async def put(self, key: typing.Any, value: typing.Any, *, skip_if_exists: bool = False) -> bool: ...
53
53
  async def __setitem__(self, key: typing.Any, value: typing.Any) -> None: ...
54
54
  async def pop(self, key: typing.Any) -> typing.Any: ...
55
55
  async def __delitem__(self, key: typing.Any) -> typing.Any: ...
@@ -161,8 +161,8 @@ class Dict(modal.object.Object):
161
161
  update: __update_spec[typing_extensions.Self]
162
162
 
163
163
  class __put_spec(typing_extensions.Protocol[SUPERSELF]):
164
- def __call__(self, /, key: typing.Any, value: typing.Any) -> None: ...
165
- async def aio(self, /, key: typing.Any, value: typing.Any) -> None: ...
164
+ def __call__(self, /, key: typing.Any, value: typing.Any, *, skip_if_exists: bool = False) -> bool: ...
165
+ async def aio(self, /, key: typing.Any, value: typing.Any, *, skip_if_exists: bool = False) -> bool: ...
166
166
 
167
167
  put: __put_spec[typing_extensions.Self]
168
168
 
@@ -94,8 +94,6 @@ class FilePatternMatcher(_AbstractPatternMatcher):
94
94
  raise ValueError('Illegal exclusion pattern: "!"')
95
95
  new_pattern.exclusion = True
96
96
  pattern = pattern[1:]
97
- if os.path.isabs(pattern):
98
- raise ValueError("Ignore patterns cannot be absolute paths")
99
97
  # In Python, we can proceed without explicit syntax checking
100
98
  new_pattern.cleaned_pattern = pattern
101
99
  new_pattern.dirs = pattern.split(os.path.sep)
modal/functions.pyi CHANGED
@@ -234,11 +234,11 @@ class Function(
234
234
 
235
235
  _call_generator_nowait: ___call_generator_nowait_spec[typing_extensions.Self]
236
236
 
237
- class __remote_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
237
+ class __remote_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
238
238
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> ReturnType_INNER: ...
239
239
  async def aio(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> ReturnType_INNER: ...
240
240
 
241
- remote: __remote_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
241
+ remote: __remote_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
242
242
 
243
243
  class __remote_gen_spec(typing_extensions.Protocol[SUPERSELF]):
244
244
  def __call__(self, /, *args, **kwargs) -> typing.Generator[typing.Any, None, None]: ...
@@ -253,12 +253,12 @@ class Function(
253
253
  self, *args: modal._functions.P.args, **kwargs: modal._functions.P.kwargs
254
254
  ) -> modal._functions.OriginalReturnType: ...
255
255
 
256
- class ___experimental_spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
256
+ class ___experimental_spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
257
257
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
258
258
  async def aio(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
259
259
 
260
260
  _experimental_spawn: ___experimental_spawn_spec[
261
- modal._functions.P, modal._functions.ReturnType, typing_extensions.Self
261
+ modal._functions.ReturnType, modal._functions.P, typing_extensions.Self
262
262
  ]
263
263
 
264
264
  class ___spawn_map_inner_spec(typing_extensions.Protocol[P_INNER, SUPERSELF]):
@@ -267,11 +267,11 @@ class Function(
267
267
 
268
268
  _spawn_map_inner: ___spawn_map_inner_spec[modal._functions.P, typing_extensions.Self]
269
269
 
270
- class __spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
270
+ class __spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
271
271
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
272
272
  async def aio(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
273
273
 
274
- spawn: __spawn_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
274
+ spawn: __spawn_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
275
275
 
276
276
  def get_raw_f(self) -> collections.abc.Callable[..., typing.Any]: ...
277
277
 
modal/image.py CHANGED
@@ -279,9 +279,8 @@ def _create_context_mount(
279
279
  include_fn = FilePatternMatcher(*copy_patterns)
280
280
 
281
281
  def ignore_with_include(source: Path) -> bool:
282
- if source.is_absolute():
283
- source = source.relative_to(context_dir)
284
- if not include_fn(source) or ignore_fn(source):
282
+ relative_source = source.relative_to(context_dir)
283
+ if not include_fn(relative_source) or ignore_fn(relative_source):
285
284
  return True
286
285
 
287
286
  return False
modal/mount.py CHANGED
@@ -153,9 +153,9 @@ class _MountDir(_MountEntry):
153
153
 
154
154
  for local_filename in gen:
155
155
  local_path = Path(local_filename)
156
- rel_local_path = local_path.relative_to(local_dir)
157
- if not self.ignore(rel_local_path):
158
- mount_path = self.remote_path / rel_local_path.as_posix()
156
+ if not self.ignore(local_path):
157
+ local_relpath = local_path.expanduser().absolute().relative_to(local_dir)
158
+ mount_path = self.remote_path / local_relpath.as_posix()
159
159
  yield local_path.resolve(), mount_path
160
160
 
161
161
  def watch_entry(self):
@@ -339,8 +339,8 @@ class _Mount(_Object, type_prefix="mo"):
339
339
  return _Mount._new()._extend(
340
340
  _MountDir(
341
341
  local_dir=local_path,
342
- remote_path=remote_path,
343
342
  ignore=ignore,
343
+ remote_path=remote_path,
344
344
  recursive=True,
345
345
  ),
346
346
  )
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: modal
3
- Version: 0.75.1
3
+ Version: 0.75.3
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -22,7 +22,7 @@ modal/app.py,sha256=xojuGZv4LaQwZU5ntj7WbmMjeNuB9Gll8Mzqe2LyiEs,51323
22
22
  modal/app.pyi,sha256=zNwR1_2LpmQc9AhemuAeVdk90XNYDw9keOkXAwAATeA,28732
23
23
  modal/call_graph.py,sha256=1g2DGcMIJvRy-xKicuf63IVE98gJSnQsr8R_NVMptNc,2581
24
24
  modal/client.py,sha256=o-aQThHpvDHUzg_kUafyhWzACViUBhY2WLZ2EitnSHA,16787
25
- modal/client.pyi,sha256=pPWEh6maNF0V7pvpLnCRw60B_aUEpB8Au_EhJsW9C3M,8383
25
+ modal/client.pyi,sha256=6gdjIEcvM8cVSKHfdpfQfQj6aqI5HWxDEbcpSavE5Ls,8383
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=aHoMEWMZUN7bOezs3tRPxzS1FP3gTxZBORVjbPmtxyg,35338
@@ -30,22 +30,22 @@ modal/cls.pyi,sha256=klQkXHXhl5gTjRo_XDV6PGecnJeFOudSMXid_kanumI,12090
30
30
  modal/config.py,sha256=OOMEJ5LHNFbHRW5wUpuhl0TH6EPW8D1XV9I3OJXrZrk,12668
31
31
  modal/container_process.py,sha256=vvyK3DVPUMsuqvkKdUiQ49cDLF9JawGrxpglLk5vfgI,6208
32
32
  modal/container_process.pyi,sha256=cR4aRHTbcVvmxGCc1_k2Ey8JllJIAQYq9PBKx0_1TuI,2916
33
- modal/dict.py,sha256=rYOTNtUIOoIVibC1KGC6-fnsXD88ij8o0xNKFpR9BVs,14134
34
- modal/dict.pyi,sha256=ou2rospKvSNl0PffrsSABruTjspUlqwbkjJ22fzg-yE,8021
33
+ modal/dict.py,sha256=wOifmVwJS-X6FS0JcvJCfL9mrny6zWxk3EHmp-2MDpU,14481
34
+ modal/dict.pyi,sha256=RBaQyOd1ABRNN7vIf5L_rv94y7Kq5Qn9IlKHSr4j8N0,8120
35
35
  modal/environments.py,sha256=t_TdUyORfIjlEAOS7I4My1qHi0cVsjPxwKloLmAAZMc,6935
36
36
  modal/environments.pyi,sha256=4HbI0kywveaUVI7HqDtZ4HphCTGXYi_wie2hz87up5A,3425
37
37
  modal/exception.py,sha256=4JyO-SACaLNDe2QC48EjsK8GMkZ8AgEurZ8j1YdRu8E,5263
38
38
  modal/file_io.py,sha256=lcMs_E9Xfm0YX1t9U2wNIBPnqHRxmImqjLW1GHqVmyg,20945
39
39
  modal/file_io.pyi,sha256=oB7x-rKq7bmm8cA7Z7W9C9yeko7KK9m9i5GidFnkGK4,9569
40
- modal/file_pattern_matcher.py,sha256=R0BChpQUXiCDUv__dWTovRMrw1iBTg0kw5p1vwOqcqk,6613
40
+ modal/file_pattern_matcher.py,sha256=trosX-Bp7dOubudN1bLLhRAoidWy1TcoaR4Pv8CedWw,6497
41
41
  modal/functions.py,sha256=kcNHvqeGBxPI7Cgd57NIBBghkfbeFJzXO44WW0jSmao,325
42
- modal/functions.pyi,sha256=xc6igwNS77kYkyuVpWOlMPkanxriesIaKFTYlvmuXp0,16993
42
+ modal/functions.pyi,sha256=3kHFXeXV0YYg0w1hmBqSV4k7djmjU9qfrmUyURlp8lE,16993
43
43
  modal/gpu.py,sha256=Kbhs_u49FaC2Zi0TjCdrpstpRtT5eZgecynmQi5IZVE,6752
44
- modal/image.py,sha256=lg3XxIWNYV8je88oEPbihzSEq_9gumc0NLbzZLAhm74,92807
44
+ modal/image.py,sha256=ZCghS6l1O7pezXcdMHk6RoJpW3qWszfWGJTW38lNXaU,92797
45
45
  modal/image.pyi,sha256=MDq7tNJevElK78VxFYrZRe_00kz9gPdg98MN5c6fFoE,25644
46
46
  modal/io_streams.py,sha256=YDZVQSDv05DeXg5TwcucC9Rj5hQBx2GXdluan9rIUpw,15467
47
47
  modal/io_streams.pyi,sha256=1UK6kWLREASQfq-wL9wSp5iqjLU0egRZPDn4LXs1PZY,5136
48
- modal/mount.py,sha256=lGxbu6tTEA8EdePQWYs2uu0Z3f3H_PjiXwDXOmStwrM,32674
48
+ modal/mount.py,sha256=_YoJ1TYgttZYGCS-698e8XpExt-mYr994-cMBSOU6EE,32696
49
49
  modal/mount.pyi,sha256=PHs-N9LGSDfYWw70UrhvGZW_6uWwyx-1GAieROzCNNs,12583
50
50
  modal/network_file_system.py,sha256=9_4EFTM1tKSTdwk0f4beSlWCiat6gC6ZjbsOWje07TM,14779
51
51
  modal/network_file_system.pyi,sha256=58DiUqHGlARmI3cz-Yo7IFObKKFIiGh5UIU5JxGNFfc,8333
@@ -146,7 +146,7 @@ modal/requirements/2024.10.txt,sha256=qD-5cVIVM9wXesJ6JC89Ew-3m2KjEElUz3jaw_MddR
146
146
  modal/requirements/PREVIEW.txt,sha256=qD-5cVIVM9wXesJ6JC89Ew-3m2KjEElUz3jaw_MddRo,296
147
147
  modal/requirements/README.md,sha256=9tK76KP0Uph7O0M5oUgsSwEZDj5y-dcUPsnpR0Sc-Ik,854
148
148
  modal/requirements/base-images.json,sha256=57vMSqzMbLBxw5tFWSaMiIkkVEps4JfX5PAtXGnkS4U,740
149
- modal-0.75.1.dist-info/licenses/LICENSE,sha256=psuoW8kuDP96RQsdhzwOqi6fyWv0ct8CR6Jr7He_P_k,10173
149
+ modal-0.75.3.dist-info/licenses/LICENSE,sha256=psuoW8kuDP96RQsdhzwOqi6fyWv0ct8CR6Jr7He_P_k,10173
150
150
  modal_docs/__init__.py,sha256=svYKtV8HDwDCN86zbdWqyq5T8sMdGDj0PVlzc2tIxDM,28
151
151
  modal_docs/gen_cli_docs.py,sha256=c1yfBS_x--gL5bs0N4ihMwqwX8l3IBWSkBAKNNIi6bQ,3801
152
152
  modal_docs/gen_reference_docs.py,sha256=d_CQUGQ0rfw28u75I2mov9AlS773z9rG40-yq5o7g2U,6359
@@ -171,9 +171,9 @@ modal_proto/options_pb2_grpc.pyi,sha256=CImmhxHsYnF09iENPoe8S4J-n93jtgUYD2JPAc0y
171
171
  modal_proto/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
172
172
  modal_version/__init__.py,sha256=PenIvZdwt-HVdbetAyxuPoyZTtzx2moKZoJLK8iZ804,470
173
173
  modal_version/__main__.py,sha256=2FO0yYQQwDTh6udt1h-cBnGd1c4ZyHnHSI4BksxzVac,105
174
- modal_version/_version_generated.py,sha256=_nDSI8hGHxzdQjG0q_a6V1UkzG2U0-lDGnd-alp1KWU,148
175
- modal-0.75.1.dist-info/METADATA,sha256=rI8s9zk_8oi-zEgq1ErtRePjGukBN2ino__FTqWzJ1E,2450
176
- modal-0.75.1.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
177
- modal-0.75.1.dist-info/entry_points.txt,sha256=An-wYgeEUnm6xzrAP9_NTSTSciYvvEWsMZILtYrvpAI,46
178
- modal-0.75.1.dist-info/top_level.txt,sha256=4BWzoKYREKUZ5iyPzZpjqx4G8uB5TWxXPDwibLcVa7k,43
179
- modal-0.75.1.dist-info/RECORD,,
174
+ modal_version/_version_generated.py,sha256=k8TneCpSkVGQTxxbVJf-7dhd8Izlr_51vflyqUsT-sk,148
175
+ modal-0.75.3.dist-info/METADATA,sha256=JHDZFqFdzeLOo5uYaICidpvckcdVc815cE9FRsyou7I,2450
176
+ modal-0.75.3.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
177
+ modal-0.75.3.dist-info/entry_points.txt,sha256=An-wYgeEUnm6xzrAP9_NTSTSciYvvEWsMZILtYrvpAI,46
178
+ modal-0.75.3.dist-info/top_level.txt,sha256=4BWzoKYREKUZ5iyPzZpjqx4G8uB5TWxXPDwibLcVa7k,43
179
+ modal-0.75.3.dist-info/RECORD,,
@@ -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 = 1 # git: 0f5c526
4
+ build_number = 3 # git: 3c9ad21
File without changes