ob-metaflow 2.11.13.1__py2.py3-none-any.whl → 2.19.7.1rc0__py2.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.
Files changed (289) hide show
  1. metaflow/R.py +10 -7
  2. metaflow/__init__.py +40 -25
  3. metaflow/_vendor/imghdr/__init__.py +186 -0
  4. metaflow/_vendor/importlib_metadata/__init__.py +1063 -0
  5. metaflow/_vendor/importlib_metadata/_adapters.py +68 -0
  6. metaflow/_vendor/importlib_metadata/_collections.py +30 -0
  7. metaflow/_vendor/importlib_metadata/_compat.py +71 -0
  8. metaflow/_vendor/importlib_metadata/_functools.py +104 -0
  9. metaflow/_vendor/importlib_metadata/_itertools.py +73 -0
  10. metaflow/_vendor/importlib_metadata/_meta.py +48 -0
  11. metaflow/_vendor/importlib_metadata/_text.py +99 -0
  12. metaflow/_vendor/importlib_metadata/py.typed +0 -0
  13. metaflow/_vendor/typeguard/__init__.py +48 -0
  14. metaflow/_vendor/typeguard/_checkers.py +1070 -0
  15. metaflow/_vendor/typeguard/_config.py +108 -0
  16. metaflow/_vendor/typeguard/_decorators.py +233 -0
  17. metaflow/_vendor/typeguard/_exceptions.py +42 -0
  18. metaflow/_vendor/typeguard/_functions.py +308 -0
  19. metaflow/_vendor/typeguard/_importhook.py +213 -0
  20. metaflow/_vendor/typeguard/_memo.py +48 -0
  21. metaflow/_vendor/typeguard/_pytest_plugin.py +127 -0
  22. metaflow/_vendor/typeguard/_suppression.py +86 -0
  23. metaflow/_vendor/typeguard/_transformer.py +1229 -0
  24. metaflow/_vendor/typeguard/_union_transformer.py +55 -0
  25. metaflow/_vendor/typeguard/_utils.py +173 -0
  26. metaflow/_vendor/typeguard/py.typed +0 -0
  27. metaflow/_vendor/typing_extensions.py +3641 -0
  28. metaflow/_vendor/v3_7/importlib_metadata/__init__.py +1063 -0
  29. metaflow/_vendor/v3_7/importlib_metadata/_adapters.py +68 -0
  30. metaflow/_vendor/v3_7/importlib_metadata/_collections.py +30 -0
  31. metaflow/_vendor/v3_7/importlib_metadata/_compat.py +71 -0
  32. metaflow/_vendor/v3_7/importlib_metadata/_functools.py +104 -0
  33. metaflow/_vendor/v3_7/importlib_metadata/_itertools.py +73 -0
  34. metaflow/_vendor/v3_7/importlib_metadata/_meta.py +48 -0
  35. metaflow/_vendor/v3_7/importlib_metadata/_text.py +99 -0
  36. metaflow/_vendor/v3_7/importlib_metadata/py.typed +0 -0
  37. metaflow/_vendor/v3_7/typeguard/__init__.py +48 -0
  38. metaflow/_vendor/v3_7/typeguard/_checkers.py +906 -0
  39. metaflow/_vendor/v3_7/typeguard/_config.py +108 -0
  40. metaflow/_vendor/v3_7/typeguard/_decorators.py +237 -0
  41. metaflow/_vendor/v3_7/typeguard/_exceptions.py +42 -0
  42. metaflow/_vendor/v3_7/typeguard/_functions.py +310 -0
  43. metaflow/_vendor/v3_7/typeguard/_importhook.py +213 -0
  44. metaflow/_vendor/v3_7/typeguard/_memo.py +48 -0
  45. metaflow/_vendor/v3_7/typeguard/_pytest_plugin.py +100 -0
  46. metaflow/_vendor/v3_7/typeguard/_suppression.py +88 -0
  47. metaflow/_vendor/v3_7/typeguard/_transformer.py +1207 -0
  48. metaflow/_vendor/v3_7/typeguard/_union_transformer.py +54 -0
  49. metaflow/_vendor/v3_7/typeguard/_utils.py +169 -0
  50. metaflow/_vendor/v3_7/typeguard/py.typed +0 -0
  51. metaflow/_vendor/v3_7/typing_extensions.py +3072 -0
  52. metaflow/_vendor/yaml/__init__.py +427 -0
  53. metaflow/_vendor/yaml/composer.py +139 -0
  54. metaflow/_vendor/yaml/constructor.py +748 -0
  55. metaflow/_vendor/yaml/cyaml.py +101 -0
  56. metaflow/_vendor/yaml/dumper.py +62 -0
  57. metaflow/_vendor/yaml/emitter.py +1137 -0
  58. metaflow/_vendor/yaml/error.py +75 -0
  59. metaflow/_vendor/yaml/events.py +86 -0
  60. metaflow/_vendor/yaml/loader.py +63 -0
  61. metaflow/_vendor/yaml/nodes.py +49 -0
  62. metaflow/_vendor/yaml/parser.py +589 -0
  63. metaflow/_vendor/yaml/reader.py +185 -0
  64. metaflow/_vendor/yaml/representer.py +389 -0
  65. metaflow/_vendor/yaml/resolver.py +227 -0
  66. metaflow/_vendor/yaml/scanner.py +1435 -0
  67. metaflow/_vendor/yaml/serializer.py +111 -0
  68. metaflow/_vendor/yaml/tokens.py +104 -0
  69. metaflow/cards.py +5 -0
  70. metaflow/cli.py +331 -785
  71. metaflow/cli_args.py +17 -0
  72. metaflow/cli_components/__init__.py +0 -0
  73. metaflow/cli_components/dump_cmd.py +96 -0
  74. metaflow/cli_components/init_cmd.py +52 -0
  75. metaflow/cli_components/run_cmds.py +546 -0
  76. metaflow/cli_components/step_cmd.py +334 -0
  77. metaflow/cli_components/utils.py +140 -0
  78. metaflow/client/__init__.py +1 -0
  79. metaflow/client/core.py +467 -73
  80. metaflow/client/filecache.py +75 -35
  81. metaflow/clone_util.py +7 -1
  82. metaflow/cmd/code/__init__.py +231 -0
  83. metaflow/cmd/develop/stub_generator.py +756 -288
  84. metaflow/cmd/develop/stubs.py +12 -28
  85. metaflow/cmd/main_cli.py +6 -4
  86. metaflow/cmd/make_wrapper.py +78 -0
  87. metaflow/datastore/__init__.py +1 -0
  88. metaflow/datastore/content_addressed_store.py +41 -10
  89. metaflow/datastore/datastore_set.py +11 -2
  90. metaflow/datastore/flow_datastore.py +156 -10
  91. metaflow/datastore/spin_datastore.py +91 -0
  92. metaflow/datastore/task_datastore.py +154 -39
  93. metaflow/debug.py +5 -0
  94. metaflow/decorators.py +404 -78
  95. metaflow/exception.py +8 -2
  96. metaflow/extension_support/__init__.py +527 -376
  97. metaflow/extension_support/_empty_file.py +2 -2
  98. metaflow/extension_support/plugins.py +49 -31
  99. metaflow/flowspec.py +482 -33
  100. metaflow/graph.py +210 -42
  101. metaflow/includefile.py +84 -40
  102. metaflow/lint.py +141 -22
  103. metaflow/meta_files.py +13 -0
  104. metaflow/{metadata → metadata_provider}/heartbeat.py +24 -8
  105. metaflow/{metadata → metadata_provider}/metadata.py +86 -1
  106. metaflow/metaflow_config.py +175 -28
  107. metaflow/metaflow_config_funcs.py +51 -3
  108. metaflow/metaflow_current.py +4 -10
  109. metaflow/metaflow_environment.py +139 -53
  110. metaflow/metaflow_git.py +115 -0
  111. metaflow/metaflow_profile.py +18 -0
  112. metaflow/metaflow_version.py +150 -66
  113. metaflow/mflog/__init__.py +4 -3
  114. metaflow/mflog/save_logs.py +2 -2
  115. metaflow/multicore_utils.py +31 -14
  116. metaflow/package/__init__.py +673 -0
  117. metaflow/packaging_sys/__init__.py +880 -0
  118. metaflow/packaging_sys/backend.py +128 -0
  119. metaflow/packaging_sys/distribution_support.py +153 -0
  120. metaflow/packaging_sys/tar_backend.py +99 -0
  121. metaflow/packaging_sys/utils.py +54 -0
  122. metaflow/packaging_sys/v1.py +527 -0
  123. metaflow/parameters.py +149 -28
  124. metaflow/plugins/__init__.py +74 -5
  125. metaflow/plugins/airflow/airflow.py +40 -25
  126. metaflow/plugins/airflow/airflow_cli.py +22 -5
  127. metaflow/plugins/airflow/airflow_decorator.py +1 -1
  128. metaflow/plugins/airflow/airflow_utils.py +5 -3
  129. metaflow/plugins/airflow/sensors/base_sensor.py +4 -4
  130. metaflow/plugins/airflow/sensors/external_task_sensor.py +2 -2
  131. metaflow/plugins/airflow/sensors/s3_sensor.py +2 -2
  132. metaflow/plugins/argo/argo_client.py +78 -33
  133. metaflow/plugins/argo/argo_events.py +6 -6
  134. metaflow/plugins/argo/argo_workflows.py +2410 -527
  135. metaflow/plugins/argo/argo_workflows_cli.py +571 -121
  136. metaflow/plugins/argo/argo_workflows_decorator.py +43 -12
  137. metaflow/plugins/argo/argo_workflows_deployer.py +106 -0
  138. metaflow/plugins/argo/argo_workflows_deployer_objects.py +453 -0
  139. metaflow/plugins/argo/capture_error.py +73 -0
  140. metaflow/plugins/argo/conditional_input_paths.py +35 -0
  141. metaflow/plugins/argo/exit_hooks.py +209 -0
  142. metaflow/plugins/argo/jobset_input_paths.py +15 -0
  143. metaflow/plugins/argo/param_val.py +19 -0
  144. metaflow/plugins/aws/aws_client.py +10 -3
  145. metaflow/plugins/aws/aws_utils.py +55 -2
  146. metaflow/plugins/aws/batch/batch.py +72 -5
  147. metaflow/plugins/aws/batch/batch_cli.py +33 -10
  148. metaflow/plugins/aws/batch/batch_client.py +4 -3
  149. metaflow/plugins/aws/batch/batch_decorator.py +102 -35
  150. metaflow/plugins/aws/secrets_manager/aws_secrets_manager_secrets_provider.py +13 -10
  151. metaflow/plugins/aws/step_functions/dynamo_db_client.py +0 -3
  152. metaflow/plugins/aws/step_functions/production_token.py +1 -1
  153. metaflow/plugins/aws/step_functions/step_functions.py +65 -8
  154. metaflow/plugins/aws/step_functions/step_functions_cli.py +101 -7
  155. metaflow/plugins/aws/step_functions/step_functions_decorator.py +1 -2
  156. metaflow/plugins/aws/step_functions/step_functions_deployer.py +97 -0
  157. metaflow/plugins/aws/step_functions/step_functions_deployer_objects.py +264 -0
  158. metaflow/plugins/azure/azure_exceptions.py +1 -1
  159. metaflow/plugins/azure/azure_secret_manager_secrets_provider.py +240 -0
  160. metaflow/plugins/azure/azure_tail.py +1 -1
  161. metaflow/plugins/azure/includefile_support.py +2 -0
  162. metaflow/plugins/cards/card_cli.py +66 -30
  163. metaflow/plugins/cards/card_creator.py +25 -1
  164. metaflow/plugins/cards/card_datastore.py +21 -49
  165. metaflow/plugins/cards/card_decorator.py +132 -8
  166. metaflow/plugins/cards/card_modules/basic.py +112 -17
  167. metaflow/plugins/cards/card_modules/bundle.css +1 -1
  168. metaflow/plugins/cards/card_modules/card.py +16 -1
  169. metaflow/plugins/cards/card_modules/chevron/renderer.py +1 -1
  170. metaflow/plugins/cards/card_modules/components.py +665 -28
  171. metaflow/plugins/cards/card_modules/convert_to_native_type.py +36 -7
  172. metaflow/plugins/cards/card_modules/json_viewer.py +232 -0
  173. metaflow/plugins/cards/card_modules/main.css +1 -0
  174. metaflow/plugins/cards/card_modules/main.js +68 -49
  175. metaflow/plugins/cards/card_modules/renderer_tools.py +1 -0
  176. metaflow/plugins/cards/card_modules/test_cards.py +26 -12
  177. metaflow/plugins/cards/card_server.py +39 -14
  178. metaflow/plugins/cards/component_serializer.py +2 -9
  179. metaflow/plugins/cards/metadata.py +22 -0
  180. metaflow/plugins/catch_decorator.py +9 -0
  181. metaflow/plugins/datastores/azure_storage.py +10 -1
  182. metaflow/plugins/datastores/gs_storage.py +6 -2
  183. metaflow/plugins/datastores/local_storage.py +12 -6
  184. metaflow/plugins/datastores/spin_storage.py +12 -0
  185. metaflow/plugins/datatools/local.py +2 -0
  186. metaflow/plugins/datatools/s3/s3.py +126 -75
  187. metaflow/plugins/datatools/s3/s3op.py +254 -121
  188. metaflow/plugins/env_escape/__init__.py +3 -3
  189. metaflow/plugins/env_escape/client_modules.py +102 -72
  190. metaflow/plugins/env_escape/server.py +7 -0
  191. metaflow/plugins/env_escape/stub.py +24 -5
  192. metaflow/plugins/events_decorator.py +343 -185
  193. metaflow/plugins/exit_hook/__init__.py +0 -0
  194. metaflow/plugins/exit_hook/exit_hook_decorator.py +46 -0
  195. metaflow/plugins/exit_hook/exit_hook_script.py +52 -0
  196. metaflow/plugins/gcp/__init__.py +1 -1
  197. metaflow/plugins/gcp/gcp_secret_manager_secrets_provider.py +11 -6
  198. metaflow/plugins/gcp/gs_tail.py +10 -6
  199. metaflow/plugins/gcp/includefile_support.py +3 -0
  200. metaflow/plugins/kubernetes/kube_utils.py +108 -0
  201. metaflow/plugins/kubernetes/kubernetes.py +411 -130
  202. metaflow/plugins/kubernetes/kubernetes_cli.py +168 -36
  203. metaflow/plugins/kubernetes/kubernetes_client.py +104 -2
  204. metaflow/plugins/kubernetes/kubernetes_decorator.py +246 -88
  205. metaflow/plugins/kubernetes/kubernetes_job.py +253 -581
  206. metaflow/plugins/kubernetes/kubernetes_jobsets.py +1071 -0
  207. metaflow/plugins/kubernetes/spot_metadata_cli.py +69 -0
  208. metaflow/plugins/kubernetes/spot_monitor_sidecar.py +109 -0
  209. metaflow/plugins/logs_cli.py +359 -0
  210. metaflow/plugins/{metadata → metadata_providers}/local.py +144 -84
  211. metaflow/plugins/{metadata → metadata_providers}/service.py +103 -26
  212. metaflow/plugins/metadata_providers/spin.py +16 -0
  213. metaflow/plugins/package_cli.py +36 -24
  214. metaflow/plugins/parallel_decorator.py +128 -11
  215. metaflow/plugins/parsers.py +16 -0
  216. metaflow/plugins/project_decorator.py +51 -5
  217. metaflow/plugins/pypi/bootstrap.py +357 -105
  218. metaflow/plugins/pypi/conda_decorator.py +82 -81
  219. metaflow/plugins/pypi/conda_environment.py +187 -52
  220. metaflow/plugins/pypi/micromamba.py +157 -47
  221. metaflow/plugins/pypi/parsers.py +268 -0
  222. metaflow/plugins/pypi/pip.py +88 -13
  223. metaflow/plugins/pypi/pypi_decorator.py +37 -1
  224. metaflow/plugins/pypi/utils.py +48 -2
  225. metaflow/plugins/resources_decorator.py +2 -2
  226. metaflow/plugins/secrets/__init__.py +3 -0
  227. metaflow/plugins/secrets/secrets_decorator.py +26 -181
  228. metaflow/plugins/secrets/secrets_func.py +49 -0
  229. metaflow/plugins/secrets/secrets_spec.py +101 -0
  230. metaflow/plugins/secrets/utils.py +74 -0
  231. metaflow/plugins/tag_cli.py +4 -7
  232. metaflow/plugins/test_unbounded_foreach_decorator.py +41 -6
  233. metaflow/plugins/timeout_decorator.py +3 -3
  234. metaflow/plugins/uv/__init__.py +0 -0
  235. metaflow/plugins/uv/bootstrap.py +128 -0
  236. metaflow/plugins/uv/uv_environment.py +72 -0
  237. metaflow/procpoll.py +1 -1
  238. metaflow/pylint_wrapper.py +5 -1
  239. metaflow/runner/__init__.py +0 -0
  240. metaflow/runner/click_api.py +717 -0
  241. metaflow/runner/deployer.py +470 -0
  242. metaflow/runner/deployer_impl.py +201 -0
  243. metaflow/runner/metaflow_runner.py +714 -0
  244. metaflow/runner/nbdeploy.py +132 -0
  245. metaflow/runner/nbrun.py +225 -0
  246. metaflow/runner/subprocess_manager.py +650 -0
  247. metaflow/runner/utils.py +335 -0
  248. metaflow/runtime.py +1078 -260
  249. metaflow/sidecar/sidecar_worker.py +1 -1
  250. metaflow/system/__init__.py +5 -0
  251. metaflow/system/system_logger.py +85 -0
  252. metaflow/system/system_monitor.py +108 -0
  253. metaflow/system/system_utils.py +19 -0
  254. metaflow/task.py +521 -225
  255. metaflow/tracing/__init__.py +7 -7
  256. metaflow/tracing/span_exporter.py +31 -38
  257. metaflow/tracing/tracing_modules.py +38 -43
  258. metaflow/tuple_util.py +27 -0
  259. metaflow/user_configs/__init__.py +0 -0
  260. metaflow/user_configs/config_options.py +563 -0
  261. metaflow/user_configs/config_parameters.py +598 -0
  262. metaflow/user_decorators/__init__.py +0 -0
  263. metaflow/user_decorators/common.py +144 -0
  264. metaflow/user_decorators/mutable_flow.py +512 -0
  265. metaflow/user_decorators/mutable_step.py +424 -0
  266. metaflow/user_decorators/user_flow_decorator.py +264 -0
  267. metaflow/user_decorators/user_step_decorator.py +749 -0
  268. metaflow/util.py +243 -27
  269. metaflow/vendor.py +23 -7
  270. metaflow/version.py +1 -1
  271. ob_metaflow-2.19.7.1rc0.data/data/share/metaflow/devtools/Makefile +355 -0
  272. ob_metaflow-2.19.7.1rc0.data/data/share/metaflow/devtools/Tiltfile +726 -0
  273. ob_metaflow-2.19.7.1rc0.data/data/share/metaflow/devtools/pick_services.sh +105 -0
  274. ob_metaflow-2.19.7.1rc0.dist-info/METADATA +87 -0
  275. ob_metaflow-2.19.7.1rc0.dist-info/RECORD +445 -0
  276. {ob_metaflow-2.11.13.1.dist-info → ob_metaflow-2.19.7.1rc0.dist-info}/WHEEL +1 -1
  277. {ob_metaflow-2.11.13.1.dist-info → ob_metaflow-2.19.7.1rc0.dist-info}/entry_points.txt +1 -0
  278. metaflow/_vendor/v3_5/__init__.py +0 -1
  279. metaflow/_vendor/v3_5/importlib_metadata/__init__.py +0 -644
  280. metaflow/_vendor/v3_5/importlib_metadata/_compat.py +0 -152
  281. metaflow/package.py +0 -188
  282. ob_metaflow-2.11.13.1.dist-info/METADATA +0 -85
  283. ob_metaflow-2.11.13.1.dist-info/RECORD +0 -308
  284. /metaflow/_vendor/{v3_5/zipp.py → zipp.py} +0 -0
  285. /metaflow/{metadata → metadata_provider}/__init__.py +0 -0
  286. /metaflow/{metadata → metadata_provider}/util.py +0 -0
  287. /metaflow/plugins/{metadata → metadata_providers}/__init__.py +0 -0
  288. {ob_metaflow-2.11.13.1.dist-info → ob_metaflow-2.19.7.1rc0.dist-info/licenses}/LICENSE +0 -0
  289. {ob_metaflow-2.11.13.1.dist-info → ob_metaflow-2.19.7.1rc0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,726 @@
1
+ # Tilt configuration for running Metaflow on a local Kubernetes stack
2
+ #
3
+ # Usage:
4
+ # Start the development environment:
5
+ # $ tilt up
6
+ # Stop and clean up:
7
+ # $ tilt down
8
+
9
+ # TODO:
10
+ # 1. move away from temporary images
11
+ # 2. introduce kueue and jobsets
12
+ # 3. lock versions
13
+
14
+ version_settings(constraint='>=0.22.2')
15
+ allow_k8s_contexts('minikube')
16
+
17
+ # Version configuration for components
18
+ JOBSET_VERSION = os.getenv("JOBSET_VERSION", "v0.8.2")
19
+
20
+ # Argo Workflows versions
21
+ ARGO_WORKFLOWS_HELM_CHART_VERSION = os.getenv("ARGO_WORKFLOWS_HELM_CHART_VERSION", "0.45.2") # Helm chart version
22
+ ARGO_WORKFLOWS_IMAGE_TAG = os.getenv("ARGO_WORKFLOWS_IMAGE_TAG", "v3.6.0") # Argo Workflows application version
23
+
24
+ # Argo Events versions
25
+ ARGO_EVENTS_HELM_CHART_VERSION = os.getenv("ARGO_EVENTS_HELM_CHART_VERSION", "2.4.8") # Helm chart version
26
+ ARGO_EVENTS_IMAGE_TAG = os.getenv("ARGO_EVENTS_IMAGE_TAG", "v1.9.2") # Argo Events application version
27
+
28
+ components = {
29
+ "metadata-service": ["postgresql"],
30
+ "ui": ["postgresql", "minio"],
31
+ "minio": [],
32
+ "postgresql": [],
33
+ "argo-workflows": [],
34
+ "argo-events": ["argo-workflows"],
35
+ "jobset": [],
36
+ }
37
+
38
+ services_env = os.getenv("SERVICES", "all").strip().lower()
39
+
40
+ if services_env:
41
+ if services_env == "all":
42
+ requested_components = list(components.keys())
43
+ else:
44
+ requested_components = services_env.split(",")
45
+ else:
46
+ requested_components = list(components.keys())
47
+
48
+ metaflow_config = {}
49
+ metaflow_config["METAFLOW_KUBERNETES_NAMESPACE"] = "default"
50
+
51
+ aws_config = []
52
+
53
+ def write_config_files():
54
+ metaflow_json = encode_json(metaflow_config)
55
+ cmd = '''cat > .devtools/config_local.json <<EOF
56
+ %s
57
+ EOF
58
+ ''' % (metaflow_json)
59
+ if aws_config and aws_config.strip():
60
+ cmd += '''cat > .devtools/aws_config <<EOF
61
+ %s
62
+ EOF
63
+ ''' % (aws_config.strip())
64
+ return cmd
65
+
66
+ load('ext://helm_resource', 'helm_resource', 'helm_repo')
67
+ load('ext://helm_remote', 'helm_remote')
68
+
69
+
70
+ def resolve(component, resolved=None):
71
+ if resolved == None:
72
+ resolved = []
73
+ if component in resolved:
74
+ return resolved
75
+ if component in components:
76
+ for dep in components[component]:
77
+ resolve(dep, resolved)
78
+ resolved.append(component)
79
+ return resolved
80
+
81
+ valid_components = []
82
+ for component in components.keys():
83
+ if component not in valid_components:
84
+ valid_components.append(component)
85
+ for deps in components.values():
86
+ for dep in deps:
87
+ if dep not in valid_components:
88
+ valid_components.append(dep)
89
+
90
+ enabled_components = []
91
+ for component in requested_components:
92
+ if component not in valid_components:
93
+ fail("Unknown component: " + component)
94
+ for result in resolve(component):
95
+ if result not in enabled_components:
96
+ enabled_components.append(result)
97
+
98
+ # Print a friendly summary when running `tilt up`.
99
+ if config.tilt_subcommand == 'up':
100
+ print("\n📦 Components to install:")
101
+ for component in enabled_components:
102
+ print("• " + component)
103
+ if component in components and components[component]:
104
+ print(" ↳ requires: " + ", ".join(components[component]))
105
+
106
+ config_resources = []
107
+
108
+ #################################################
109
+ # MINIO
110
+ #################################################
111
+ if "minio" in enabled_components:
112
+ helm_remote(
113
+ 'minio',
114
+ repo_name='minio-s3',
115
+ repo_url='https://charts.min.io/',
116
+ set=[
117
+ 'rootUser=rootuser',
118
+ 'rootPassword=rootpass123',
119
+ # TODO: perturb the bucket name to avoid conflicts
120
+ 'buckets[0].name=metaflow-test',
121
+ 'buckets[0].policy=none',
122
+ 'buckets[0].purge=false',
123
+ 'mode=standalone',
124
+ 'replicas=1',
125
+ 'persistence.enabled=false',
126
+ 'resources.requests.memory=128Mi',
127
+ 'resources.requests.cpu=50m',
128
+ 'resources.limits.memory=256Mi',
129
+ 'resources.limits.cpu=100m',
130
+ ]
131
+ )
132
+
133
+ k8s_resource(
134
+ 'minio',
135
+ port_forwards=[
136
+ '9000:9000',
137
+ '9001:9001'
138
+ ],
139
+ links=[
140
+ link('http://localhost:9000', 'MinIO API'),
141
+ link('http://localhost:9001/login', 'MinIO Console (rootuser/rootpass123)')
142
+ ],
143
+ labels=['minio'],
144
+ )
145
+
146
+ k8s_resource(
147
+ "minio-post-job",
148
+ labels=['minio'],
149
+ )
150
+
151
+ k8s_yaml(encode_yaml({
152
+ 'apiVersion': 'v1',
153
+ 'kind': 'Secret',
154
+ 'metadata': {'name': 'minio-secret'},
155
+ 'type': 'Opaque',
156
+ 'stringData': {
157
+ 'AWS_ACCESS_KEY_ID': 'rootuser',
158
+ 'AWS_SECRET_ACCESS_KEY': 'rootpass123',
159
+ 'AWS_ENDPOINT_URL_S3': 'http://minio.default.svc.cluster.local:9000',
160
+ }
161
+ }))
162
+
163
+ metaflow_config["METAFLOW_DEFAULT_DATASTORE"] = "s3"
164
+ metaflow_config["METAFLOW_DATASTORE_SYSROOT_S3"] = "s3://metaflow-test/metaflow"
165
+ metaflow_config["METAFLOW_KUBERNETES_SECRETS"] = "minio-secret"
166
+
167
+ aws_config = """[default]
168
+ aws_access_key_id = rootuser
169
+ aws_secret_access_key = rootpass123
170
+ endpoint_url = http://localhost:9000
171
+ """
172
+ config_resources.append('minio')
173
+
174
+ #################################################
175
+ # POSTGRESQL
176
+ #################################################
177
+ if "postgresql" in enabled_components:
178
+ helm_remote(
179
+ 'postgresql',
180
+ version='12.5.6',
181
+ repo_name='postgresql',
182
+ repo_url='https://charts.bitnami.com/bitnami',
183
+ set=[
184
+ 'auth.username=metaflow',
185
+ 'auth.password=metaflow123',
186
+ 'auth.database=metaflow',
187
+ 'image.repository=bitnamilegacy/postgresql',
188
+ 'primary.persistence.enabled=false',
189
+ 'primary.resources.requests.memory=128Mi',
190
+ 'primary.resources.requests.cpu=50m',
191
+ 'primary.resources.limits.memory=256Mi',
192
+ 'primary.resources.limits.cpu=100m',
193
+ 'primary.terminationGracePeriodSeconds=1',
194
+ 'primary.podSecurityContext.enabled=false',
195
+ 'primary.containerSecurityContext.enabled=false',
196
+ 'volumePermissions.enabled=false',
197
+ 'shmVolume.enabled=false',
198
+ 'primary.extraVolumes=null',
199
+ 'primary.extraVolumeMounts=null'
200
+ ]
201
+ )
202
+
203
+ k8s_resource(
204
+ 'postgresql',
205
+ port_forwards=['5432:5432'],
206
+ links=[
207
+ link('postgresql://metaflow:metaflow@localhost:5432/metaflow', 'PostgreSQL Connection')
208
+ ],
209
+ labels=['postgresql'],
210
+ resource_deps=components['postgresql'],
211
+ )
212
+
213
+ config_resources.append('postgresql')
214
+
215
+ #################################################
216
+ # ARGO WORKFLOWS
217
+ #################################################
218
+ if "argo-workflows" in enabled_components:
219
+ helm_remote(
220
+ 'argo-workflows',
221
+ version=ARGO_WORKFLOWS_HELM_CHART_VERSION,
222
+ repo_name='argo',
223
+ repo_url='https://argoproj.github.io/argo-helm',
224
+ set=[
225
+ 'server.extraArgs[0]=--auth-mode=server',
226
+ 'workflow.serviceAccount.create=true',
227
+ 'workflow.rbac.create=true',
228
+ 'server.livenessProbe.initialDelaySeconds=1',
229
+ 'server.readinessProbe.initialDelaySeconds=1',
230
+ 'server.resources.requests.memory=128Mi',
231
+ 'server.resources.requests.cpu=50m',
232
+ 'server.resources.limits.memory=256Mi',
233
+ 'server.resources.limits.cpu=100m',
234
+ 'controller.resources.requests.memory=128Mi',
235
+ 'controller.resources.requests.cpu=50m',
236
+ 'controller.resources.limits.memory=256Mi',
237
+ 'controller.resources.limits.cpu=100m',
238
+ # Image version overrides
239
+ 'images.tag=%s' % ARGO_WORKFLOWS_IMAGE_TAG,
240
+ ]
241
+ )
242
+
243
+ # This fixes issue described in: https://github.com/argoproj/argo-workflows/issues/10340
244
+ k8s_yaml(encode_yaml({
245
+ 'apiVersion': 'v1',
246
+ 'kind': 'Secret',
247
+ 'metadata': {
248
+ 'name': 'default.service-account-token',
249
+ 'annotations': {
250
+ 'kubernetes.io/service-account.name': 'default'
251
+ }
252
+ },
253
+ 'type': 'kubernetes.io/service-account-token'
254
+ }))
255
+
256
+ k8s_yaml(encode_yaml({
257
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
258
+ 'kind': 'Role',
259
+ 'metadata': {
260
+ 'name': 'argo-workflowtaskresults-role',
261
+ 'namespace': 'default'
262
+ },
263
+ 'rules': [
264
+ {
265
+ 'apiGroups': ['argoproj.io'],
266
+ 'resources': ['workflowtaskresults'],
267
+ 'verbs': ['create', 'patch', 'get', 'list']
268
+ },
269
+ {
270
+ 'apiGroups': ['argoproj.io'],
271
+ 'resources': ['workflowtasksets'],
272
+ 'verbs': ['watch', 'list']
273
+ },
274
+ {
275
+ 'apiGroups': ['argoproj.io'],
276
+ 'resources': ['workflowtasksets/status'],
277
+ 'verbs': ['patch']
278
+ },
279
+ ]
280
+ }))
281
+
282
+ k8s_yaml(encode_yaml({
283
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
284
+ 'kind': 'RoleBinding',
285
+ 'metadata': {
286
+ 'name': 'default-argo-workflowtaskresults-binding',
287
+ 'namespace': 'default'
288
+ },
289
+ 'subjects': [{
290
+ 'kind': 'ServiceAccount',
291
+ 'name': 'default',
292
+ 'namespace': 'default'
293
+ }],
294
+ 'roleRef': {
295
+ 'kind': 'Role',
296
+ 'name': 'argo-workflowtaskresults-role',
297
+ 'apiGroup': 'rbac.authorization.k8s.io'
298
+ }
299
+ }))
300
+
301
+ k8s_resource(
302
+ workload='argo-workflows-server',
303
+ port_forwards=['2746:2746'],
304
+ links=[
305
+ link('http://localhost:2746', 'Argo Workflows UI')
306
+ ],
307
+ labels=['argo-workflows'],
308
+ resource_deps=components['argo-workflows']
309
+ )
310
+
311
+ k8s_resource(
312
+ workload='argo-workflows-workflow-controller',
313
+ labels=['argo-workflows'],
314
+ resource_deps=components['argo-workflows']
315
+ )
316
+
317
+ config_resources.append('argo-workflows-workflow-controller')
318
+ config_resources.append('argo-workflows-server')
319
+
320
+ #################################################
321
+ # ARGO EVENTS
322
+ #################################################
323
+ if "argo-events" in enabled_components:
324
+ helm_remote(
325
+ 'argo-events',
326
+ version=ARGO_EVENTS_HELM_CHART_VERSION,
327
+ repo_name='argo',
328
+ repo_url='https://argoproj.github.io/argo-helm',
329
+ set=[
330
+ 'crds.install=true',
331
+ 'controller.metrics.enabled=true',
332
+ 'controller.livenessProbe.initialDelaySeconds=1',
333
+ 'controller.readinessProbe.initialDelaySeconds=1',
334
+ 'controller.resources.requests.memory=64Mi',
335
+ 'controller.resources.requests.cpu=25m',
336
+ 'controller.resources.limits.memory=128Mi',
337
+ 'controller.resources.limits.cpu=50m',
338
+ 'configs.jetstream.streamConfig.maxAge=72h',
339
+ 'configs.jetstream.streamConfig.replicas=1',
340
+ 'controller.rbac.enabled=true',
341
+ 'controller.rbac.namespaced=false',
342
+ 'controller.serviceAccount.create=true',
343
+ 'controller.serviceAccount.name=argo-events-events-controller-sa',
344
+ 'configs.jetstream.versions[0].configReloaderImage=natsio/nats-server-config-reloader:latest',
345
+ 'configs.jetstream.versions[0].metricsExporterImage=natsio/prometheus-nats-exporter:latest',
346
+ 'configs.jetstream.versions[0].natsImage=nats:latest',
347
+ 'configs.jetstream.versions[0].startCommand=/nats-server',
348
+ 'configs.jetstream.versions[0].version=latest',
349
+ 'configs.jetstream.versions[1].configReloaderImage=natsio/nats-server-config-reloader:latest',
350
+ 'configs.jetstream.versions[1].metricsExporterImage=natsio/prometheus-nats-exporter:latest',
351
+ 'configs.jetstream.versions[1].natsImage=nats:2.9.15',
352
+ 'configs.jetstream.versions[1].startCommand=/nats-server',
353
+ 'configs.jetstream.versions[1].version=2.9.15',
354
+ # Image version overrides
355
+ 'global.image.tag=%s' % ARGO_EVENTS_IMAGE_TAG,
356
+ ]
357
+ )
358
+
359
+ k8s_yaml(encode_yaml({
360
+ 'apiVersion': 'v1',
361
+ 'kind': 'ServiceAccount',
362
+ 'metadata': {
363
+ 'name': 'operate-workflow-sa',
364
+ 'namespace': 'default'
365
+ }
366
+ }))
367
+
368
+ k8s_yaml(encode_yaml({
369
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
370
+ 'kind': 'Role',
371
+ 'metadata': {
372
+ 'name': 'operate-workflow-role',
373
+ 'namespace': 'default'
374
+ },
375
+ 'rules': [{
376
+ 'apiGroups': ['argoproj.io'],
377
+ 'resources': [
378
+ 'workflows',
379
+ 'workflowtemplates',
380
+ 'cronworkflows',
381
+ 'clusterworkflowtemplates'
382
+ ],
383
+ 'verbs': ['*']
384
+ }]
385
+ }))
386
+
387
+ k8s_yaml(encode_yaml({
388
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
389
+ 'kind': 'RoleBinding',
390
+ 'metadata': {
391
+ 'name': 'operate-workflow-role-binding',
392
+ 'namespace': 'default'
393
+ },
394
+ 'roleRef': {
395
+ 'apiGroup': 'rbac.authorization.k8s.io',
396
+ 'kind': 'Role',
397
+ 'name': 'operate-workflow-role'
398
+ },
399
+ 'subjects': [{
400
+ 'kind': 'ServiceAccount',
401
+ 'name': 'operate-workflow-sa'
402
+ }]
403
+ }))
404
+
405
+ k8s_yaml(encode_yaml({
406
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
407
+ 'kind': 'Role',
408
+ 'metadata': {
409
+ 'name': 'view-events-role',
410
+ 'namespace': 'default'
411
+ },
412
+ 'rules': [{
413
+ 'apiGroups': ['argoproj.io'],
414
+ 'resources': [
415
+ 'eventsources',
416
+ 'eventbuses',
417
+ 'sensors'
418
+ ],
419
+ 'verbs': [
420
+ 'get',
421
+ 'list',
422
+ 'watch'
423
+ ]
424
+ }]
425
+ }))
426
+
427
+ k8s_yaml(encode_yaml({
428
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
429
+ 'kind': 'RoleBinding',
430
+ 'metadata': {
431
+ 'name': 'view-events-role-binding',
432
+ 'namespace': 'default'
433
+ },
434
+ 'roleRef': {
435
+ 'apiGroup': 'rbac.authorization.k8s.io',
436
+ 'kind': 'Role',
437
+ 'name': 'view-events-role'
438
+ },
439
+ 'subjects': [{
440
+ 'kind': 'ServiceAccount',
441
+ 'name': 'argo-workflows',
442
+ 'namespace': 'default'
443
+ }]
444
+ }))
445
+
446
+ k8s_yaml(encode_yaml({
447
+ 'apiVersion': 'argoproj.io/v1alpha1',
448
+ 'kind': 'EventBus',
449
+ 'metadata': {
450
+ 'name': 'default',
451
+ 'namespace': 'default'
452
+ },
453
+ 'spec': {
454
+ 'jetstream': {
455
+ 'version': '2.9.15',
456
+ 'replicas': 3,
457
+ 'containerTemplate': {
458
+ 'resources': {
459
+ 'limits': {
460
+ 'cpu': '100m',
461
+ 'memory': '128Mi'
462
+ },
463
+ 'requests': {
464
+ 'cpu': '100m',
465
+ 'memory': '128Mi'
466
+ }
467
+ }
468
+ }
469
+ }
470
+ }
471
+ }))
472
+
473
+ k8s_yaml(encode_yaml({
474
+ 'apiVersion': 'argoproj.io/v1alpha1',
475
+ 'kind': 'EventSource',
476
+ 'metadata': {
477
+ 'name': 'argo-events-webhook',
478
+ 'namespace': 'default'
479
+ },
480
+ 'spec': {
481
+ 'template': {
482
+ 'container': {
483
+ 'resources': {
484
+ 'requests': {
485
+ 'cpu': '25m',
486
+ 'memory': '50Mi'
487
+ },
488
+ 'limits': {
489
+ 'cpu': '25m',
490
+ 'memory': '50Mi'
491
+ }
492
+ }
493
+ }
494
+ },
495
+ 'service': {
496
+ 'ports': [
497
+ {
498
+ 'port': 12000,
499
+ 'targetPort': 12000
500
+ }
501
+ ]
502
+ },
503
+ 'webhook': {
504
+ 'metaflow-event': {
505
+ 'port': '12000',
506
+ 'endpoint': '/metaflow-event',
507
+ 'method': 'POST'
508
+ }
509
+ }
510
+ }
511
+ }))
512
+
513
+ # Create a custom service and port-forward it because tilt :/
514
+ k8s_yaml(encode_yaml(
515
+ {
516
+ 'apiVersion': 'v1',
517
+ 'kind': 'Service',
518
+ 'metadata': {
519
+ 'name': 'argo-events-webhook-eventsource-svc-tilt',
520
+ 'namespace': 'default',
521
+ },
522
+ 'spec': {
523
+ 'ports': [{
524
+ 'port': 12000,
525
+ 'protocol': 'TCP',
526
+ 'targetPort': 12000
527
+ }],
528
+ 'selector': {
529
+ 'controller': 'eventsource-controller',
530
+ 'eventsource-name': 'argo-events-webhook',
531
+ 'owner-name': 'argo-events-webhook'
532
+ },
533
+ 'type': 'ClusterIP'
534
+ }
535
+ }
536
+ ))
537
+
538
+ local_resource(
539
+ name='argo-events-webhook-eventsource-svc',
540
+ serve_cmd='while ! kubectl get service/argo-events-webhook-eventsource-svc-tilt >/dev/null 2>&1 || ! kubectl get pods -l eventsource-name=argo-events-webhook -o jsonpath="{.items[*].status.phase}" | grep -q "Running"; do sleep 5; done && kubectl port-forward service/argo-events-webhook-eventsource-svc-tilt 12000:12000',
541
+ links=[
542
+ link('http://localhost:12000/metaflow-event', 'Argo Events Webhook'),
543
+ ],
544
+ labels=['argo-events']
545
+ )
546
+
547
+ k8s_resource(
548
+ 'argo-events-controller-manager',
549
+ labels=['argo-events'],
550
+ )
551
+
552
+ metaflow_config["METAFLOW_ARGO_EVENTS_EVENT"] = "metaflow-event"
553
+ metaflow_config["METAFLOW_ARGO_EVENTS_EVENT_BUS"] = "default"
554
+ metaflow_config["METAFLOW_ARGO_EVENTS_EVENT_SOURCE"] = "argo-events-webhook"
555
+ metaflow_config["METAFLOW_ARGO_EVENTS_SERVICE_ACCOUNT"] = "operate-workflow-sa"
556
+ metaflow_config["METAFLOW_ARGO_EVENTS_WEBHOOK_AUTH"] = "service"
557
+ metaflow_config["METAFLOW_ARGO_EVENTS_INTERNAL_WEBHOOK_URL"] = "http://argo-events-webhook-eventsource-svc:12000/metaflow-event"
558
+ metaflow_config["METAFLOW_ARGO_EVENTS_WEBHOOK_URL"] = "http://localhost:12000/metaflow-event"
559
+
560
+ config_resources.append('argo-events-controller-manager')
561
+ config_resources.append('argo-events-webhook-eventsource-svc')
562
+
563
+ #################################################
564
+ # JOBSET
565
+ #################################################
566
+ if "jobset" in enabled_components:
567
+ # Apply JobSet manifests directly from GitHub releases
568
+ jobset_manifest_url = "https://github.com/kubernetes-sigs/jobset/releases/download/%s/manifests.yaml" % JOBSET_VERSION
569
+
570
+ cmd = "curl -sSL %s" % (jobset_manifest_url)
571
+ k8s_yaml(
572
+ local(
573
+ cmd,
574
+ )
575
+ )
576
+
577
+ k8s_resource(
578
+ 'jobset-controller-manager',
579
+ labels=['jobset'],
580
+ )
581
+
582
+ metaflow_config["METAFLOW_KUBERNETES_JOBSET_ENABLED"] = "true"
583
+
584
+ config_resources.append('jobset-controller-manager')
585
+
586
+ # ClusterRole for jobset operations
587
+ k8s_yaml(encode_yaml({
588
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
589
+ 'kind': 'ClusterRole',
590
+ 'metadata': {
591
+ 'name': 'jobset-full-access'
592
+ },
593
+ 'rules': [{
594
+ 'apiGroups': ['jobset.x-k8s.io'],
595
+ 'resources': ['jobsets'],
596
+ 'verbs': ['*']
597
+ }]
598
+ }))
599
+
600
+ # ClusterRoleBinding for default service account to access jobsets
601
+ k8s_yaml(encode_yaml({
602
+ 'apiVersion': 'rbac.authorization.k8s.io/v1',
603
+ 'kind': 'ClusterRoleBinding',
604
+ 'metadata': {
605
+ 'name': 'default-jobset-binding'
606
+ },
607
+ 'subjects': [{
608
+ 'kind': 'ServiceAccount',
609
+ 'name': 'default',
610
+ 'namespace': 'default'
611
+ }],
612
+ 'roleRef': {
613
+ 'kind': 'ClusterRole',
614
+ 'name': 'jobset-full-access',
615
+ 'apiGroup': 'rbac.authorization.k8s.io'
616
+ }
617
+ }))
618
+
619
+ #################################################
620
+ # METADATA SERVICE
621
+ #################################################
622
+ if "metadata-service" in enabled_components:
623
+ helm_remote(
624
+ 'metaflow-service',
625
+ repo_name='metaflow-tools',
626
+ repo_url='https://outerbounds.github.io/metaflow-tools',
627
+ set=[
628
+ 'metadatadb.user=metaflow',
629
+ 'metadatadb.password=metaflow123',
630
+ 'metadatadb.database=metaflow',
631
+ 'metadatadb.host=postgresql',
632
+ 'image.repository=public.ecr.aws/outerbounds/metaflow_metadata_service',
633
+ 'image.tag=2.5.0',
634
+ 'resources.requests.cpu=25m',
635
+ 'resources.requests.memory=64Mi',
636
+ 'resources.limits.cpu=50m',
637
+ 'resources.limits.memory=128Mi'
638
+ ]
639
+ )
640
+
641
+ k8s_resource(
642
+ 'metaflow-service',
643
+ port_forwards=['8080:8080'],
644
+ links=[link('http://localhost:8080/ping', 'Ping Metaflow Service')],
645
+ labels=['metadata-service'],
646
+ resource_deps=components['metadata-service']
647
+ )
648
+
649
+ metaflow_config["METAFLOW_DEFAULT_METADATA"] = "service"
650
+ metaflow_config["METAFLOW_SERVICE_URL"] = "http://localhost:8080"
651
+ metaflow_config["METAFLOW_SERVICE_INTERNAL_URL"] = "http://metaflow-service.default.svc.cluster.local:8080"
652
+
653
+ config_resources.append('metaflow-service')
654
+
655
+ #################################################
656
+ # METAFLOW UI
657
+ #################################################
658
+ if "ui" in enabled_components:
659
+ helm_remote(
660
+ 'metaflow-ui',
661
+ repo_name='metaflow-tools',
662
+ repo_url='https://outerbounds.github.io/metaflow-tools',
663
+ set=[
664
+ 'uiBackend.metadatadb.user=metaflow',
665
+ 'uiBackend.metadatadb.password=metaflow123',
666
+ 'uiBackend.metadatadb.name=metaflow',
667
+ 'uiBackend.metadatadb.host=postgresql',
668
+ 'uiBackend.metaflowDatastoreSysRootS3=s3://metaflow-test',
669
+ 'uiBackend.metaflowS3EndpointURL=http://minio.default.svc.cluster.local:9000',
670
+ 'uiBackend.image.name=public.ecr.aws/outerbounds/metaflow_metadata_service',
671
+ 'uiBackend.image.tag=2.5.0',
672
+ 'uiBackend.env[0].name=AWS_ACCESS_KEY_ID',
673
+ 'uiBackend.env[0].value=rootuser',
674
+ 'uiBackend.env[1].name=AWS_SECRET_ACCESS_KEY',
675
+ 'uiBackend.env[1].value=rootpass123',
676
+ # TODO: configure lower cache limits
677
+ 'uiBackend.resources.requests.cpu=100m',
678
+ 'uiBackend.resources.requests.memory=256Mi',
679
+ 'uiStatic.metaflowUIBackendURL=http://localhost:8083/api',
680
+ 'uiStatic.image.name=public.ecr.aws/outerbounds/metaflow_ui',
681
+ 'uiStatic.image.tag=v1.3.14',
682
+ 'uiStatic.resources.requests.cpu=25m',
683
+ 'uiStatic.resources.requests.memory=64Mi',
684
+ 'uiStatic.resources.limits.cpu=50m',
685
+ 'uiStatic.resources.limits.memory=128Mi',
686
+ ]
687
+ )
688
+
689
+ k8s_resource(
690
+ 'metaflow-ui-static',
691
+ port_forwards=['3000:3000'],
692
+ links=[link('http://localhost:3000', 'Metaflow UI')],
693
+ labels=['metaflow-ui'],
694
+ resource_deps=components['ui']
695
+ )
696
+
697
+ k8s_resource(
698
+ 'metaflow-ui',
699
+ port_forwards=['8083:8083'],
700
+ links=[link('http://localhost:3000', 'Metaflow UI')],
701
+ labels=['metaflow-ui'],
702
+ resource_deps=components['ui']
703
+ )
704
+
705
+ metaflow_config["METAFLOW_UI_URL"] = "http://localhost:3000"
706
+
707
+ config_resources.append('metaflow-ui')
708
+ config_resources.append('metaflow-ui-static')
709
+
710
+ cmd = '''
711
+ ARCH=$(kubectl get nodes -o jsonpath='{.items[0].status.nodeInfo.architecture}')
712
+ case "$ARCH" in
713
+ arm64) echo linux-aarch64 ;;
714
+ amd64) echo linux-64 ;;
715
+ *) echo linux-64 ;;
716
+ esac
717
+ '''
718
+
719
+ # For @conda/@pypi emulation
720
+ metaflow_config["METAFLOW_KUBERNETES_CONDA_ARCH"] = str(local(cmd)).strip()
721
+
722
+ local_resource(
723
+ name="generate-configs",
724
+ cmd=write_config_files(),
725
+ resource_deps=config_resources,
726
+ )