modal 1.0.5.dev27__tar.gz → 1.0.5.dev30__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 (184) hide show
  1. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/PKG-INFO +1 -1
  2. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_serialization.py +25 -2
  3. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/client.pyi +2 -2
  4. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/functions.pyi +6 -6
  5. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/parallel_map.py +42 -3
  6. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/parallel_map.pyi +5 -0
  7. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/PKG-INFO +1 -1
  8. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api.proto +15 -1
  9. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api_grpc.py +16 -0
  10. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api_pb2.py +624 -593
  11. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api_pb2.pyi +48 -0
  12. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api_pb2_grpc.py +33 -0
  13. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/api_pb2_grpc.pyi +10 -0
  14. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/modal_api_grpc.py +1 -0
  15. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_version/__init__.py +1 -1
  16. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/LICENSE +0 -0
  17. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/README.md +0 -0
  18. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/__init__.py +0 -0
  19. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/__main__.py +0 -0
  20. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_clustered_functions.py +0 -0
  21. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_clustered_functions.pyi +0 -0
  22. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_container_entrypoint.py +0 -0
  23. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_functions.py +0 -0
  24. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_ipython.py +0 -0
  25. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_location.py +0 -0
  26. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_object.py +0 -0
  27. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_output.py +0 -0
  28. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_partial_function.py +0 -0
  29. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_pty.py +0 -0
  30. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_resolver.py +0 -0
  31. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_resources.py +0 -0
  32. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/__init__.py +0 -0
  33. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/asgi.py +0 -0
  34. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/container_io_manager.py +0 -0
  35. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/container_io_manager.pyi +0 -0
  36. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/execution_context.py +0 -0
  37. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/execution_context.pyi +0 -0
  38. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/gpu_memory_snapshot.py +0 -0
  39. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/telemetry.py +0 -0
  40. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_runtime/user_code_imports.py +0 -0
  41. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_traceback.py +0 -0
  42. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_tunnel.py +0 -0
  43. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_tunnel.pyi +0 -0
  44. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_type_manager.py +0 -0
  45. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/__init__.py +0 -0
  46. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/app_utils.py +0 -0
  47. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/async_utils.py +0 -0
  48. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/blob_utils.py +0 -0
  49. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/bytes_io_segment_payload.py +0 -0
  50. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/deprecation.py +0 -0
  51. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/docker_utils.py +0 -0
  52. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/function_utils.py +0 -0
  53. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/git_utils.py +0 -0
  54. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/grpc_testing.py +0 -0
  55. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/grpc_utils.py +0 -0
  56. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/hash_utils.py +0 -0
  57. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/http_utils.py +0 -0
  58. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/jwt_utils.py +0 -0
  59. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/logger.py +0 -0
  60. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/mount_utils.py +0 -0
  61. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/name_utils.py +0 -0
  62. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/package_utils.py +0 -0
  63. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/pattern_utils.py +0 -0
  64. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/rand_pb_testing.py +0 -0
  65. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/shell_utils.py +0 -0
  66. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_utils/time_utils.py +0 -0
  67. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_vendor/__init__.py +0 -0
  68. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_vendor/a2wsgi_wsgi.py +0 -0
  69. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_vendor/cloudpickle.py +0 -0
  70. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_vendor/tblib.py +0 -0
  71. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/_watcher.py +0 -0
  72. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/app.py +0 -0
  73. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/app.pyi +0 -0
  74. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/call_graph.py +0 -0
  75. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/__init__.py +0 -0
  76. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/_download.py +0 -0
  77. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/_traceback.py +0 -0
  78. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/app.py +0 -0
  79. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/cluster.py +0 -0
  80. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/config.py +0 -0
  81. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/container.py +0 -0
  82. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/dict.py +0 -0
  83. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/entry_point.py +0 -0
  84. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/environment.py +0 -0
  85. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/import_refs.py +0 -0
  86. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/launch.py +0 -0
  87. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/network_file_system.py +0 -0
  88. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/profile.py +0 -0
  89. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/programs/__init__.py +0 -0
  90. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/programs/run_jupyter.py +0 -0
  91. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/programs/vscode.py +0 -0
  92. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/queues.py +0 -0
  93. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/run.py +0 -0
  94. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/secret.py +0 -0
  95. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/token.py +0 -0
  96. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/utils.py +0 -0
  97. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cli/volume.py +0 -0
  98. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/client.py +0 -0
  99. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cloud_bucket_mount.py +0 -0
  100. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cloud_bucket_mount.pyi +0 -0
  101. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cls.py +0 -0
  102. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/cls.pyi +0 -0
  103. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/config.py +0 -0
  104. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/container_process.py +0 -0
  105. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/container_process.pyi +0 -0
  106. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/dict.py +0 -0
  107. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/dict.pyi +0 -0
  108. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/environments.py +0 -0
  109. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/environments.pyi +0 -0
  110. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/exception.py +0 -0
  111. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/experimental/__init__.py +0 -0
  112. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/experimental/ipython.py +0 -0
  113. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/file_io.py +0 -0
  114. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/file_io.pyi +0 -0
  115. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/file_pattern_matcher.py +0 -0
  116. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/functions.py +0 -0
  117. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/gpu.py +0 -0
  118. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/image.py +0 -0
  119. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/image.pyi +0 -0
  120. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/io_streams.py +0 -0
  121. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/io_streams.pyi +0 -0
  122. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/mount.py +0 -0
  123. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/mount.pyi +0 -0
  124. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/network_file_system.py +0 -0
  125. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/network_file_system.pyi +0 -0
  126. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/object.py +0 -0
  127. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/object.pyi +0 -0
  128. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/output.py +0 -0
  129. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/partial_function.py +0 -0
  130. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/partial_function.pyi +0 -0
  131. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/proxy.py +0 -0
  132. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/proxy.pyi +0 -0
  133. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/py.typed +0 -0
  134. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/queue.py +0 -0
  135. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/queue.pyi +0 -0
  136. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/2023.12.312.txt +0 -0
  137. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/2023.12.txt +0 -0
  138. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/2024.04.txt +0 -0
  139. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/2024.10.txt +0 -0
  140. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/PREVIEW.txt +0 -0
  141. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/README.md +0 -0
  142. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/requirements/base-images.json +0 -0
  143. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/retries.py +0 -0
  144. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/runner.py +0 -0
  145. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/runner.pyi +0 -0
  146. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/running_app.py +0 -0
  147. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/sandbox.py +0 -0
  148. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/sandbox.pyi +0 -0
  149. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/schedule.py +0 -0
  150. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/scheduler_placement.py +0 -0
  151. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/secret.py +0 -0
  152. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/secret.pyi +0 -0
  153. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/serving.py +0 -0
  154. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/serving.pyi +0 -0
  155. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/snapshot.py +0 -0
  156. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/snapshot.pyi +0 -0
  157. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/stream_type.py +0 -0
  158. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/token_flow.py +0 -0
  159. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/token_flow.pyi +0 -0
  160. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/volume.py +0 -0
  161. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal/volume.pyi +0 -0
  162. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/SOURCES.txt +0 -0
  163. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/dependency_links.txt +0 -0
  164. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/entry_points.txt +0 -0
  165. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/requires.txt +0 -0
  166. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal.egg-info/top_level.txt +0 -0
  167. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/__init__.py +0 -0
  168. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/gen_cli_docs.py +0 -0
  169. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/gen_reference_docs.py +0 -0
  170. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/mdmd/__init__.py +0 -0
  171. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/mdmd/mdmd.py +0 -0
  172. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_docs/mdmd/signatures.py +0 -0
  173. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/__init__.py +0 -0
  174. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/modal_options_grpc.py +0 -0
  175. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options.proto +0 -0
  176. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options_grpc.py +0 -0
  177. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options_pb2.py +0 -0
  178. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options_pb2.pyi +0 -0
  179. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options_pb2_grpc.py +0 -0
  180. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/options_pb2_grpc.pyi +0 -0
  181. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_proto/py.typed +0 -0
  182. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/modal_version/__main__.py +0 -0
  183. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/pyproject.toml +0 -0
  184. {modal-1.0.5.dev27 → modal-1.0.5.dev30}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: modal
