modal 0.73.164__py3-none-any.whl → 0.73.165__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.
@@ -46,13 +46,13 @@ from modal.exception import ExecutionError, InputCancellation, InvalidError
46
46
  from modal.running_app import RunningApp, running_app_from_layout
47
47
  from modal_proto import api_pb2
48
48
 
49
+ from ._runtime import execution_context
49
50
  from ._runtime.container_io_manager import (
50
51
  ContainerIOManager,
51
52
  IOContext,
52
53
  UserException,
53
54
  _ContainerIOManager,
54
55
  )
55
- from ._runtime.execution_context import _set_current_context_ids
56
56
 
57
57
  if TYPE_CHECKING:
58
58
  import modal._object
@@ -190,7 +190,7 @@ def call_function(
190
190
  async def run_input_async(io_context: IOContext) -> None:
191
191
  started_at = time.time()
192
192
  input_ids, function_call_ids = io_context.input_ids, io_context.function_call_ids
193
- reset_context = _set_current_context_ids(input_ids, function_call_ids)
193
+ reset_context = execution_context._set_current_context_ids(input_ids, function_call_ids)
194
194
  async with container_io_manager.handle_input_exception.aio(io_context, started_at):
195
195
  res = io_context.call_finalized_function()
196
196
  # TODO(erikbern): any exception below shouldn't be considered a user exception
@@ -240,7 +240,7 @@ def call_function(
240
240
  def run_input_sync(io_context: IOContext) -> None:
241
241
  started_at = time.time()
242
242
  input_ids, function_call_ids = io_context.input_ids, io_context.function_call_ids
243
- reset_context = _set_current_context_ids(input_ids, function_call_ids)
243
+ reset_context = execution_context._set_current_context_ids(input_ids, function_call_ids)
244
244
  with container_io_manager.handle_input_exception(io_context, started_at):
245
245
  res = io_context.call_finalized_function()
246
246
 
@@ -407,33 +407,34 @@ def main(container_args: api_pb2.ContainerArguments, client: Client):
407
407
  param_args = ()
408
408
  param_kwargs = {}
409
409
 
410
- if function_def.is_class:
411
- # this is a bit ugly - match the function and class based on function name to get metadata
412
- # This metadata is required in order to hydrate the class in case it's not globally
413
- # decorated (or serialized)
414
- service_base_function_id = container_args.app_layout.function_ids[function_def.function_name]
415
- service_function_hydration_data = [
416
- o for o in container_args.app_layout.objects if o.object_id == service_base_function_id
417
- ][0]
418
- class_id = container_args.app_layout.class_ids[function_def.function_name.removesuffix(".*")]
419
-
420
- service = import_class_service(
421
- function_def,
422
- service_function_hydration_data,
423
- class_id,
424
- client,
425
- ser_usr_cls,
426
- param_args,
427
- param_kwargs,
428
- )
429
- else:
430
- service = import_single_function_service(
431
- function_def,
432
- ser_usr_cls,
433
- ser_fun,
434
- param_args,
435
- param_kwargs,
436
- )
410
+ with execution_context._import_context():
411
+ if function_def.is_class:
412
+ # this is a bit ugly - match the function and class based on function name to get metadata
413
+ # This metadata is required in order to hydrate the class in case it's not globally
414
+ # decorated (or serialized)
415
+ service_base_function_id = container_args.app_layout.function_ids[function_def.function_name]
416
+ service_function_hydration_data = [
417
+ o for o in container_args.app_layout.objects if o.object_id == service_base_function_id
418
+ ][0]
419
+ class_id = container_args.app_layout.class_ids[function_def.function_name.removesuffix(".*")]
420
+
421
+ service = import_class_service(
422
+ function_def,
423
+ service_function_hydration_data,
424
+ class_id,
425
+ client,
426
+ ser_usr_cls,
427
+ param_args,
428
+ param_kwargs,
429
+ )
430
+ else:
431
+ service = import_single_function_service(
432
+ function_def,
433
+ ser_usr_cls,
434
+ ser_fun,
435
+ param_args,
436
+ param_kwargs,
437
+ )
437
438
 
438
439
  # If the cls/function decorator was applied in local scope, but the app is global, we can look it up
439
440
  if service.app is not None:
@@ -1,4 +1,5 @@
1
1
  # Copyright Modal Labs 2024
2
+ from contextlib import contextmanager
2
3
  from contextvars import ContextVar
3
4
  from typing import Callable, Optional
4
5
 
@@ -87,3 +88,15 @@ def _set_current_context_ids(input_ids: list[str], function_call_ids: list[str])
87
88
 
88
89
  _current_input_id: ContextVar = ContextVar("_current_input_id")
89
90
  _current_function_call_id: ContextVar = ContextVar("_current_function_call_id")
91
+
92
+ _is_currently_importing = False # we set this to True while a container is importing user code
93
+
94
+
95
+ @contextmanager
96
+ def _import_context():
97
+ global _is_currently_importing
98
+ _is_currently_importing = True
99
+ try:
100
+ yield
101
+ finally:
102
+ _is_currently_importing = False
@@ -17,6 +17,7 @@ def current_function_call_id() -> typing.Optional[str]: ...
17
17
  def _set_current_context_ids(
18
18
  input_ids: list[str], function_call_ids: list[str]
19
19
  ) -> collections.abc.Callable[[], None]: ...
20
+ def _import_context(): ...
20
21
 
21
22
  _current_input_id: contextvars.ContextVar
22
23
 
modal/client.pyi CHANGED
@@ -31,7 +31,7 @@ class _Client:
31
31
  server_url: str,
32
32
  client_type: int,
33
33
  credentials: typing.Optional[tuple[str, str]],
34
- version: str = "0.73.164",
34
+ version: str = "0.73.165",
35
35
  ): ...
36
36
  def is_closed(self) -> bool: ...
37
37
  @property
@@ -93,7 +93,7 @@ class Client:
93
93
  server_url: str,
94
94
  client_type: int,
95
95
  credentials: typing.Optional[tuple[str, str]],
96
- version: str = "0.73.164",
96
+ version: str = "0.73.165",
97
97
  ): ...
