ob-metaflow-extensions 1.1.45rc3__py2.py3-none-any.whl → 1.5.1__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.

Potentially problematic release.


This version of ob-metaflow-extensions might be problematic. Click here for more details.

Files changed (128) hide show
  1. metaflow_extensions/outerbounds/__init__.py +1 -7
  2. metaflow_extensions/outerbounds/config/__init__.py +35 -0
  3. metaflow_extensions/outerbounds/plugins/__init__.py +186 -57
  4. metaflow_extensions/outerbounds/plugins/apps/__init__.py +0 -0
  5. metaflow_extensions/outerbounds/plugins/apps/app_cli.py +0 -0
  6. metaflow_extensions/outerbounds/plugins/apps/app_utils.py +187 -0
  7. metaflow_extensions/outerbounds/plugins/apps/consts.py +3 -0
  8. metaflow_extensions/outerbounds/plugins/apps/core/__init__.py +15 -0
  9. metaflow_extensions/outerbounds/plugins/apps/core/_state_machine.py +506 -0
  10. metaflow_extensions/outerbounds/plugins/apps/core/_vendor/__init__.py +0 -0
  11. metaflow_extensions/outerbounds/plugins/apps/core/_vendor/spinner/__init__.py +4 -0
  12. metaflow_extensions/outerbounds/plugins/apps/core/_vendor/spinner/spinners.py +478 -0
  13. metaflow_extensions/outerbounds/plugins/apps/core/app_config.py +128 -0
  14. metaflow_extensions/outerbounds/plugins/apps/core/app_deploy_decorator.py +330 -0
  15. metaflow_extensions/outerbounds/plugins/apps/core/artifacts.py +0 -0
  16. metaflow_extensions/outerbounds/plugins/apps/core/capsule.py +958 -0
  17. metaflow_extensions/outerbounds/plugins/apps/core/click_importer.py +24 -0
  18. metaflow_extensions/outerbounds/plugins/apps/core/code_package/__init__.py +3 -0
  19. metaflow_extensions/outerbounds/plugins/apps/core/code_package/code_packager.py +618 -0
  20. metaflow_extensions/outerbounds/plugins/apps/core/code_package/examples.py +125 -0
  21. metaflow_extensions/outerbounds/plugins/apps/core/config/__init__.py +15 -0
  22. metaflow_extensions/outerbounds/plugins/apps/core/config/cli_generator.py +165 -0
  23. metaflow_extensions/outerbounds/plugins/apps/core/config/config_utils.py +966 -0
  24. metaflow_extensions/outerbounds/plugins/apps/core/config/schema_export.py +299 -0
  25. metaflow_extensions/outerbounds/plugins/apps/core/config/typed_configs.py +233 -0
  26. metaflow_extensions/outerbounds/plugins/apps/core/config/typed_init_generator.py +537 -0
  27. metaflow_extensions/outerbounds/plugins/apps/core/config/unified_config.py +1125 -0
  28. metaflow_extensions/outerbounds/plugins/apps/core/config_schema.yaml +337 -0
  29. metaflow_extensions/outerbounds/plugins/apps/core/dependencies.py +115 -0
  30. metaflow_extensions/outerbounds/plugins/apps/core/deployer.py +959 -0
  31. metaflow_extensions/outerbounds/plugins/apps/core/experimental/__init__.py +89 -0
  32. metaflow_extensions/outerbounds/plugins/apps/core/perimeters.py +87 -0
  33. metaflow_extensions/outerbounds/plugins/apps/core/secrets.py +164 -0
  34. metaflow_extensions/outerbounds/plugins/apps/core/utils.py +233 -0
  35. metaflow_extensions/outerbounds/plugins/apps/core/validations.py +17 -0
  36. metaflow_extensions/outerbounds/plugins/apps/deploy_decorator.py +201 -0
  37. metaflow_extensions/outerbounds/plugins/apps/supervisord_utils.py +243 -0
  38. metaflow_extensions/outerbounds/plugins/auth_server.py +28 -8
  39. metaflow_extensions/outerbounds/plugins/aws/__init__.py +4 -0
  40. metaflow_extensions/outerbounds/plugins/aws/assume_role.py +3 -0
  41. metaflow_extensions/outerbounds/plugins/aws/assume_role_decorator.py +118 -0
  42. metaflow_extensions/outerbounds/plugins/card_utilities/__init__.py +0 -0
  43. metaflow_extensions/outerbounds/plugins/card_utilities/async_cards.py +142 -0
  44. metaflow_extensions/outerbounds/plugins/card_utilities/extra_components.py +545 -0
  45. metaflow_extensions/outerbounds/plugins/card_utilities/injector.py +70 -0
  46. metaflow_extensions/outerbounds/plugins/checkpoint_datastores/__init__.py +2 -0
  47. metaflow_extensions/outerbounds/plugins/checkpoint_datastores/coreweave.py +71 -0
  48. metaflow_extensions/outerbounds/plugins/checkpoint_datastores/external_chckpt.py +85 -0
  49. metaflow_extensions/outerbounds/plugins/checkpoint_datastores/nebius.py +73 -0
  50. metaflow_extensions/outerbounds/plugins/fast_bakery/__init__.py +0 -0
  51. metaflow_extensions/outerbounds/plugins/fast_bakery/baker.py +110 -0
  52. metaflow_extensions/outerbounds/plugins/fast_bakery/docker_environment.py +391 -0
  53. metaflow_extensions/outerbounds/plugins/fast_bakery/fast_bakery.py +188 -0
  54. metaflow_extensions/outerbounds/plugins/fast_bakery/fast_bakery_cli.py +54 -0
  55. metaflow_extensions/outerbounds/plugins/fast_bakery/fast_bakery_decorator.py +50 -0
  56. metaflow_extensions/outerbounds/plugins/kubernetes/kubernetes_client.py +79 -0
  57. metaflow_extensions/outerbounds/plugins/kubernetes/pod_killer.py +374 -0
  58. metaflow_extensions/outerbounds/plugins/nim/card.py +140 -0
  59. metaflow_extensions/outerbounds/plugins/nim/nim_decorator.py +101 -0
  60. metaflow_extensions/outerbounds/plugins/nim/nim_manager.py +379 -0
  61. metaflow_extensions/outerbounds/plugins/nim/utils.py +36 -0
  62. metaflow_extensions/outerbounds/plugins/nvcf/__init__.py +0 -0
  63. metaflow_extensions/outerbounds/plugins/nvcf/constants.py +3 -0
  64. metaflow_extensions/outerbounds/plugins/nvcf/exceptions.py +94 -0
  65. metaflow_extensions/outerbounds/plugins/nvcf/heartbeat_store.py +178 -0
  66. metaflow_extensions/outerbounds/plugins/nvcf/nvcf.py +417 -0
  67. metaflow_extensions/outerbounds/plugins/nvcf/nvcf_cli.py +280 -0
  68. metaflow_extensions/outerbounds/plugins/nvcf/nvcf_decorator.py +242 -0
  69. metaflow_extensions/outerbounds/plugins/nvcf/utils.py +6 -0
  70. metaflow_extensions/outerbounds/plugins/nvct/__init__.py +0 -0
  71. metaflow_extensions/outerbounds/plugins/nvct/exceptions.py +71 -0
  72. metaflow_extensions/outerbounds/plugins/nvct/nvct.py +131 -0
  73. metaflow_extensions/outerbounds/plugins/nvct/nvct_cli.py +289 -0
  74. metaflow_extensions/outerbounds/plugins/nvct/nvct_decorator.py +286 -0
  75. metaflow_extensions/outerbounds/plugins/nvct/nvct_runner.py +218 -0
  76. metaflow_extensions/outerbounds/plugins/nvct/utils.py +29 -0
  77. metaflow_extensions/outerbounds/plugins/ollama/__init__.py +225 -0
  78. metaflow_extensions/outerbounds/plugins/ollama/constants.py +1 -0
  79. metaflow_extensions/outerbounds/plugins/ollama/exceptions.py +22 -0
  80. metaflow_extensions/outerbounds/plugins/ollama/ollama.py +1924 -0
  81. metaflow_extensions/outerbounds/plugins/ollama/status_card.py +292 -0
  82. metaflow_extensions/outerbounds/plugins/optuna/__init__.py +48 -0
  83. metaflow_extensions/outerbounds/plugins/perimeters.py +19 -5
  84. metaflow_extensions/outerbounds/plugins/profilers/deco_injector.py +70 -0
  85. metaflow_extensions/outerbounds/plugins/profilers/gpu_profile_decorator.py +88 -0
  86. metaflow_extensions/outerbounds/plugins/profilers/simple_card_decorator.py +96 -0
  87. metaflow_extensions/outerbounds/plugins/s3_proxy/__init__.py +7 -0
  88. metaflow_extensions/outerbounds/plugins/s3_proxy/binary_caller.py +132 -0
  89. metaflow_extensions/outerbounds/plugins/s3_proxy/constants.py +11 -0
  90. metaflow_extensions/outerbounds/plugins/s3_proxy/exceptions.py +13 -0
  91. metaflow_extensions/outerbounds/plugins/s3_proxy/proxy_bootstrap.py +59 -0
  92. metaflow_extensions/outerbounds/plugins/s3_proxy/s3_proxy_api.py +93 -0
  93. metaflow_extensions/outerbounds/plugins/s3_proxy/s3_proxy_decorator.py +250 -0
  94. metaflow_extensions/outerbounds/plugins/s3_proxy/s3_proxy_manager.py +225 -0
  95. metaflow_extensions/outerbounds/plugins/secrets/__init__.py +0 -0
  96. metaflow_extensions/outerbounds/plugins/secrets/secrets.py +204 -0
  97. metaflow_extensions/outerbounds/plugins/snowflake/__init__.py +3 -0
  98. metaflow_extensions/outerbounds/plugins/snowflake/snowflake.py +378 -0
  99. metaflow_extensions/outerbounds/plugins/snowpark/__init__.py +0 -0
  100. metaflow_extensions/outerbounds/plugins/snowpark/snowpark.py +309 -0
  101. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_cli.py +277 -0
  102. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_client.py +150 -0
  103. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_decorator.py +273 -0
  104. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_exceptions.py +13 -0
  105. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_job.py +241 -0
  106. metaflow_extensions/outerbounds/plugins/snowpark/snowpark_service_spec.py +259 -0
  107. metaflow_extensions/outerbounds/plugins/tensorboard/__init__.py +50 -0
  108. metaflow_extensions/outerbounds/plugins/torchtune/__init__.py +163 -0
  109. metaflow_extensions/outerbounds/plugins/vllm/__init__.py +255 -0
  110. metaflow_extensions/outerbounds/plugins/vllm/constants.py +1 -0
  111. metaflow_extensions/outerbounds/plugins/vllm/exceptions.py +1 -0
  112. metaflow_extensions/outerbounds/plugins/vllm/status_card.py +352 -0
  113. metaflow_extensions/outerbounds/plugins/vllm/vllm_manager.py +621 -0
  114. metaflow_extensions/outerbounds/profilers/gpu.py +131 -47
  115. metaflow_extensions/outerbounds/remote_config.py +53 -16
  116. metaflow_extensions/outerbounds/toplevel/global_aliases_for_metaflow_package.py +138 -2
  117. metaflow_extensions/outerbounds/toplevel/ob_internal.py +4 -0
  118. metaflow_extensions/outerbounds/toplevel/plugins/ollama/__init__.py +1 -0
  119. metaflow_extensions/outerbounds/toplevel/plugins/optuna/__init__.py +1 -0
  120. metaflow_extensions/outerbounds/toplevel/plugins/snowflake/__init__.py +1 -0
  121. metaflow_extensions/outerbounds/toplevel/plugins/torchtune/__init__.py +1 -0
  122. metaflow_extensions/outerbounds/toplevel/plugins/vllm/__init__.py +1 -0
  123. metaflow_extensions/outerbounds/toplevel/s3_proxy.py +88 -0
  124. {ob_metaflow_extensions-1.1.45rc3.dist-info → ob_metaflow_extensions-1.5.1.dist-info}/METADATA +2 -2
  125. ob_metaflow_extensions-1.5.1.dist-info/RECORD +133 -0
  126. ob_metaflow_extensions-1.1.45rc3.dist-info/RECORD +0 -19
  127. {ob_metaflow_extensions-1.1.45rc3.dist-info → ob_metaflow_extensions-1.5.1.dist-info}/WHEEL +0 -0
  128. {ob_metaflow_extensions-1.1.45rc3.dist-info → ob_metaflow_extensions-1.5.1.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,89 @@
1
+ import os
2
+ from typing import TYPE_CHECKING
3
+
4
+ if TYPE_CHECKING:
5
+ from ..app_config import AppConfig
6
+
7
+ DEFAULT_BRANCH = "test"
8
+
9
+
10
+ # Account for project / branch and the capsule input.
11
+ def capsule_input_overrides(app_config: "AppConfig", capsule_input: dict):
12
+ project = app_config.get_state("project", None)
13
+ # Update the project/branch related configurations.
14
+ if project is not None:
15
+ branch = app_config.get_state("branch", DEFAULT_BRANCH)
16
+ capsule_input["tags"].extend(
17
+ [dict(key="project", value=project), dict(key="branch", value=branch)]
18
+ )
19
+
20
+ persistence = app_config.get_state("persistence", None)
21
+ if persistence is not None and persistence != "none":
22
+ capsule_input["persistence"] = persistence
23
+
24
+ capsule_input["generateStaticUrl"] = app_config.get_state(
25
+ "generate_static_url", False
26
+ )
27
+
28
+ model_asset_conf = app_config.get_state("models", None)
29
+ data_asset_conf = app_config.get_state("data", None)
30
+ code_info = _code_info(app_config)
31
+ # todo:fix me
32
+ _objects_key = "associatedObjects"
33
+ if model_asset_conf or data_asset_conf or code_info:
34
+ capsule_input[_objects_key] = {}
35
+
36
+ if model_asset_conf:
37
+ capsule_input[_objects_key]["models"] = [
38
+ {"assetId": x["asset_id"], "assetInstanceId": x["asset_instance_id"]}
39
+ for x in model_asset_conf
40
+ ]
41
+ if data_asset_conf:
42
+ capsule_input[_objects_key]["data"] = [
43
+ {"assetId": x["asset_id"], "assetInstanceId": x["asset_instance_id"]}
44
+ for x in data_asset_conf
45
+ ]
46
+ if code_info:
47
+ capsule_input[_objects_key]["code"] = code_info
48
+
49
+ return capsule_input
50
+
51
+
52
+ def _code_info(app_config: "AppConfig"):
53
+ from metaflow.metaflow_git import get_repository_info
54
+
55
+ try:
56
+ from metaflow.metaflow_git import _call_git # type: ignore
57
+ except ImportError:
58
+ # Fallback if _call_git is not available
59
+ def _call_git(args, path=None):
60
+ return "", 1, True
61
+
62
+ package_dirs = app_config.get_state("packaging_directories")
63
+ if package_dirs is None:
64
+ return None
65
+ for directory in package_dirs:
66
+ repo_info = get_repository_info(directory)
67
+ if len(repo_info) == 0:
68
+ continue
69
+ git_log_info, returncode, failed = _call_git(
70
+ ["log", "-1", "--pretty=%B"],
71
+ path=directory,
72
+ )
73
+ repo_url = repo_info["repo_url"]
74
+ if isinstance(repo_url, str):
75
+ _url = (
76
+ repo_url if not repo_url.endswith(".git") else repo_url.rstrip(".git")
77
+ )
78
+ else:
79
+ _url = str(repo_url)
80
+ _code_info = {
81
+ "commitId": repo_info["commit_sha"],
82
+ "commitLink": os.path.join(_url, "commit", str(repo_info["commit_sha"])),
83
+ }
84
+ if not failed and returncode == 0 and isinstance(git_log_info, str):
85
+ _code_info["commitMessage"] = git_log_info.strip()
86
+
87
+ return _code_info
88
+
89
+ return None
@@ -0,0 +1,87 @@
1
+ import os
2
+ import json
3
+ from typing import Tuple, Union
4
+
5
+
6
+ class PerimeterExtractor:
7
+ @classmethod
8
+ def for_ob_cli(
9
+ cls, config_dir: str, profile: str
10
+ ) -> Union[Tuple[str, str], Tuple[None, None]]:
11
+ """
12
+ This function will be called when we are trying to extract the perimeter
13
+ via the ob cli's execution. We will rely on the following logic:
14
+ 1. check environment variables like OB_CURRENT_PERIMETER / OBP_PERIMETER
15
+ 2. run init config to extract the perimeter related configurations.
16
+
17
+ Returns
18
+ -------
19
+ Tuple[str, str] : Tuple containing perimeter name , API server url.
20
+ """
21
+ from outerbounds.utils import metaflowconfig
22
+
23
+ perimeter = None
24
+ api_server = None
25
+ if os.environ.get("OB_CURRENT_PERIMETER") or os.environ.get("OBP_PERIMETER"):
26
+ perimeter = os.environ.get("OB_CURRENT_PERIMETER") or os.environ.get(
27
+ "OBP_PERIMETER"
28
+ )
29
+
30
+ if os.environ.get("OBP_API_SERVER"):
31
+ api_server = os.environ.get("OBP_API_SERVER")
32
+
33
+ if perimeter is None or api_server is None:
34
+ metaflow_config = metaflowconfig.init_config(config_dir, profile)
35
+ perimeter = metaflow_config.get("OBP_PERIMETER")
36
+ api_server = metaflowconfig.get_sanitized_url_from_config(
37
+ config_dir, profile, "OBP_API_SERVER"
38
+ )
39
+
40
+ return perimeter, api_server # type: ignore
41
+
42
+ @classmethod
43
+ def during_programmatic_access(cls) -> Union[Tuple[str, str], Tuple[None, None]]:
44
+ from metaflow.metaflow_config_funcs import init_config
45
+
46
+ clean_url = (
47
+ lambda url: f"https://{url}".rstrip("/")
48
+ if not url.startswith("https://")
49
+ else url
50
+ )
51
+
52
+ config = init_config()
53
+ api_server, perimeter, integrations_url = None, None, None
54
+ perimeter = config.get(
55
+ "OBP_PERIMETER", os.environ.get("OBP_PERIMETER", perimeter)
56
+ )
57
+ if perimeter is None:
58
+ raise RuntimeError(
59
+ "Perimeter not found in metaflow config or environment variables"
60
+ )
61
+
62
+ api_server = config.get(
63
+ "OBP_API_SERVER", os.environ.get("OBP_API_SERVER", api_server)
64
+ )
65
+
66
+ if api_server is not None and not api_server.startswith("https://"):
67
+ api_server = clean_url(api_server)
68
+
69
+ if api_server is not None:
70
+ return perimeter, api_server
71
+
72
+ integrations_url = config.get(
73
+ "OBP_INTEGRATIONS_URL", os.environ.get("OBP_INTEGRATIONS_URL", None)
74
+ )
75
+
76
+ if integrations_url is not None and not integrations_url.startswith("https://"):
77
+ integrations_url = clean_url(integrations_url)
78
+
79
+ if integrations_url is not None:
80
+ api_server = integrations_url.rstrip("/integrations")
81
+
82
+ if api_server is None:
83
+ raise RuntimeError(
84
+ "API server not found in metaflow config or environment variables"
85
+ )
86
+
87
+ return perimeter, api_server
@@ -0,0 +1,164 @@
1
+ from typing import Dict
2
+
3
+ import base64
4
+ import json
5
+ import requests
6
+ import random
7
+ import time
8
+ import sys
9
+
10
+ from .utils import safe_requests_wrapper, TODOException
11
+
12
+
13
+ class OuterboundsSecretsException(Exception):
14
+ pass
15
+
16
+
17
+ class SecretNotFound(OuterboundsSecretsException):
18
+ pass
19
+
20
+
21
+ class OuterboundsSecretsApiResponse:
22
+ def __init__(self, response):
23
+ self.response = response
24
+
25
+ @property
26
+ def secret_resource_id(self):
27
+ return self.response["secret_resource_id"]
28
+
29
+ @property
30
+ def secret_backend_type(self):
31
+ return self.response["secret_backend_type"]
32
+
33
+
34
+ class SecretRetriever:
35
+ def get_secret_as_dict(self, secret_id, options={}, role=None):
36
+ """
37
+ Supports a special way of specifying secrets sources in outerbounds using the format:
38
+ @secrets(sources=["outerbounds.<integrations_name>"])
39
+
40
+ When invoked it makes a requests to the integrations secrets metadata endpoint on the
41
+ keywest server to get the cloud resource id for a secret. It then uses that to invoke
42
+ secrets manager on the core oss and returns the secrets.
43
+ """
44
+ headers = {"Content-Type": "application/json", "Connection": "keep-alive"}
45
+ perimeter, integrations_url = self._get_secret_configs()
46
+ integration_name = secret_id
47
+ request_payload = {
48
+ "perimeter_name": perimeter,
49
+ "integration_name": integration_name,
50
+ }
51
+ response = self._make_request(integrations_url, headers, request_payload)
52
+ secret_resource_id = response.secret_resource_id
53
+ secret_backend_type = response.secret_backend_type
54
+
55
+ from metaflow.plugins.secrets.secrets_decorator import (
56
+ get_secrets_backend_provider,
57
+ )
58
+
59
+ secrets_provider = get_secrets_backend_provider(secret_backend_type)
60
+ secret_dict = secrets_provider.get_secret_as_dict(
61
+ secret_resource_id, options={}, role=role
62
+ )
63
+
64
+ # Outerbounds stores secrets as binaries. Hence we expect the returned secret to be
65
+ # {<cloud-secret-name>: <base64 encoded full secret>}. We decode the secret here like:
66
+ # 1. decode the base64 encoded full secret
67
+ # 2. load the decoded secret as a json
68
+ # 3. decode the base64 encoded values in the dict
69
+ # 4. return the decoded dict
70
+ binary_secret = next(iter(secret_dict.values()))
71
+ return self._decode_secret(binary_secret)
72
+
73
+ def _is_base64_encoded(self, data):
74
+ try:
75
+ if isinstance(data, str):
76
+ # Check if the string can be base64 decoded
77
+ base64.b64decode(data).decode("utf-8")
78
+ return True
79
+ return False
80
+ except Exception:
81
+ return False
82
+
83
+ def _decode_secret(self, secret):
84
+ try:
85
+ result = {}
86
+ secret_str = secret
87
+ if self._is_base64_encoded(secret):
88
+ # we check if the secret string is base64 encoded because the returned secret from
89
+ # AWS secret manager is base64 encoded while the secret from GCP is not
90
+ secret_str = base64.b64decode(secret).decode("utf-8")
91
+
92
+ secret_dict = json.loads(secret_str)
93
+ for key, value in secret_dict.items():
94
+ result[key] = base64.b64decode(value).decode("utf-8")
95
+
96
+ return result
97
+ except Exception as e:
98
+ raise OuterboundsSecretsException(f"Error decoding secret: {e}")
99
+
100
+ def _get_secret_configs(self):
101
+ from metaflow_extensions.outerbounds.remote_config import init_config # type: ignore
102
+ from os import environ
103
+
104
+ conf = init_config()
105
+ if "OBP_PERIMETER" in conf:
106
+ perimeter = conf["OBP_PERIMETER"]
107
+ else:
108
+ # if the perimeter is not in metaflow config, try to get it from the environment
109
+ perimeter = environ.get("OBP_PERIMETER", "")
110
+
111
+ if "OBP_INTEGRATIONS_URL" in conf:
112
+ integrations_url = conf["OBP_INTEGRATIONS_URL"]
113
+ else:
114
+ # if the integrations is not in metaflow config, try to get it from the environment
115
+ integrations_url = environ.get("OBP_INTEGRATIONS_URL", "")
116
+
117
+ if not perimeter:
118
+ raise OuterboundsSecretsException(
119
+ "No perimeter set. Please make sure to run `outerbounds configure <...>` command which can be found on the Outerbounds UI or reach out to your Outerbounds support team."
120
+ )
121
+
122
+ if not integrations_url:
123
+ raise OuterboundsSecretsException(
124
+ "No integrations url set. Please notify your Outerbounds support team about this issue."
125
+ )
126
+
127
+ integrations_secrets_metadata_url = f"{integrations_url}/secrets/metadata"
128
+ return perimeter, integrations_secrets_metadata_url
129
+
130
+ def _make_request(self, url, headers: Dict, payload: Dict):
131
+ try:
132
+ from metaflow.metaflow_config import SERVICE_HEADERS
133
+
134
+ request_headers = {**headers, **(SERVICE_HEADERS or {})}
135
+ except ImportError:
136
+ raise OuterboundsSecretsException(
137
+ "Failed to create app: No Metaflow service headers found"
138
+ )
139
+
140
+ response = safe_requests_wrapper(
141
+ requests.get,
142
+ url,
143
+ data=json.dumps(payload),
144
+ headers=request_headers,
145
+ conn_error_retries=5,
146
+ retryable_status_codes=[409],
147
+ )
148
+ self._handle_error_response(response)
149
+ return OuterboundsSecretsApiResponse(response.json())
150
+
151
+ @staticmethod
152
+ def _handle_error_response(response: requests.Response):
153
+ if response.status_code >= 500:
154
+ raise OuterboundsSecretsException(
155
+ f"Server error: {response.text}. Please reach out to your Outerbounds support team."
156
+ )
157
+ status_code = response.status_code
158
+ if status_code == 404:
159
+ raise SecretNotFound(f"Secret not found: {response.text}")
160
+
161
+ if status_code >= 400:
162
+ raise OuterboundsSecretsException(
163
+ f"status_code={status_code}\t\n\t\t{response.text}"
164
+ )
@@ -0,0 +1,233 @@
1
+ import random
2
+ import time
3
+ import sys
4
+ import json
5
+ import requests
6
+ from typing import Optional
7
+
8
+ # This click import is not used to construct any ob
9
+ # package cli. Its used only for printing stuff.
10
+ # So we can use the static metaflow._vendor import path
11
+ from metaflow._vendor import click
12
+ from .app_config import CAPSULE_DEBUG
13
+ import sys
14
+ import threading
15
+ import time
16
+ import logging
17
+ import itertools
18
+ from typing import Union, Callable, Any, List
19
+
20
+ from ._vendor.spinner import (
21
+ Spinners,
22
+ )
23
+
24
+
25
+ class MultiStepSpinner:
26
+ """
27
+ A spinner that supports multi-step progress and configurable alignment.
28
+
29
+ Parameters
30
+ ----------
31
+ spinner : Spinners
32
+ Which spinner frames/interval to use.
33
+ text : str
34
+ Static text to display beside the spinner.
35
+ color : str, optional
36
+ Click color name.
37
+ align : {'left','right'}
38
+ Whether to render the spinner to the left (default) or right of the text.
39
+ """
40
+
41
+ def __init__(
42
+ self,
43
+ spinner: Spinners = Spinners.dots,
44
+ text: str = "",
45
+ color: Optional[str] = None,
46
+ align: str = "right",
47
+ file=sys.stdout,
48
+ ):
49
+ cfg = spinner.value
50
+ self.frames = cfg["frames"]
51
+ self.interval = float(cfg["interval"]) / 1000.0 # type: ignore
52
+ self.text = text
53
+ self.color = color
54
+ if align not in ("left", "right"):
55
+ raise ValueError("align must be 'left' or 'right'")
56
+ self.align = align
57
+ self._write_file = file
58
+ # precompute clear length: max frame width + space + text length
59
+ max_frame = max(self.frames, key=lambda x: len(x)) # type: ignore
60
+ self.clear_len = len(self.main_text) + len(max_frame) + 1
61
+
62
+ self._stop_evt = threading.Event()
63
+ self._pause_evt = threading.Event()
64
+ self._thread = None
65
+ self._write_lock = threading.Lock()
66
+
67
+ @property
68
+ def main_text(self):
69
+ # if self.text is a callable then call it
70
+ if callable(self.text):
71
+ return self.text()
72
+ return self.text
73
+
74
+ def _spin(self):
75
+ for frame in itertools.cycle(self.frames):
76
+ if self._stop_evt.is_set():
77
+ break
78
+ if self._pause_evt.is_set():
79
+ time.sleep(0.05)
80
+ continue
81
+
82
+ # ---- Core logging critical section ----
83
+ with self._write_lock:
84
+ symbol = click.style(frame, fg=self.color) if self.color else frame
85
+ if self.align == "left":
86
+ msg = f"{symbol} {self.main_text}"
87
+ else:
88
+ msg = f"{self.main_text} {symbol}"
89
+
90
+ click.echo(msg, nl=False, file=self._write_file)
91
+ click.echo("\r", nl=False, file=self._write_file)
92
+ self._write_file.flush()
93
+ # ---- End of critical section ----
94
+ time.sleep(self.interval)
95
+ # clear the line when done
96
+ self._clear_line()
97
+
98
+ def _clear_line(self):
99
+ with self._write_lock:
100
+ click.echo(" " * self.clear_len, nl=False, file=self._write_file)
101
+ click.echo("\r", nl=False, file=self._write_file)
102
+ self._write_file.flush()
103
+
104
+ def start(self):
105
+ if self._thread and self._thread.is_alive():
106
+ return
107
+ self._stop_evt.clear()
108
+ self._pause_evt.clear()
109
+ self._thread = threading.Thread(target=self._spin, daemon=True)
110
+ self._thread.start()
111
+
112
+ def stop(self):
113
+ self._stop_evt.set()
114
+ if self._thread:
115
+ self._thread.join()
116
+
117
+ def log(self, *messages: str):
118
+ """Pause the spinner, emit a ✔ + message, then resume."""
119
+ self._pause_evt.set()
120
+ self._clear_line()
121
+ # ---- Core logging critical section ----
122
+ with self._write_lock:
123
+ self._write_file.flush()
124
+ for message in messages:
125
+ click.echo(f"{message}", file=self._write_file, nl=True)
126
+ self._write_file.flush()
127
+ # ---- End of critical section ----
128
+ self._pause_evt.clear()
129
+
130
+ def __enter__(self):
131
+ self.start()
132
+ return self
133
+
134
+ def __exit__(self, exc_type, exc, tb):
135
+ self.stop()
136
+
137
+
138
+ class SpinnerLogHandler(logging.Handler):
139
+ def __init__(self, spinner: MultiStepSpinner, *args, **kwargs):
140
+ super().__init__(*args, **kwargs)
141
+ self.spinner = spinner
142
+
143
+ def emit(self, record):
144
+ msg = self.format(record)
145
+ self.spinner.log(msg)
146
+
147
+
148
+ class MaximumRetriesExceeded(Exception):
149
+ def __init__(self, url, method, status_code, text):
150
+ self.url = url
151
+ self.method = method
152
+ self.status_code = status_code
153
+ self.text = text
154
+
155
+ def __str__(self):
156
+ return f"Maximum retries exceeded for {self.url}[{self.method}] {self.status_code} {self.text}"
157
+
158
+
159
+ class TODOException(Exception):
160
+ pass
161
+
162
+
163
+ requests_funcs = [
164
+ requests.get,
165
+ requests.post,
166
+ requests.put,
167
+ requests.delete,
168
+ requests.patch,
169
+ requests.head,
170
+ requests.options,
171
+ ]
172
+
173
+
174
+ def safe_requests_wrapper(
175
+ requests_module_fn,
176
+ *args,
177
+ conn_error_retries=2,
178
+ retryable_status_codes=[409],
179
+ logger_fn=None,
180
+ **kwargs,
181
+ ):
182
+ """
183
+ There are two categories of errors that we need to handle when dealing with any API server.
184
+ 1. HTTP errors. These are are errors that are returned from the API server.
185
+ - How to handle retries for this case will be application specific.
186
+ 2. Errors when the API server may not be reachable (DNS resolution / network issues)
187
+ - In this scenario, we know that something external to the API server is going wrong causing the issue.
188
+ - Failing prematurely in the case might not be the best course of action since critical user jobs might crash on intermittent issues.
189
+ - So in this case, we can just plainly retry the request.
190
+
191
+ This function handles the second case. It's a simple wrapper to handle the retry logic for connection errors.
192
+ If this function is provided a `conn_error_retries` of 5, then the last retry will have waited 32 seconds.
193
+ Generally this is a safe enough number of retries after which we can assume that something is really broken. Until then,
194
+ there can be intermittent issues that would resolve themselves if we retry gracefully.
195
+ """
196
+ if requests_module_fn not in requests_funcs:
197
+ raise ValueError(
198
+ f"safe_requests_wrapper doesn't support {requests_module_fn.__name__}. You can only use the following functions: {requests_funcs}"
199
+ )
200
+
201
+ _num_retries = 0
202
+ noise = random.uniform(-0.5, 0.5)
203
+ response = None
204
+ while _num_retries < conn_error_retries:
205
+ try:
206
+ response = requests_module_fn(*args, **kwargs)
207
+ if response.status_code not in retryable_status_codes:
208
+ return response
209
+ if CAPSULE_DEBUG:
210
+ if logger_fn:
211
+ logger_fn(
212
+ f"[outerbounds-debug] safe_requests_wrapper: {response.url}[{requests_module_fn.__name__}] {response.status_code} {response.text}",
213
+ )
214
+ else:
215
+ print(
216
+ f"[outerbounds-debug] safe_requests_wrapper: {response.url}[{requests_module_fn.__name__}] {response.status_code} {response.text}",
217
+ file=sys.stderr,
218
+ )
219
+ _num_retries += 1
220
+ time.sleep((2 ** (_num_retries + 1)) + noise)
221
+ except requests.exceptions.ConnectionError:
222
+ if _num_retries <= conn_error_retries - 1:
223
+ # Exponential backoff with 2^(_num_retries+1) seconds
224
+ time.sleep((2 ** (_num_retries + 1)) + noise)
225
+ _num_retries += 1
226
+ else:
227
+ raise
228
+ raise MaximumRetriesExceeded(
229
+ response.url,
230
+ requests_module_fn.__name__,
231
+ response.status_code,
232
+ response.text,
233
+ )
@@ -0,0 +1,17 @@
1
+ import os
2
+ from typing import List
3
+ from .app_config import AppConfig, AppConfigError
4
+ from .secrets import SecretRetriever, SecretNotFound
5
+
6
+
7
+ def secrets_validator(secrets: List[str]):
8
+ secret_retriever = SecretRetriever()
9
+ for secret in secrets:
10
+ try:
11
+ secret_retriever.get_secret_as_dict(secret)
12
+ except SecretNotFound:
13
+ raise Exception(f"Secret named `{secret}` not found")
14
+
15
+
16
+ def run_validations(app_config: AppConfig):
17
+ pass