3
- Version: 1.0.5.dev27
3
+ Version: 1.0.5.dev30
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -6,6 +6,8 @@ import typing
6
6
  from inspect import Parameter
7
7
  from typing import Any
8
8
 
9
+ import google.protobuf.message
10
+
9
11
  from modal._utils.async_utils import synchronizer
10
12
  from modal_proto import api_pb2
11
13
 
@@ -470,10 +472,31 @@ def deserialize_params(serialized_params: bytes, function_def: api_pb2.Function,
470
472
  api_pb2.ClassParameterInfo.PARAM_SERIALIZATION_FORMAT_PICKLE,
471
473
  ):
472
474
  # legacy serialization format - pickle of `(args, kwargs)` w/ support for modal object arguments
473
- param_args, param_kwargs = deserialize(serialized_params, _client)
475
+ try:
476
+ param_args, param_kwargs = deserialize(serialized_params, _client)
477
+ except DeserializationError as original_exc:
478
+ # Fallback in case of proto -> pickle downgrades of a parameter serialization format
479
+ # I.e. FunctionBindParams binding proto serialized params to a function defintion
480
+ # that now assumes pickled data according to class_parameter_info
481
+ param_args = ()
482
+ try:
483
+ param_kwargs = deserialize_proto_params(serialized_params)
484
+ except Exception:
485
+ raise original_exc
486
+
474
487
  elif function_def.class_parameter_info.format == api_pb2.ClassParameterInfo.PARAM_SERIALIZATION_FORMAT_PROTO:
475
488
  param_args = () # we use kwargs only for our implicit constructors
476
- param_kwargs = deserialize_proto_params(serialized_params)
489
+ try:
490
+ param_kwargs = deserialize_proto_params(serialized_params)
491
+ except google.protobuf.message.DecodeError as original_exc:
492
+ # Fallback in case of pickle -> proto upgrades of a parameter serialization format
493
+ # I.e. FunctionBindParams binding pickle serialized params to a function defintion
494
+ # that now assumes proto data according to class_parameter_info
495
+ try:
496
+ param_args, param_kwargs = deserialize(serialized_params, _client)
497
+ except Exception:
498
+ raise original_exc
499
+
477
500
  else:
478
501
  raise ExecutionError(
479
502
  f"Unknown class parameter serialization format: {function_def.class_parameter_info.format}"
@@ -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 = "1.0.5.dev27",
34
+ version: str = "1.0.5.dev30",
35
35
  ):
36
36
  """mdmd:hidden
37
37
  The Modal client object is not intended to be instantiated directly by users.
@@ -160,7 +160,7 @@ class Client:
160
160
  server_url: str,
161
161
  client_type: int,
162
162
  credentials: typing.Optional[tuple[str, str]],
163
- version: str = "1.0.5.dev27",
163
+ version: str = "1.0.5.dev30",
164
164
  ):
165
165
  """mdmd:hidden
166
166
  The Modal client object is not intended to be instantiated directly by users.