98
98
  def is_closed(self) -> bool: ...
99
99
  @property
modal/functions.pyi CHANGED
@@ -200,11 +200,11 @@ class Function(
200
200
 
201
201
  _call_generator_nowait: ___call_generator_nowait_spec[typing_extensions.Self]
202
202
 
203
- class __remote_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
203
+ class __remote_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
204
204
  def __call__(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> ReturnType_INNER: ...
205
205
  async def aio(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> ReturnType_INNER: ...
206
206
 
207
- remote: __remote_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
207
+ remote: __remote_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
208
208
 
209
209
  class __remote_gen_spec(typing_extensions.Protocol[SUPERSELF]):
210
210
  def __call__(self, *args, **kwargs) -> typing.Generator[typing.Any, None, None]: ...
@@ -219,19 +219,19 @@ class Function(
219
219
  self, *args: modal._functions.P.args, **kwargs: modal._functions.P.kwargs
220
220
  ) -> modal._functions.OriginalReturnType: ...
221
221
 
222
- class ___experimental_spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
222
+ class ___experimental_spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
223
223
  def __call__(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
224
224
  async def aio(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
225
225
 
226
226
  _experimental_spawn: ___experimental_spawn_spec[
227
- modal._functions.ReturnType, modal._functions.P, typing_extensions.Self
227
+ modal._functions.P, modal._functions.ReturnType, typing_extensions.Self
228
228
  ]
229
229
 
230
- class __spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
230
+ class __spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
231
231
  def __call__(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
232
232
  async def aio(self, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]: ...
233
233
 
234
- spawn: __spawn_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
234
+ spawn: __spawn_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
235
235
 
236
236
  def get_raw_f(self) -> collections.abc.Callable[..., typing.Any]: ...
237
237
 
modal/mount.py CHANGED
@@ -655,8 +655,6 @@ class _Mount(_Object, type_prefix="mo"):
655
655
  condition: Optional[Callable[[str], bool]] = None,
656
656
  ignore: Optional[Union[Sequence[str], Callable[[Path], bool]]] = None,
657
657
  ) -> "_Mount":
658
- # Don't re-run inside container.
659
-
660
658
  if condition is not None:
661
659
  if ignore is not None:
662
660
  raise InvalidError("Cannot specify both `ignore` and `condition`")
@@ -669,10 +667,6 @@ class _Mount(_Object, type_prefix="mo"):
669
667
  ignore = FilePatternMatcher(*ignore)
670
668
 
671
669
  mount = _Mount._new()
672
- from ._runtime.execution_context import is_local
673
-
674
- if not is_local():
675
- return mount # empty/non-mountable mount in case it's used from within a container
676
670
  for module_name in module_names:
677
671
  mount = mount._extend(_MountedPythonModule(module_name, remote_dir, ignore))
678
672
  return mount
modal/runner.py CHANGED
@@ -12,6 +12,7 @@ from typing import TYPE_CHECKING, Any, Optional, TypeVar
12
12
  from grpclib import GRPCError, Status
13
13
  from synchronicity.async_wrap import asynccontextmanager
14
14
 
15
+ import modal._runtime.execution_context
15
16
  import modal_proto.api_pb2
16
17
  from modal_proto import api_pb2
17
18
 
@@ -19,7 +20,6 @@ from ._functions import _Function
19
20
  from ._object import _get_environment_name, _Object
20
21
  from ._pty import get_pty_info
21
22
  from ._resolver import Resolver
22
- from ._runtime.execution_context import is_local
23
23
  from ._traceback import print_server_warnings, traceback_contains_remote_call
24
24
  from ._utils.async_utils import TaskContext, gather_cancel_on_exc, synchronize_api
25
25
  from ._utils.deprecation import deprecation_error
@@ -266,12 +266,9 @@ async def _run_app(
266
266
  if environment_name is None:
267
267
  environment_name = typing.cast(str, config.get("environment"))
268
268
 
269
- if not is_local():
270
- raise InvalidError(
271
- "Can not run an app from within a container."
272
- " Are you calling app.run() directly?"
273
- " Consider using the `modal run` shell command."
274
- )
269
+ if modal._runtime.execution_context._is_currently_importing:
270
+ raise InvalidError("Can not run an app in global scope within a container")
271
+
275
272
  if app._running_app:
276
273
  raise InvalidError(
277
274
  "App is already running and can't be started again.\n"
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: modal
3
- Version: 0.73.164
3
+ Version: 0.73.165
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -2,7 +2,7 @@ modal/__init__.py,sha256=7wz1AT_bpWJJEzXsAo3QMb7i87y7UGXwfneb0bGDhRg,2502
2
2
  modal/__main__.py,sha256=CgIjP8m1xJjjd4AXc-delmR6LdBCZclw2A_V38CFIio,2870
3
3
  modal/_clustered_functions.py,sha256=kTf-9YBXY88NutC1akI-gCbvf01RhMPCw-zoOI_YIUE,2700
4
4
  modal/_clustered_functions.pyi,sha256=vllkegc99A0jrUOWa8mdlSbdp6uz36TsHhGxysAOpaQ,771
5
- modal/_container_entrypoint.py,sha256=ueRlANz7w9dgBlkuP9o3iFGxU5dAnx-IRAu2JMYaq1c,28997
5
+ modal/_container_entrypoint.py,sha256=85KptUAmlftjHR-mpl32H5deATqqDrRWp0Z1XrKmWyk,29166
6
6
  modal/_functions.py,sha256=M94gzMA9xfW9086djoG2yYFVihcslKnsleacmNbVrG0,74996
7
7
  modal/_ipython.py,sha256=TW1fkVOmZL3YYqdS2YlM1hqpf654Yf8ZyybHdBnlhSw,301
8
8
  modal/_location.py,sha256=joiX-0ZeutEUDTrrqLF1GHXCdVLF-rHzstocbMcd_-k,366
@@ -23,7 +23,7 @@ modal/app.py,sha256=bJp7W3liuVG2VwWkG31tMFogDh84EKppzP8YJFWl3eQ,48140
23
23
  modal/app.pyi,sha256=SkqXNrdnGIZ4MmNNvpGtzNLoUdyuvi9IjQQR_DRiRHk,26968
24
24
  modal/call_graph.py,sha256=1g2DGcMIJvRy-xKicuf63IVE98gJSnQsr8R_NVMptNc,2581
25
25
  modal/client.py,sha256=U-YKSw0n7J1ZLREt9cbEJCtmHe5YoPKFxl0xlkan2yc,15565
26
- modal/client.pyi,sha256=DBHwH_dRIRw4etokK1g9IRaFv1a6lDKHIglO74F01Go,7661
26
+ modal/client.pyi,sha256=ezGzumCVjsThwi36KrD6sOkRHXCKfqVsVC1T1nS8PbY,7661
27
27
  modal/cloud_bucket_mount.py,sha256=YOe9nnvSr4ZbeCn587d7_VhE9IioZYRvF9VYQTQux08,5914
28
28
  modal/cloud_bucket_mount.pyi,sha256=30T3K1a89l6wzmEJ_J9iWv9SknoGqaZDx59Xs-ZQcmk,1607
29
29
  modal/cls.py,sha256=8tvSw7QFTS1FnX2MXaxagu3KwuR6y_DMwhqHv3MZ0Nk,32963
@@ -40,13 +40,13 @@ modal/file_io.py,sha256=lcMs_E9Xfm0YX1t9U2wNIBPnqHRxmImqjLW1GHqVmyg,20945
40
40
  modal/file_io.pyi,sha256=NTRft1tbPSWf9TlWVeZmTlgB5AZ_Zhu2srWIrWr7brk,9445
41
41
  modal/file_pattern_matcher.py,sha256=trosX-Bp7dOubudN1bLLhRAoidWy1TcoaR4Pv8CedWw,6497
42
42
  modal/functions.py,sha256=kcNHvqeGBxPI7Cgd57NIBBghkfbeFJzXO44WW0jSmao,325
43
- modal/functions.pyi,sha256=Bg_zlYSHMYuEXQYLcFU-j1Kwq5zHQ9jggMA_5ZO_du8,14785
43
+ modal/functions.pyi,sha256=m1PL2pwO-lnGV0uZDVCmzZ_v7Mu8ISRtxmxS15aEIAQ,14785
44
44
  modal/gpu.py,sha256=Kbhs_u49FaC2Zi0TjCdrpstpRtT5eZgecynmQi5IZVE,6752
45
45
  modal/image.py,sha256=qagLzHPbeEoM7TqvIuFJ8JV_IEJtRqiR0lik4O_K4Gw,92857
46
46
  modal/image.pyi,sha256=89zv12C1sFrJs7Es9SnX23_m208ASAdeNGCVTrhjzHI,25632
47
47
  modal/io_streams.py,sha256=h5O2LmbRoT9l777z3TQhCAm-JF1r7avZ2ykXlejztDs,15163
48
48
  modal/io_streams.pyi,sha256=bJ7ZLmSmJ0nKoa6r4FJpbqvzdUVa0lEe0Fa-MMpMezU,5071
49
- modal/mount.py,sha256=JII0zTS1fPCcCbZgO18okkOuTDqYCxY1DIVa6i1E9cI,32196
49
+ modal/mount.py,sha256=Y2sXIbUbr3ymVvfvknwbouBkkAUPIW1nkSToRGh8w7g,31973
50
50
  modal/mount.pyi,sha256=CmHa7zKSxHA_7-vMQLnGfw_ZXvAvHlafvUEVJcQ1LQA,12535
51
51
  modal/network_file_system.py,sha256=WXdyL7du_fvjvuG6hSAREyJ83sSEP2xSLAIAhBsisdI,14869
52
52
  modal/network_file_system.pyi,sha256=4N3eqMbTSlqmS8VV_aJK-uvrgJC8xnf_YtW5FHfRfc8,8156
@@ -63,7 +63,7 @@ modal/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
63
63
  modal/queue.py,sha256=OIYmve1a4GTP54Vj2CcLatLPIAWToU7hWBNeu7IJiBY,18985
64
64
  modal/queue.pyi,sha256=sgvELCK4bJXMZIZw7gllooGFZNipGjI3BT4rmUuyD9M,10282
65
65
  modal/retries.py,sha256=IvNLDM0f_GLUDD5VgEDoN09C88yoxSrCquinAuxT1Sc,5205
66
- modal/runner.py,sha256=8jBg-gOjjYyteXMI01u6f7izYsGrABsFaByixeQlTmo,25126
66
+ modal/runner.py,sha256=V17Fb9OtTGplvILc4ogT-waHYjxyCnjf0PP4aYy_0ho,25036
67
67
  modal/runner.pyi,sha256=HW2pvC_PLwg1Es_EkrfQgMZsktIr9zzVEtmjOVFG6Dw,5351
68
68
  modal/running_app.py,sha256=v61mapYNV1-O-Uaho5EfJlryMLvIT9We0amUOSvSGx8,1188
69
69
  modal/sandbox.py,sha256=FRcMkVDfp39_ACbhR71iDJcT6xaqFXuCI3LKsLETHPc,32619
@@ -85,8 +85,8 @@ modal/_runtime/__init__.py,sha256=MIEP8jhXUeGq_eCjYFcqN5b1bxBM4fdk0VESpjWR0fc,28
85
85
  modal/_runtime/asgi.py,sha256=KNarxvZI9z8fnmZl2vbkWTjnoLXs9kqOahkrbsTLkyc,22429
86
86
  modal/_runtime/container_io_manager.py,sha256=WZnvKlmHaF7yteHMoeX-jIgU4OEPLhQyAvhNXmjjVy0,44318
87
87
  modal/_runtime/container_io_manager.pyi,sha256=wRd2wHMFru0NmNgiCBVdDTrJGkeVZsZvWwA1fzn8wi8,17009
88
- modal/_runtime/execution_context.py,sha256=E6ofm6j1POXGPxS841X3V7JU6NheVb8OkQc7JpLq4Kg,2712
89
- modal/_runtime/execution_context.pyi,sha256=wQZwMNADExkeNdB9yKX0PPojovxlFHbap3441wAsiMY,634
88
+ modal/_runtime/execution_context.py,sha256=73Y5zH_o-MhVCrkJXakYVlFkKqCa2CWvqoHjOfJrJGg,3034
89
+ modal/_runtime/execution_context.pyi,sha256=TAxQq7uLj7i9r9XbXgFZiSVBWxObFWN-rkssS0I7Vkk,661
90
90
  modal/_runtime/gpu_memory_snapshot.py,sha256=tA3m1d1cwnmHpvpCeN_WijDd6n8byn7LWlpicbIxiOI,3144
91
91
  modal/_runtime/telemetry.py,sha256=T1RoAGyjBDr1swiM6pPsGRSITm7LI5FDK18oNXxY08U,5163
92
92
  modal/_runtime/user_code_imports.py,sha256=kAv37Pl1TmGKduv0Kozum0xNTD42bDLloSIsT7zf84o,16884
@@ -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.73.164.dist-info/licenses/LICENSE,sha256=psuoW8kuDP96RQsdhzwOqi6fyWv0ct8CR6Jr7He_P_k,10173
149
+ modal-0.73.165.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=cvTgltucqYLLIX84QxAwf51Z5Vc2n6cLxS8VcrxNCAo,6401
@@ -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=wiJQ53c-OMs0Xf1UeXOxQ7FwlV1VzIjnX6o-pRYZ_Pk,470
173
173
  modal_version/__main__.py,sha256=2FO0yYQQwDTh6udt1h-cBnGd1c4ZyHnHSI4BksxzVac,105
174
- modal_version/_version_generated.py,sha256=2I0ucIXjzi2tKSflkJUmKq9nN0hIKaLV0cEjBQVxcf0,150
175
- modal-0.73.164.dist-info/METADATA,sha256=QEcE3qTvoJmPOms19OEunLqOXOzHXbG7TA7ZiJy7FNk,2475
176
- modal-0.73.164.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
177
- modal-0.73.164.dist-info/entry_points.txt,sha256=An-wYgeEUnm6xzrAP9_NTSTSciYvvEWsMZILtYrvpAI,46
178
- modal-0.73.164.dist-info/top_level.txt,sha256=4BWzoKYREKUZ5iyPzZpjqx4G8uB5TWxXPDwibLcVa7k,43
179
- modal-0.73.164.dist-info/RECORD,,
174
+ modal_version/_version_generated.py,sha256=hRg9FYMu45Ztb3wBnbqwihpEOxZwsdjeAwcV1SMwKL8,150
175
+ modal-0.73.165.dist-info/METADATA,sha256=nu47OVg9iyCvAYlGLNM3-TRZS8pa3_019VMJ5cyBiVE,2475
176
+ modal-0.73.165.dist-info/WHEEL,sha256=1tXe9gY0PYatrMPMDd6jXqjfpz_B-Wqm32CPfRC58XU,91
177
+ modal-0.73.165.dist-info/entry_points.txt,sha256=An-wYgeEUnm6xzrAP9_NTSTSciYvvEWsMZILtYrvpAI,46
178
+ modal-0.73.165.dist-info/top_level.txt,sha256=4BWzoKYREKUZ5iyPzZpjqx4G8uB5TWxXPDwibLcVa7k,43
179
+ modal-0.73.165.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 = 164 # git: 9d8e6de
4
+ build_number = 165 # git: 063c3c2