nextmv 1.0.0__py3-none-any.whl → 1.0.0.dev0__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.
- nextmv/__about__.py +1 -1
- nextmv/__entrypoint__.py +2 -1
- nextmv/__init__.py +4 -0
- nextmv/cli/CONTRIBUTING.md +40 -112
- nextmv/cli/cloud/__init__.py +0 -4
- nextmv/cli/cloud/acceptance/create.py +22 -20
- nextmv/cli/cloud/acceptance/delete.py +12 -8
- nextmv/cli/cloud/acceptance/get.py +10 -9
- nextmv/cli/cloud/acceptance/list.py +3 -3
- nextmv/cli/cloud/acceptance/update.py +6 -6
- nextmv/cli/cloud/account/__init__.py +3 -3
- nextmv/cli/cloud/account/create.py +11 -11
- nextmv/cli/cloud/account/delete.py +8 -7
- nextmv/cli/cloud/account/get.py +3 -3
- nextmv/cli/cloud/account/update.py +5 -5
- nextmv/cli/cloud/app/create.py +26 -25
- nextmv/cli/cloud/app/delete.py +7 -6
- nextmv/cli/cloud/app/exists.py +2 -2
- nextmv/cli/cloud/app/get.py +2 -2
- nextmv/cli/cloud/app/list.py +3 -3
- nextmv/cli/cloud/app/push.py +54 -349
- nextmv/cli/cloud/app/update.py +12 -12
- nextmv/cli/cloud/batch/create.py +28 -26
- nextmv/cli/cloud/batch/delete.py +10 -6
- nextmv/cli/cloud/batch/get.py +9 -9
- nextmv/cli/cloud/batch/list.py +3 -3
- nextmv/cli/cloud/batch/metadata.py +4 -4
- nextmv/cli/cloud/batch/update.py +6 -6
- nextmv/cli/cloud/data/__init__.py +1 -1
- nextmv/cli/cloud/data/upload.py +15 -15
- nextmv/cli/cloud/ensemble/__init__.py +0 -2
- nextmv/cli/cloud/ensemble/create.py +22 -21
- nextmv/cli/cloud/ensemble/delete.py +10 -6
- nextmv/cli/cloud/ensemble/get.py +4 -4
- nextmv/cli/cloud/ensemble/update.py +9 -9
- nextmv/cli/cloud/input_set/__init__.py +0 -2
- nextmv/cli/cloud/input_set/create.py +22 -22
- nextmv/cli/cloud/input_set/get.py +3 -3
- nextmv/cli/cloud/input_set/list.py +3 -3
- nextmv/cli/cloud/input_set/update.py +24 -24
- nextmv/cli/cloud/instance/create.py +15 -14
- nextmv/cli/cloud/instance/delete.py +7 -6
- nextmv/cli/cloud/instance/exists.py +2 -2
- nextmv/cli/cloud/instance/get.py +2 -2
- nextmv/cli/cloud/instance/list.py +3 -3
- nextmv/cli/cloud/instance/update.py +14 -14
- nextmv/cli/cloud/managed_input/create.py +16 -14
- nextmv/cli/cloud/managed_input/delete.py +8 -7
- nextmv/cli/cloud/managed_input/get.py +3 -3
- nextmv/cli/cloud/managed_input/list.py +3 -3
- nextmv/cli/cloud/managed_input/update.py +9 -9
- nextmv/cli/cloud/run/cancel.py +2 -2
- nextmv/cli/cloud/run/create.py +40 -34
- nextmv/cli/cloud/run/get.py +8 -8
- nextmv/cli/cloud/run/input.py +4 -4
- nextmv/cli/cloud/run/list.py +6 -6
- nextmv/cli/cloud/run/logs.py +10 -9
- nextmv/cli/cloud/run/metadata.py +4 -4
- nextmv/cli/cloud/run/track.py +33 -32
- nextmv/cli/cloud/scenario/create.py +21 -21
- nextmv/cli/cloud/scenario/delete.py +10 -6
- nextmv/cli/cloud/scenario/get.py +9 -9
- nextmv/cli/cloud/scenario/list.py +3 -3
- nextmv/cli/cloud/scenario/metadata.py +4 -4
- nextmv/cli/cloud/scenario/update.py +6 -6
- nextmv/cli/cloud/secrets/create.py +17 -17
- nextmv/cli/cloud/secrets/delete.py +10 -6
- nextmv/cli/cloud/secrets/get.py +4 -4
- nextmv/cli/cloud/secrets/list.py +3 -3
- nextmv/cli/cloud/secrets/update.py +20 -17
- nextmv/cli/cloud/upload/create.py +2 -2
- nextmv/cli/cloud/version/create.py +10 -9
- nextmv/cli/cloud/version/delete.py +7 -6
- nextmv/cli/cloud/version/exists.py +2 -2
- nextmv/cli/cloud/version/get.py +2 -2
- nextmv/cli/cloud/version/list.py +3 -3
- nextmv/cli/cloud/version/update.py +8 -8
- nextmv/cli/community/__init__.py +1 -1
- nextmv/cli/community/clone.py +204 -20
- nextmv/cli/community/list.py +125 -60
- nextmv/cli/configuration/config.py +10 -43
- nextmv/cli/configuration/create.py +7 -7
- nextmv/cli/configuration/delete.py +8 -8
- nextmv/cli/configuration/list.py +3 -3
- nextmv/cli/main.py +36 -26
- nextmv/cli/message.py +54 -71
- nextmv/cli/options.py +0 -28
- nextmv/cli/version.py +1 -1
- nextmv/cloud/__init__.py +38 -14
- nextmv/cloud/acceptance_test.py +65 -1
- nextmv/cloud/account.py +6 -1
- nextmv/cloud/application/__init__.py +75 -18
- nextmv/cloud/application/_acceptance.py +8 -13
- nextmv/cloud/application/_batch_scenario.py +19 -4
- nextmv/cloud/application/_input_set.py +6 -42
- nextmv/cloud/application/_instance.py +3 -3
- nextmv/cloud/application/_managed_input.py +2 -2
- nextmv/cloud/application/_version.py +3 -4
- nextmv/cloud/batch_experiment.py +1 -3
- nextmv/cloud/integration.py +4 -7
- nextmv/deprecated.py +3 -5
- nextmv/input.py +52 -0
- nextmv/local/runner.py +1 -1
- nextmv/model.py +11 -50
- nextmv/options.py +256 -11
- nextmv/output.py +62 -0
- nextmv/run.py +10 -1
- nextmv/status.py +51 -1
- {nextmv-1.0.0.dist-info → nextmv-1.0.0.dev0.dist-info}/METADATA +4 -5
- nextmv-1.0.0.dev0.dist-info/RECORD +158 -0
- nextmv/cli/cloud/ensemble/list.py +0 -63
- nextmv/cli/cloud/input_set/delete.py +0 -64
- nextmv/cli/cloud/shadow/__init__.py +0 -33
- nextmv/cli/cloud/shadow/create.py +0 -184
- nextmv/cli/cloud/shadow/delete.py +0 -64
- nextmv/cli/cloud/shadow/get.py +0 -61
- nextmv/cli/cloud/shadow/list.py +0 -63
- nextmv/cli/cloud/shadow/metadata.py +0 -66
- nextmv/cli/cloud/shadow/start.py +0 -43
- nextmv/cli/cloud/shadow/stop.py +0 -53
- nextmv/cli/cloud/shadow/update.py +0 -96
- nextmv/cli/cloud/switchback/__init__.py +0 -33
- nextmv/cli/cloud/switchback/create.py +0 -151
- nextmv/cli/cloud/switchback/delete.py +0 -64
- nextmv/cli/cloud/switchback/get.py +0 -62
- nextmv/cli/cloud/switchback/list.py +0 -63
- nextmv/cli/cloud/switchback/metadata.py +0 -68
- nextmv/cli/cloud/switchback/start.py +0 -43
- nextmv/cli/cloud/switchback/stop.py +0 -53
- nextmv/cli/cloud/switchback/update.py +0 -96
- nextmv/cli/confirm.py +0 -34
- nextmv/cloud/application/_shadow.py +0 -320
- nextmv/cloud/application/_switchback.py +0 -332
- nextmv/cloud/community.py +0 -446
- nextmv/cloud/shadow.py +0 -254
- nextmv/cloud/switchback.py +0 -228
- nextmv-1.0.0.dist-info/RECORD +0 -185
- nextmv-1.0.0.dist-info/entry_points.txt +0 -2
- {nextmv-1.0.0.dist-info → nextmv-1.0.0.dev0.dist-info}/WHEEL +0 -0
- {nextmv-1.0.0.dist-info → nextmv-1.0.0.dev0.dist-info}/licenses/LICENSE +0 -0
nextmv/cloud/community.py
DELETED
|
@@ -1,446 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
This module contains functionality for working with Nextmv community apps.
|
|
3
|
-
|
|
4
|
-
Community apps are pre-built decision models. They are maintained in the
|
|
5
|
-
following GitHub repository: https://github.com/nextmv-io/community-apps
|
|
6
|
-
|
|
7
|
-
Classes
|
|
8
|
-
-------
|
|
9
|
-
CommunityApp
|
|
10
|
-
Representation of a Nextmv Cloud Community App.
|
|
11
|
-
|
|
12
|
-
Functions
|
|
13
|
-
---------
|
|
14
|
-
list_community_apps
|
|
15
|
-
List the available Nextmv community apps.
|
|
16
|
-
clone_community_app
|
|
17
|
-
Clone a community app locally.
|
|
18
|
-
"""
|
|
19
|
-
|
|
20
|
-
import os
|
|
21
|
-
import shutil
|
|
22
|
-
import sys
|
|
23
|
-
import tarfile
|
|
24
|
-
import tempfile
|
|
25
|
-
from collections.abc import Callable
|
|
26
|
-
from typing import Any
|
|
27
|
-
|
|
28
|
-
import requests
|
|
29
|
-
import rich
|
|
30
|
-
import yaml
|
|
31
|
-
from pydantic import AliasChoices, Field
|
|
32
|
-
|
|
33
|
-
from nextmv.base_model import BaseModel
|
|
34
|
-
from nextmv.cloud.client import Client
|
|
35
|
-
from nextmv.logger import log
|
|
36
|
-
|
|
37
|
-
# Helpful constants.
|
|
38
|
-
LATEST_VERSION = "latest"
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
class CommunityApp(BaseModel):
|
|
42
|
-
"""
|
|
43
|
-
Information about a Nextmv community app.
|
|
44
|
-
|
|
45
|
-
You can import the `CommunityApp` class directly from `cloud`:
|
|
46
|
-
|
|
47
|
-
```python
|
|
48
|
-
from nextmv.cloud import CommunityApp
|
|
49
|
-
```
|
|
50
|
-
|
|
51
|
-
Parameters
|
|
52
|
-
----------
|
|
53
|
-
app_versions : list[str]
|
|
54
|
-
Available versions of the community app.
|
|
55
|
-
description : str
|
|
56
|
-
Description of the community app.
|
|
57
|
-
latest_app_version : str
|
|
58
|
-
The latest version of the community app.
|
|
59
|
-
latest_marketplace_version : str
|
|
60
|
-
The latest version of the community app in the Nextmv Marketplace.
|
|
61
|
-
marketplace_versions : list[str]
|
|
62
|
-
Available versions of the community app in the Nextmv Marketplace.
|
|
63
|
-
name : str
|
|
64
|
-
Name of the community app.
|
|
65
|
-
app_type : str
|
|
66
|
-
Type of the community app.
|
|
67
|
-
"""
|
|
68
|
-
|
|
69
|
-
description: str
|
|
70
|
-
"""Description of the community app."""
|
|
71
|
-
name: str
|
|
72
|
-
"""Name of the community app."""
|
|
73
|
-
app_type: str = Field(
|
|
74
|
-
serialization_alias="type",
|
|
75
|
-
validation_alias=AliasChoices("type", "app_type"),
|
|
76
|
-
)
|
|
77
|
-
"""Type of the community app."""
|
|
78
|
-
|
|
79
|
-
app_versions: list[str] | None = None
|
|
80
|
-
"""Available versions of the community app."""
|
|
81
|
-
latest_app_version: str | None = None
|
|
82
|
-
"""The latest version of the community app."""
|
|
83
|
-
latest_marketplace_version: str | None = None
|
|
84
|
-
"""The latest version of the community app in the Nextmv Marketplace."""
|
|
85
|
-
marketplace_versions: list[str] | None = None
|
|
86
|
-
"""Available versions of the community app in the Nextmv Marketplace."""
|
|
87
|
-
|
|
88
|
-
def has_version(self, version: str) -> bool:
|
|
89
|
-
"""
|
|
90
|
-
Check if the community app has the specified version.
|
|
91
|
-
|
|
92
|
-
Parameters
|
|
93
|
-
----------
|
|
94
|
-
version : str
|
|
95
|
-
The version to check.
|
|
96
|
-
|
|
97
|
-
Returns
|
|
98
|
-
-------
|
|
99
|
-
bool
|
|
100
|
-
True if the app has the specified version, False otherwise.
|
|
101
|
-
"""
|
|
102
|
-
|
|
103
|
-
if version == LATEST_VERSION:
|
|
104
|
-
version = self.latest_app_version
|
|
105
|
-
|
|
106
|
-
if self.app_versions is not None and version in self.app_versions:
|
|
107
|
-
return True
|
|
108
|
-
|
|
109
|
-
return False
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
def list_community_apps(client: Client) -> list[CommunityApp]:
|
|
113
|
-
"""
|
|
114
|
-
List the available Nextmv community apps.
|
|
115
|
-
|
|
116
|
-
You can import the `list_community_apps` function directly from `cloud`:
|
|
117
|
-
|
|
118
|
-
```python
|
|
119
|
-
from nextmv.cloud import list_community_apps
|
|
120
|
-
```
|
|
121
|
-
|
|
122
|
-
Parameters
|
|
123
|
-
----------
|
|
124
|
-
client : Client
|
|
125
|
-
The Nextmv Cloud client to use for the request.
|
|
126
|
-
|
|
127
|
-
Returns
|
|
128
|
-
-------
|
|
129
|
-
list[CommunityApp]
|
|
130
|
-
A list of available community apps.
|
|
131
|
-
"""
|
|
132
|
-
|
|
133
|
-
manifest = _download_manifest(client)
|
|
134
|
-
dict_apps = manifest.get("apps", [])
|
|
135
|
-
apps = [CommunityApp.from_dict(app) for app in dict_apps]
|
|
136
|
-
|
|
137
|
-
return apps
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
def clone_community_app(
|
|
141
|
-
client: Client,
|
|
142
|
-
app: str,
|
|
143
|
-
directory: str | None = None,
|
|
144
|
-
version: str | None = LATEST_VERSION,
|
|
145
|
-
verbose: bool = False,
|
|
146
|
-
rich_print: bool = False,
|
|
147
|
-
) -> None:
|
|
148
|
-
"""
|
|
149
|
-
Clone a community app locally.
|
|
150
|
-
|
|
151
|
-
By default, the `latest` version will be used. You can
|
|
152
|
-
specify a version with the `version` parameter, and customize the output
|
|
153
|
-
directory with the `directory` parameter. If you want to list the available
|
|
154
|
-
apps, use the `list_community_apps` function.
|
|
155
|
-
|
|
156
|
-
You can import the `clone_community_app` function directly from `cloud`:
|
|
157
|
-
|
|
158
|
-
```python
|
|
159
|
-
from nextmv.cloud import clone_community_app
|
|
160
|
-
```
|
|
161
|
-
|
|
162
|
-
Parameters
|
|
163
|
-
----------
|
|
164
|
-
client : Client
|
|
165
|
-
The Nextmv Cloud client to use for the request.
|
|
166
|
-
app : str
|
|
167
|
-
The name of the community app to clone.
|
|
168
|
-
directory : str | None, optional
|
|
169
|
-
The directory in which to clone the app. Default is the name of the app at current directory.
|
|
170
|
-
version : str | None, optional
|
|
171
|
-
The version of the community app to clone. Default is `latest`.
|
|
172
|
-
verbose : bool, optional
|
|
173
|
-
Whether to print verbose output.
|
|
174
|
-
rich_print : bool, optional
|
|
175
|
-
Whether to use rich printing for output messages.
|
|
176
|
-
"""
|
|
177
|
-
comm_app = _find_app(client, app)
|
|
178
|
-
|
|
179
|
-
if version is not None and version == "":
|
|
180
|
-
raise ValueError("`version` cannot be an empty string.")
|
|
181
|
-
|
|
182
|
-
if not comm_app.has_version(version):
|
|
183
|
-
raise ValueError(f"Community app '{app}' does not have version '{version}'.")
|
|
184
|
-
|
|
185
|
-
original_version = version
|
|
186
|
-
if version == LATEST_VERSION:
|
|
187
|
-
version = comm_app.latest_app_version
|
|
188
|
-
|
|
189
|
-
# Clean and normalize directory path in an OS-independent way
|
|
190
|
-
if directory is not None and directory != "":
|
|
191
|
-
destination = os.path.normpath(directory)
|
|
192
|
-
else:
|
|
193
|
-
destination = app
|
|
194
|
-
|
|
195
|
-
full_destination = _get_valid_path(destination, os.stat)
|
|
196
|
-
os.makedirs(full_destination, exist_ok=True)
|
|
197
|
-
|
|
198
|
-
tarball = f"{app}_{version}.tar.gz"
|
|
199
|
-
s3_file_path = f"{app}/{version}/{tarball}"
|
|
200
|
-
downloaded_object = _download_object(
|
|
201
|
-
client=client,
|
|
202
|
-
file=s3_file_path,
|
|
203
|
-
path="community-apps",
|
|
204
|
-
output_dir=full_destination,
|
|
205
|
-
output_file=tarball,
|
|
206
|
-
)
|
|
207
|
-
|
|
208
|
-
# Extract the tarball to a temporary directory to handle nested structure
|
|
209
|
-
with tempfile.TemporaryDirectory() as temp_dir:
|
|
210
|
-
with tarfile.open(downloaded_object, "r:gz") as tar:
|
|
211
|
-
tar.extractall(path=temp_dir)
|
|
212
|
-
|
|
213
|
-
# Find the extracted directory (typically the app name)
|
|
214
|
-
extracted_items = os.listdir(temp_dir)
|
|
215
|
-
if len(extracted_items) == 1 and os.path.isdir(os.path.join(temp_dir, extracted_items[0])):
|
|
216
|
-
# Move contents from the extracted directory to full_destination
|
|
217
|
-
extracted_dir = os.path.join(temp_dir, extracted_items[0])
|
|
218
|
-
for item in os.listdir(extracted_dir):
|
|
219
|
-
shutil.move(os.path.join(extracted_dir, item), full_destination)
|
|
220
|
-
else:
|
|
221
|
-
# If structure is unexpected, move everything directly
|
|
222
|
-
for item in extracted_items:
|
|
223
|
-
shutil.move(os.path.join(temp_dir, item), full_destination)
|
|
224
|
-
|
|
225
|
-
# Remove the tarball after extraction
|
|
226
|
-
os.remove(downloaded_object)
|
|
227
|
-
|
|
228
|
-
if not verbose:
|
|
229
|
-
return
|
|
230
|
-
|
|
231
|
-
if rich_print:
|
|
232
|
-
rich.print(
|
|
233
|
-
f":white_check_mark: Successfully cloned the [magenta]{app}[/magenta] community app, "
|
|
234
|
-
f"using version [magenta]{original_version}[/magenta] in path: [magenta]{full_destination}[/magenta].",
|
|
235
|
-
file=sys.stderr,
|
|
236
|
-
)
|
|
237
|
-
return
|
|
238
|
-
|
|
239
|
-
log(
|
|
240
|
-
f"✅ Successfully cloned the {app} community app, using version {original_version} in path: {full_destination}."
|
|
241
|
-
)
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
def _download_manifest(client: Client) -> dict[str, Any]:
|
|
245
|
-
"""
|
|
246
|
-
Downloads and returns the community apps manifest.
|
|
247
|
-
|
|
248
|
-
Parameters
|
|
249
|
-
----------
|
|
250
|
-
client : Client
|
|
251
|
-
The Nextmv Cloud client to use for the request.
|
|
252
|
-
|
|
253
|
-
Returns
|
|
254
|
-
-------
|
|
255
|
-
dict[str, Any]
|
|
256
|
-
The community apps manifest as a dictionary.
|
|
257
|
-
|
|
258
|
-
Raises
|
|
259
|
-
requests.HTTPError
|
|
260
|
-
If the response status code is not 2xx.
|
|
261
|
-
"""
|
|
262
|
-
|
|
263
|
-
response = _download_file(client=client, directory="community-apps", file="manifest.yml")
|
|
264
|
-
manifest = yaml.safe_load(response.text)
|
|
265
|
-
|
|
266
|
-
return manifest
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
def _download_file(
|
|
270
|
-
client: Client,
|
|
271
|
-
directory: str,
|
|
272
|
-
file: str,
|
|
273
|
-
) -> requests.Response:
|
|
274
|
-
"""
|
|
275
|
-
Gets a file from an internal bucket and return it.
|
|
276
|
-
|
|
277
|
-
Parameters
|
|
278
|
-
----------
|
|
279
|
-
client : Client
|
|
280
|
-
The Nextmv Cloud client to use for the request.
|
|
281
|
-
directory : str
|
|
282
|
-
The directory in the bucket where the file is located.
|
|
283
|
-
file : str
|
|
284
|
-
The name of the file to download.
|
|
285
|
-
|
|
286
|
-
Returns
|
|
287
|
-
-------
|
|
288
|
-
requests.Response
|
|
289
|
-
The response object containing the file data.
|
|
290
|
-
|
|
291
|
-
Raises
|
|
292
|
-
requests.HTTPError
|
|
293
|
-
If the response status code is not 2xx.
|
|
294
|
-
"""
|
|
295
|
-
|
|
296
|
-
# Request the download URL for the file.
|
|
297
|
-
response = client.request(
|
|
298
|
-
method="GET",
|
|
299
|
-
endpoint="v0/internal/tools",
|
|
300
|
-
headers=client.headers | {"request-source": "cli"}, # Pass `client.headers` to preserve auth.
|
|
301
|
-
query_params={"file": f"{directory}/{file}"},
|
|
302
|
-
)
|
|
303
|
-
|
|
304
|
-
# Use the URL obtained to download the file.
|
|
305
|
-
body = response.json()
|
|
306
|
-
download_response = client.request(
|
|
307
|
-
method="GET",
|
|
308
|
-
endpoint=body.get("url"),
|
|
309
|
-
headers={"Content-Type": "application/json"},
|
|
310
|
-
)
|
|
311
|
-
|
|
312
|
-
return download_response
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
def _download_object(client: Client, file: str, path: str, output_dir: str, output_file: str) -> str:
|
|
316
|
-
"""
|
|
317
|
-
Downloads an object from the internal bucket and saves it to the specified
|
|
318
|
-
output directory.
|
|
319
|
-
|
|
320
|
-
Parameters
|
|
321
|
-
----------
|
|
322
|
-
client : Client
|
|
323
|
-
The Nextmv Cloud client to use for the request.
|
|
324
|
-
file : str
|
|
325
|
-
The name of the file to download.
|
|
326
|
-
path : str
|
|
327
|
-
The directory in the bucket where the file is located.
|
|
328
|
-
output_dir : str
|
|
329
|
-
The local directory where the file will be saved.
|
|
330
|
-
output_file : str
|
|
331
|
-
The name of the output file.
|
|
332
|
-
|
|
333
|
-
Returns
|
|
334
|
-
-------
|
|
335
|
-
str
|
|
336
|
-
The path to the downloaded file.
|
|
337
|
-
"""
|
|
338
|
-
|
|
339
|
-
response = _download_file(client=client, directory=path, file=file)
|
|
340
|
-
file_name = os.path.join(output_dir, output_file)
|
|
341
|
-
|
|
342
|
-
with open(file_name, "wb") as f:
|
|
343
|
-
f.write(response.content)
|
|
344
|
-
|
|
345
|
-
return file_name
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
def _get_valid_path(path: str, stat_fn: Callable[[str], os.stat_result], ending: str = "") -> str:
|
|
349
|
-
"""
|
|
350
|
-
Validates and returns a non-existing path. If the path exists,
|
|
351
|
-
it will append a number to the path and return it. If the path does not
|
|
352
|
-
exist, it will return the path as is.
|
|
353
|
-
|
|
354
|
-
The ending parameter is used to check if the path ends with a specific
|
|
355
|
-
string. This is useful to specify if it is a file (like foo.json, in which
|
|
356
|
-
case the next iteration is foo-1.json) or a directory (like foo, in which
|
|
357
|
-
case the next iteration is foo-1).
|
|
358
|
-
|
|
359
|
-
Parameters
|
|
360
|
-
----------
|
|
361
|
-
path : str
|
|
362
|
-
The initial path to validate.
|
|
363
|
-
stat_fn : Callable[[str], os.stat_result]
|
|
364
|
-
A function that takes a path and returns its stat result.
|
|
365
|
-
ending : str, optional
|
|
366
|
-
The expected ending of the path (e.g., file extension), by default "".
|
|
367
|
-
|
|
368
|
-
Returns
|
|
369
|
-
-------
|
|
370
|
-
str
|
|
371
|
-
A valid, non-existing path.
|
|
372
|
-
|
|
373
|
-
Raises
|
|
374
|
-
------
|
|
375
|
-
RuntimeError
|
|
376
|
-
If an unexpected error occurs during path validation
|
|
377
|
-
"""
|
|
378
|
-
base_name = os.path.basename(path)
|
|
379
|
-
name_without_ending = base_name.removesuffix(ending) if ending else base_name
|
|
380
|
-
|
|
381
|
-
while True:
|
|
382
|
-
try:
|
|
383
|
-
stat_fn(path)
|
|
384
|
-
# If we get here, the path exists
|
|
385
|
-
# Get folder/file name number, increase it and create new path
|
|
386
|
-
name = os.path.basename(path)
|
|
387
|
-
|
|
388
|
-
# Get folder/file name number
|
|
389
|
-
parts = name.split("-")
|
|
390
|
-
last = parts[-1].removesuffix(ending) if ending else parts[-1]
|
|
391
|
-
|
|
392
|
-
# Save last folder name index to be changed
|
|
393
|
-
i = path.rfind(name)
|
|
394
|
-
|
|
395
|
-
try:
|
|
396
|
-
num = int(last)
|
|
397
|
-
# Increase number and create new path
|
|
398
|
-
if ending:
|
|
399
|
-
temp_path = path[:i] + f"{name_without_ending}-{num + 1}{ending}"
|
|
400
|
-
else:
|
|
401
|
-
temp_path = path[:i] + f"{base_name}-{num + 1}"
|
|
402
|
-
path = temp_path
|
|
403
|
-
except ValueError:
|
|
404
|
-
# If there is no number, add it
|
|
405
|
-
if ending:
|
|
406
|
-
temp_path = path[:i] + f"{name_without_ending}-1{ending}"
|
|
407
|
-
else:
|
|
408
|
-
temp_path = path[:i] + f"{name}-1"
|
|
409
|
-
path = temp_path
|
|
410
|
-
|
|
411
|
-
except FileNotFoundError:
|
|
412
|
-
# Path doesn't exist, we can use it
|
|
413
|
-
return path
|
|
414
|
-
except Exception as e:
|
|
415
|
-
# Re-raise unexpected errors
|
|
416
|
-
raise RuntimeError(f"An unexpected error occurred while validating the path: {path} ") from e
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
def _find_app(client: Client, app: str) -> CommunityApp:
|
|
420
|
-
"""
|
|
421
|
-
Finds and returns a community app from the manifest by its name.
|
|
422
|
-
|
|
423
|
-
Parameters
|
|
424
|
-
----------
|
|
425
|
-
client : Client
|
|
426
|
-
The Nextmv Cloud client to use for the request.
|
|
427
|
-
app : str
|
|
428
|
-
The name of the community app to find.
|
|
429
|
-
|
|
430
|
-
Returns
|
|
431
|
-
-------
|
|
432
|
-
CommunityApp
|
|
433
|
-
The community app if found.
|
|
434
|
-
|
|
435
|
-
Raises
|
|
436
|
-
------
|
|
437
|
-
ValueError
|
|
438
|
-
If the community app is not found.
|
|
439
|
-
"""
|
|
440
|
-
|
|
441
|
-
comm_apps = list_community_apps(client)
|
|
442
|
-
for comm_app in comm_apps:
|
|
443
|
-
if comm_app.name == app:
|
|
444
|
-
return comm_app
|
|
445
|
-
|
|
446
|
-
raise ValueError(f"Community app '{app}' not found.")
|
nextmv/cloud/shadow.py
DELETED
|
@@ -1,254 +0,0 @@
|
|
|
1
|
-
"""
|
|
2
|
-
Classes for working with Nextmv Cloud shadow tests.
|
|
3
|
-
|
|
4
|
-
This module provides classes for interacting with shadow tests in Nextmv Cloud.
|
|
5
|
-
It details the core data structures for these types of experiments.
|
|
6
|
-
|
|
7
|
-
Classes
|
|
8
|
-
-------
|
|
9
|
-
TestComparison
|
|
10
|
-
A structure to define comparison parameters for tests.
|
|
11
|
-
StartEvents
|
|
12
|
-
A structure to define start events for tests.
|
|
13
|
-
TerminationEvents
|
|
14
|
-
A structure to define termination events for tests.
|
|
15
|
-
ShadowTestMetadata
|
|
16
|
-
Metadata for a Nextmv Cloud shadow test.
|
|
17
|
-
ShadowTest
|
|
18
|
-
A Nextmv Cloud shadow test definition.
|
|
19
|
-
"""
|
|
20
|
-
|
|
21
|
-
from datetime import datetime
|
|
22
|
-
from enum import Enum
|
|
23
|
-
from typing import Any
|
|
24
|
-
|
|
25
|
-
from pydantic import AliasChoices, Field
|
|
26
|
-
|
|
27
|
-
from nextmv.base_model import BaseModel
|
|
28
|
-
from nextmv.cloud.batch_experiment import ExperimentStatus
|
|
29
|
-
from nextmv.run import Run
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
class TestComparison(BaseModel):
|
|
33
|
-
"""
|
|
34
|
-
A structure to define comparison parameters for tests.
|
|
35
|
-
|
|
36
|
-
You can import the `TestComparison` class directly from `cloud`:
|
|
37
|
-
|
|
38
|
-
```python
|
|
39
|
-
from nextmv.cloud import TestComparison
|
|
40
|
-
```
|
|
41
|
-
|
|
42
|
-
Parameters
|
|
43
|
-
----------
|
|
44
|
-
baseline_instance_id : str
|
|
45
|
-
ID of the baseline instance for comparison.
|
|
46
|
-
candidate_instance_ids : list[str]
|
|
47
|
-
List of candidate instance IDs to compare against the baseline.
|
|
48
|
-
"""
|
|
49
|
-
|
|
50
|
-
baseline_instance_id: str
|
|
51
|
-
"""ID of the baseline instance for comparison."""
|
|
52
|
-
candidate_instance_ids: list[str]
|
|
53
|
-
"""List of candidate instance IDs to compare against the baseline."""
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
class StartEvents(BaseModel):
|
|
57
|
-
"""
|
|
58
|
-
A structure to define start events for tests.
|
|
59
|
-
|
|
60
|
-
You can import the `StartEvents` class directly from `cloud`:
|
|
61
|
-
|
|
62
|
-
```python
|
|
63
|
-
from nextmv.cloud import StartEvents
|
|
64
|
-
```
|
|
65
|
-
|
|
66
|
-
Parameters
|
|
67
|
-
----------
|
|
68
|
-
time : datetime, optional
|
|
69
|
-
Scheduled time for the test to start.
|
|
70
|
-
"""
|
|
71
|
-
|
|
72
|
-
time: datetime | None = None
|
|
73
|
-
"""Scheduled time for the test to start."""
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
class TerminationEvents(BaseModel):
|
|
77
|
-
"""
|
|
78
|
-
A structure to define termination events for tests.
|
|
79
|
-
|
|
80
|
-
You can import the `TerminationEvents` class directly from `cloud`:
|
|
81
|
-
|
|
82
|
-
```python
|
|
83
|
-
from nextmv.cloud import TerminationEvents
|
|
84
|
-
```
|
|
85
|
-
|
|
86
|
-
Parameters
|
|
87
|
-
----------
|
|
88
|
-
time : datetime, optional
|
|
89
|
-
Scheduled time for the test to terminate.
|
|
90
|
-
"""
|
|
91
|
-
|
|
92
|
-
maximum_runs: int
|
|
93
|
-
"""
|
|
94
|
-
Maximum number of runs for the test. Value must be between 1 and 300.
|
|
95
|
-
"""
|
|
96
|
-
time: datetime | None = None
|
|
97
|
-
"""
|
|
98
|
-
Scheduled time for the test to terminate. A zero value means no
|
|
99
|
-
limit.
|
|
100
|
-
"""
|
|
101
|
-
|
|
102
|
-
def model_post_init(self, __context):
|
|
103
|
-
if self.maximum_runs < 1 or self.maximum_runs > 300:
|
|
104
|
-
raise ValueError("maximum_runs must be between 1 and 300")
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
class ShadowTestMetadata(BaseModel):
|
|
108
|
-
"""
|
|
109
|
-
Metadata for a Nextmv Cloud shadow test.
|
|
110
|
-
|
|
111
|
-
You can import the `ShadowTestMetadata` class directly from `cloud`:
|
|
112
|
-
|
|
113
|
-
```python
|
|
114
|
-
from nextmv.cloud import ShadowTestMetadata
|
|
115
|
-
```
|
|
116
|
-
|
|
117
|
-
Parameters
|
|
118
|
-
----------
|
|
119
|
-
shadow_test_id : str, optional
|
|
120
|
-
The unique identifier of the shadow test.
|
|
121
|
-
name : str, optional
|
|
122
|
-
Name of the shadow test.
|
|
123
|
-
description : str, optional
|
|
124
|
-
Description of the shadow test.
|
|
125
|
-
app_id : str, optional
|
|
126
|
-
ID of the application to which the shadow test belongs.
|
|
127
|
-
created_at : datetime, optional
|
|
128
|
-
Creation date of the shadow test.
|
|
129
|
-
updated_at : datetime, optional
|
|
130
|
-
Last update date of the shadow test.
|
|
131
|
-
status : ExperimentStatus, optional
|
|
132
|
-
The current status of the shadow test.
|
|
133
|
-
"""
|
|
134
|
-
|
|
135
|
-
shadow_test_id: str | None = Field(
|
|
136
|
-
serialization_alias="id",
|
|
137
|
-
validation_alias=AliasChoices("id", "shadow_test_id"),
|
|
138
|
-
default=None,
|
|
139
|
-
)
|
|
140
|
-
"""The unique identifier of the shadow test."""
|
|
141
|
-
name: str | None = None
|
|
142
|
-
"""Name of the shadow test."""
|
|
143
|
-
description: str | None = None
|
|
144
|
-
"""Description of the shadow test."""
|
|
145
|
-
app_id: str | None = None
|
|
146
|
-
"""ID of the application to which the shadow test belongs."""
|
|
147
|
-
created_at: datetime | None = None
|
|
148
|
-
"""Creation date of the shadow test."""
|
|
149
|
-
updated_at: datetime | None = None
|
|
150
|
-
"""Last update date of the shadow test."""
|
|
151
|
-
status: ExperimentStatus | None = None
|
|
152
|
-
"""The current status of the shadow test."""
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
# This class uses some fields defined in ShadowTestMetadata. We are not
|
|
156
|
-
# using inheritance to help the user understand the full structure when using
|
|
157
|
-
# tools like intellisense.
|
|
158
|
-
class ShadowTest(BaseModel):
|
|
159
|
-
"""
|
|
160
|
-
A Nextmv Cloud shadow test definition.
|
|
161
|
-
|
|
162
|
-
A shadow test is a type of experiment where runs are executed in parallel
|
|
163
|
-
to compare different instances.
|
|
164
|
-
|
|
165
|
-
You can import the `ShadowTest` class directly from `cloud`:
|
|
166
|
-
|
|
167
|
-
```python
|
|
168
|
-
from nextmv.cloud import ShadowTest
|
|
169
|
-
```
|
|
170
|
-
|
|
171
|
-
Parameters
|
|
172
|
-
----------
|
|
173
|
-
shadow_test_id : str, optional
|
|
174
|
-
The unique identifier of the shadow test.
|
|
175
|
-
name : str, optional
|
|
176
|
-
Name of the shadow test.
|
|
177
|
-
description : str, optional
|
|
178
|
-
Description of the shadow test.
|
|
179
|
-
app_id : str, optional
|
|
180
|
-
ID of the application to which the shadow test belongs.
|
|
181
|
-
created_at : datetime, optional
|
|
182
|
-
Creation date of the shadow test.
|
|
183
|
-
updated_at : datetime, optional
|
|
184
|
-
Last update date of the shadow test.
|
|
185
|
-
status : ExperimentStatus, optional
|
|
186
|
-
The current status of the shadow test.
|
|
187
|
-
completed_at : datetime, optional
|
|
188
|
-
Completion date of the shadow test, if applicable.
|
|
189
|
-
comparisons : list[TestComparison], optional
|
|
190
|
-
List of test comparisons defined in the shadow test.
|
|
191
|
-
start_events : StartEvents, optional
|
|
192
|
-
Start events for the shadow test.
|
|
193
|
-
termination_events : TerminationEvents, optional
|
|
194
|
-
Termination events for the shadow test.
|
|
195
|
-
grouped_distributional_summaries : list[dict[str, Any]], optional
|
|
196
|
-
Grouped distributional summaries of the shadow test.
|
|
197
|
-
runs : list[Run], optional
|
|
198
|
-
List of runs in the shadow test.
|
|
199
|
-
"""
|
|
200
|
-
|
|
201
|
-
shadow_test_id: str | None = Field(
|
|
202
|
-
serialization_alias="id",
|
|
203
|
-
validation_alias=AliasChoices("id", "shadow_test_id"),
|
|
204
|
-
default=None,
|
|
205
|
-
)
|
|
206
|
-
"""The unique identifier of the shadow test."""
|
|
207
|
-
name: str | None = None
|
|
208
|
-
"""Name of the shadow test."""
|
|
209
|
-
description: str | None = None
|
|
210
|
-
"""Description of the shadow test."""
|
|
211
|
-
app_id: str | None = None
|
|
212
|
-
"""ID of the application to which the shadow test belongs."""
|
|
213
|
-
created_at: datetime | None = None
|
|
214
|
-
"""Creation date of the shadow test."""
|
|
215
|
-
updated_at: datetime | None = None
|
|
216
|
-
"""Last update date of the shadow test."""
|
|
217
|
-
status: ExperimentStatus | None = None
|
|
218
|
-
"""The current status of the shadow test."""
|
|
219
|
-
completed_at: datetime | None = None
|
|
220
|
-
"""Completion date of the shadow test, if applicable."""
|
|
221
|
-
comparisons: list[TestComparison] | None = None
|
|
222
|
-
"""List of test comparisons defined in the shadow test."""
|
|
223
|
-
start_events: StartEvents | None = None
|
|
224
|
-
"""Start events for the shadow test."""
|
|
225
|
-
termination_events: TerminationEvents | None = None
|
|
226
|
-
"""Termination events for the shadow test."""
|
|
227
|
-
grouped_distributional_summaries: list[dict[str, Any]] | None = None
|
|
228
|
-
"""Grouped distributional summaries of the shadow test."""
|
|
229
|
-
runs: list[Run] | None = None
|
|
230
|
-
"""List of runs in the shadow test."""
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
class StopIntent(str, Enum):
|
|
234
|
-
"""
|
|
235
|
-
Intent for stopping a shadow test.
|
|
236
|
-
|
|
237
|
-
You can import the `StopIntent` class directly from `cloud`:
|
|
238
|
-
|
|
239
|
-
```python
|
|
240
|
-
from nextmv.cloud import StopIntent
|
|
241
|
-
```
|
|
242
|
-
|
|
243
|
-
Attributes
|
|
244
|
-
----------
|
|
245
|
-
complete : str
|
|
246
|
-
The test is marked as complete.
|
|
247
|
-
cancel : str
|
|
248
|
-
The test is canceled.
|
|
249
|
-
"""
|
|
250
|
-
|
|
251
|
-
complete = "complete"
|
|
252
|
-
"""The test is marked as complete."""
|
|
253
|
-
cancel = "cancel"
|
|
254
|
-
"""The test is canceled."""
|