@@ -428,7 +428,7 @@ class Function(
428
428
 
429
429
  _call_generator: ___call_generator_spec[typing_extensions.Self]
430
430
 
431
- class __remote_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
431
+ class __remote_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
432
432
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> ReturnType_INNER:
433
433
  """Calls the function remotely, executing it with the given arguments and returning the execution's result."""
434
434
  ...
@@ -437,7 +437,7 @@ class Function(
437
437
  """Calls the function remotely, executing it with the given arguments and returning the execution's result."""
438
438
  ...
439
439
 
440
- remote: __remote_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
440
+ remote: __remote_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
441
441
 
442
442
  class __remote_gen_spec(typing_extensions.Protocol[SUPERSELF]):
443
443
  def __call__(self, /, *args, **kwargs) -> typing.Generator[typing.Any, None, None]:
@@ -464,7 +464,7 @@ class Function(
464
464
  """
465
465
  ...
466
466
 
467
- class ___experimental_spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
467
+ class ___experimental_spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
468
468
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]:
469
469
  """[Experimental] Calls the function with the given arguments, without waiting for the results.
470
470
 
@@ -488,7 +488,7 @@ class Function(
488
488
  ...
489
489
 
490
490
  _experimental_spawn: ___experimental_spawn_spec[
491
- modal._functions.P, modal._functions.ReturnType, typing_extensions.Self
491
+ modal._functions.ReturnType, modal._functions.P, typing_extensions.Self
492
492
  ]
493
493
 
494
494
  class ___spawn_map_inner_spec(typing_extensions.Protocol[P_INNER, SUPERSELF]):
@@ -497,7 +497,7 @@ class Function(
497
497
 
498
498
  _spawn_map_inner: ___spawn_map_inner_spec[modal._functions.P, typing_extensions.Self]
499
499
 
500
- class __spawn_spec(typing_extensions.Protocol[P_INNER, ReturnType_INNER, SUPERSELF]):
500
+ class __spawn_spec(typing_extensions.Protocol[ReturnType_INNER, P_INNER, SUPERSELF]):
501
501
  def __call__(self, /, *args: P_INNER.args, **kwargs: P_INNER.kwargs) -> FunctionCall[ReturnType_INNER]:
502
502
  """Calls the function with the given arguments, without waiting for the results.
503
503
 
@@ -518,7 +518,7 @@ class Function(
518
518
  """
519
519
  ...
520
520
 
521
- spawn: __spawn_spec[modal._functions.P, modal._functions.ReturnType, typing_extensions.Self]
521
+ spawn: __spawn_spec[modal._functions.ReturnType, modal._functions.P, typing_extensions.Self]
522
522
 
523
523
  def get_raw_f(self) -> collections.abc.Callable[..., typing.Any]:
524
524
  """Return the inner Python object wrapped by this Modal Function."""
@@ -1,6 +1,7 @@
1
1
  # Copyright Modal Labs 2024
2
2
  import asyncio
3
3
  import enum
4
+ import inspect
4
5
  import time
5
6
  import typing
6
7
  from asyncio import FIRST_COMPLETED
@@ -27,6 +28,7 @@ from modal._utils.async_utils import (
27
28
  warn_if_generator_is_not_consumed,
28
29
  )
29
30
  from modal._utils.blob_utils import BLOB_MAX_PARALLELISM
31
+ from modal._utils.deprecation import deprecation_warning
30
32
  from modal._utils.function_utils import (
31
33
  ATTEMPT_TIMEOUT_GRACE_PERIOD,
32
34
  OUTPUTS_TIMEOUT,
@@ -409,7 +411,7 @@ async def _map_invocation(
409
411
  async_merge(drain_input_generator(), pump_inputs(), poll_outputs(), retry_inputs())
410
412
  ) as streamer:
411
413
  async for response in streamer:
412
- if response is not None:
414
+ if response is not None: # type: ignore[unreachable]
413
415
  yield response.value
414
416
  log_debug_stats_task.cancel()
415
417
  await log_debug_stats_task
@@ -434,7 +436,6 @@ async def _map_helper(
434
436
  We could make this explicit as an improvement or even let users decide what they
435
437
  prefer: throughput (prioritize queueing inputs) or latency (prioritize yielding results)
436
438
  """
437
-
438
439
  raw_input_queue: Any = SynchronizedQueue() # type: ignore
439
440
  await raw_input_queue.init.aio()
440
441
 
@@ -460,6 +461,33 @@ async def _map_helper(
460
461
  yield output
461
462
 
462
463
 
464
+ def _maybe_warn_about_exceptions(func_name: str, return_exceptions: bool, wrap_returned_exceptions: bool):
465
+ if return_exceptions and wrap_returned_exceptions:
466
+ deprecation_warning(
467
+ (2025, 6, 27),
468
+ (
469
+ f"Function.{func_name} currently leaks an internal exception wrapping type "
470
+ "(modal.exceptions.UserCodeException) when `return_exceptions=True` is set. "
471
+ "In the future, this will change, and the underlying exception will be returned directly.\n"
472
+ "To opt into the future behavior and silence this warning, add `wrap_returned_exceptions=False`:\n\n"
473
+ f" f.{func_name}(..., return_exceptions=True, wrap_returned_exceptions=False)"
474
+ ),
475
+ )
476
+
477
+
478
+ def _invoked_from_sync_wrapper() -> bool:
479
+ """Check whether the calling function was called from a sync wrapper."""
480
+ # This is temporary: we only need it to avoind double-firing the wrap_returned_exceptions warning.
481
+ # (We don't want to push the warning lower in the stack beacuse then we can't attribute to the user's code.)
482
+ try:
483
+ frame = inspect.currentframe()
484
+ caller_function_name = frame.f_back.f_back.f_code.co_name
485
+ # Embeds some assumptions about how the current calling stack works, but this is just temporary.
486
+ return caller_function_name == "asend"
487
+ except Exception:
488
+ return False
489
+
490
+
463
491
  @warn_if_generator_is_not_consumed(function_name="Function.map.aio")
464
492
  async def _map_async(
465
493
  self: "modal.functions.Function",
@@ -471,6 +499,8 @@ async def _map_async(
471
499
  return_exceptions: bool = False, # propagate exceptions (False) or aggregate them in the results list (True)
472
500
  wrap_returned_exceptions: bool = True, # wrap returned exceptions in modal.exception.UserCodeException
473
501
  ) -> typing.AsyncGenerator[Any, None]:
502
+ if not _invoked_from_sync_wrapper():
503
+ _maybe_warn_about_exceptions("map.aio", return_exceptions, wrap_returned_exceptions)
474
504
  async_input_gen = async_zip(*[sync_or_async_iter(it) for it in input_iterators])
475
505
  async for output in _map_helper(
476
506
  self,
@@ -493,6 +523,8 @@ async def _starmap_async(
493
523
  return_exceptions: bool = False,
494
524
  wrap_returned_exceptions: bool = True,
495
525
  ) -> typing.AsyncIterable[Any]:
526
+ if not _invoked_from_sync_wrapper():
527
+ _maybe_warn_about_exceptions("starmap.aio", return_exceptions, wrap_returned_exceptions)
496
528
  async for output in _map_helper(
497
529
  self,
498
530
  sync_or_async_iter(input_iterator),
@@ -509,7 +541,12 @@ async def _for_each_async(self, *input_iterators, kwargs={}, ignore_exceptions:
509
541
  # rather than iterating over the result
510
542
  async_input_gen = async_zip(*[sync_or_async_iter(it) for it in input_iterators])
511
543
  async for _ in _map_helper(
512
- self, async_input_gen, kwargs=kwargs, order_outputs=False, return_exceptions=ignore_exceptions
544
+ self,
545
+ async_input_gen,
546
+ kwargs=kwargs,
547
+ order_outputs=False,
548
+ return_exceptions=ignore_exceptions,
549
+ wrap_returned_exceptions=False,
513
550
  ):
514
551
  pass
515
552
 
@@ -559,6 +596,7 @@ def _map_sync(
559
596
  print(list(my_func.map(range(3), return_exceptions=True)))
560
597
  ```
561
598
  """
599
+ _maybe_warn_about_exceptions("map", return_exceptions, wrap_returned_exceptions)
562
600
 
563
601
  return AsyncOrSyncIterable(
564
602
  _map_async(
@@ -665,6 +703,7 @@ def _starmap_sync(
665
703
  assert list(my_func.starmap([(1, 2), (3, 4)])) == [3, 7]
666
704
  ```
667
705
  """
706
+ _maybe_warn_about_exceptions("starmap", return_exceptions, wrap_returned_exceptions)
668
707
  return AsyncOrSyncIterable(
669
708
  _starmap_async(
670
709
  self,
@@ -91,6 +91,11 @@ def _map_helper(
91
91
  """
92
92
  ...
93
93
 
94
+ def _maybe_warn_about_exceptions(func_name: str, return_exceptions: bool, wrap_returned_exceptions: bool): ...
95
+ def _invoked_from_sync_wrapper() -> bool:
96
+ """Check whether the calling function was called from a sync wrapper."""
97
+ ...
98
+
94
99
  def _map_async(
95
100
  self: modal.functions.Function,
96
101
  *input_iterators: typing.Union[typing.Iterable[typing.Any], typing.AsyncIterable[typing.Any]],
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: modal
3
- Version: 1.0.5.dev27
3
+ Version: 1.0.5.dev30
4
4
  Summary: Python client library for Modal
5
5
  Author-email: Modal Labs <support@modal.com>
6
6
  License: Apache-2.0
@@ -1246,8 +1246,21 @@ message FlashContainerDeregisterRequest {
1246
1246
  string service_name = 1;
1247
1247
  }
1248
1248
 
1249
+ message FlashContainerListRequest {
1250
+ string function_id = 1;
1251
+ }
1252
+
1253
+ message FlashContainerListResponse {
1254
+ message Container {
1255
+ string task_id = 1;
1256
+ string host = 2;
1257
+ uint32 port = 3;
1258
+ }
1259
+ repeated Container containers = 1;
1260
+ }
1261
+
1249
1262
  message FlashContainerRegisterRequest {
1250
- string service_name = 1;
1263
+ string service_name = 1; // not used?
1251
1264
  uint32 priority = 2;
1252
1265
  uint32 weight = 3;
1253
1266
  string host = 4;
@@ -3294,6 +3307,7 @@ service ModalClient {
3294
3307
 
3295
3308
  // Modal Flash (experimental)
3296
3309
  rpc FlashContainerDeregister(FlashContainerDeregisterRequest) returns (google.protobuf.Empty);
3310
+ rpc FlashContainerList(FlashContainerListRequest) returns (FlashContainerListResponse);
3297
3311
  rpc FlashContainerRegister(FlashContainerRegisterRequest) returns (FlashContainerRegisterResponse);
3298
3312
 
3299
3313
  // Functions
@@ -246,6 +246,10 @@ class ModalClientBase(abc.ABC):
246
246
  async def FlashContainerDeregister(self, stream: 'grpclib.server.Stream[modal_proto.api_pb2.FlashContainerDeregisterRequest, google.protobuf.empty_pb2.Empty]') -> None:
247
247
  pass
248
248
 
249
+ @abc.abstractmethod
250
+ async def FlashContainerList(self, stream: 'grpclib.server.Stream[modal_proto.api_pb2.FlashContainerListRequest, modal_proto.api_pb2.FlashContainerListResponse]') -> None:
251
+ pass
252
+
249
253
  @abc.abstractmethod
250
254
  async def FlashContainerRegister(self, stream: 'grpclib.server.Stream[modal_proto.api_pb2.FlashContainerRegisterRequest, modal_proto.api_pb2.FlashContainerRegisterResponse]') -> None:
251
255
  pass
@@ -982,6 +986,12 @@ class ModalClientBase(abc.ABC):
982
986
  modal_proto.api_pb2.FlashContainerDeregisterRequest,
983
987
  google.protobuf.empty_pb2.Empty,
984
988
  ),
989
+ '/modal.client.ModalClient/FlashContainerList': grpclib.const.Handler(
990
+ self.FlashContainerList,
991
+ grpclib.const.Cardinality.UNARY_UNARY,
992
+ modal_proto.api_pb2.FlashContainerListRequest,
993
+ modal_proto.api_pb2.FlashContainerListResponse,
994
+ ),
985
995
  '/modal.client.ModalClient/FlashContainerRegister': grpclib.const.Handler(
986
996
  self.FlashContainerRegister,
987
997
  grpclib.const.Cardinality.UNARY_UNARY,
@@ -1918,6 +1928,12 @@ class ModalClientStub:
1918
1928
  modal_proto.api_pb2.FlashContainerDeregisterRequest,
1919
1929
  google.protobuf.empty_pb2.Empty,
1920
1930
  )
1931
+ self.FlashContainerList = grpclib.client.UnaryUnaryMethod(
1932
+ channel,
1933
+ '/modal.client.ModalClient/FlashContainerList',
1934
+ modal_proto.api_pb2.FlashContainerListRequest,
1935
+ modal_proto.api_pb2.FlashContainerListResponse,
1936
+ )
1921
1937
  self.FlashContainerRegister = grpclib.client.UnaryUnaryMethod(
1922
1938
  channel,
1923
1939
  '/modal.client.ModalClient/FlashContainerRegister',