ob-metaflow 2.15.5.1__py2.py3-none-any.whl → 2.15.7.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 might be problematic. Click here for more details.

Files changed (53) hide show
  1. metaflow/_vendor/typeguard/_checkers.py +259 -95
  2. metaflow/_vendor/typeguard/_config.py +4 -4
  3. metaflow/_vendor/typeguard/_decorators.py +8 -12
  4. metaflow/_vendor/typeguard/_functions.py +33 -32
  5. metaflow/_vendor/typeguard/_pytest_plugin.py +40 -13
  6. metaflow/_vendor/typeguard/_suppression.py +3 -5
  7. metaflow/_vendor/typeguard/_transformer.py +84 -48
  8. metaflow/_vendor/typeguard/_union_transformer.py +1 -0
  9. metaflow/_vendor/typeguard/_utils.py +13 -9
  10. metaflow/_vendor/typing_extensions.py +1088 -500
  11. metaflow/_vendor/v3_7/__init__.py +1 -0
  12. metaflow/_vendor/v3_7/importlib_metadata/__init__.py +1063 -0
  13. metaflow/_vendor/v3_7/importlib_metadata/_adapters.py +68 -0
  14. metaflow/_vendor/v3_7/importlib_metadata/_collections.py +30 -0
  15. metaflow/_vendor/v3_7/importlib_metadata/_compat.py +71 -0
  16. metaflow/_vendor/v3_7/importlib_metadata/_functools.py +104 -0
  17. metaflow/_vendor/v3_7/importlib_metadata/_itertools.py +73 -0
  18. metaflow/_vendor/v3_7/importlib_metadata/_meta.py +48 -0
  19. metaflow/_vendor/v3_7/importlib_metadata/_text.py +99 -0
  20. metaflow/_vendor/v3_7/importlib_metadata/py.typed +0 -0
  21. metaflow/_vendor/v3_7/typeguard/__init__.py +48 -0
  22. metaflow/_vendor/v3_7/typeguard/_checkers.py +906 -0
  23. metaflow/_vendor/v3_7/typeguard/_config.py +108 -0
  24. metaflow/_vendor/v3_7/typeguard/_decorators.py +237 -0
  25. metaflow/_vendor/v3_7/typeguard/_exceptions.py +42 -0
  26. metaflow/_vendor/v3_7/typeguard/_functions.py +310 -0
  27. metaflow/_vendor/v3_7/typeguard/_importhook.py +213 -0
  28. metaflow/_vendor/v3_7/typeguard/_memo.py +48 -0
  29. metaflow/_vendor/v3_7/typeguard/_pytest_plugin.py +100 -0
  30. metaflow/_vendor/v3_7/typeguard/_suppression.py +88 -0
  31. metaflow/_vendor/v3_7/typeguard/_transformer.py +1207 -0
  32. metaflow/_vendor/v3_7/typeguard/_union_transformer.py +54 -0
  33. metaflow/_vendor/v3_7/typeguard/_utils.py +169 -0
  34. metaflow/_vendor/v3_7/typeguard/py.typed +0 -0
  35. metaflow/_vendor/v3_7/typing_extensions.py +3072 -0
  36. metaflow/_vendor/v3_7/zipp.py +329 -0
  37. metaflow/cmd/develop/stubs.py +1 -1
  38. metaflow/extension_support/__init__.py +1 -1
  39. metaflow/plugins/argo/argo_workflows.py +34 -11
  40. metaflow/plugins/argo/argo_workflows_deployer_objects.py +7 -6
  41. metaflow/plugins/pypi/utils.py +4 -0
  42. metaflow/runner/click_api.py +7 -2
  43. metaflow/vendor.py +1 -0
  44. metaflow/version.py +1 -1
  45. {ob_metaflow-2.15.5.1.data → ob_metaflow-2.15.7.1.data}/data/share/metaflow/devtools/Makefile +2 -2
  46. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info}/METADATA +4 -3
  47. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info}/RECORD +53 -27
  48. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info}/WHEEL +1 -1
  49. {ob_metaflow-2.15.5.1.data → ob_metaflow-2.15.7.1.data}/data/share/metaflow/devtools/Tiltfile +0 -0
  50. {ob_metaflow-2.15.5.1.data → ob_metaflow-2.15.7.1.data}/data/share/metaflow/devtools/pick_services.sh +0 -0
  51. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info}/entry_points.txt +0 -0
  52. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info/licenses}/LICENSE +0 -0
  53. {ob_metaflow-2.15.5.1.dist-info → ob_metaflow-2.15.7.1.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,329 @@
1
+ import io
2
+ import posixpath
3
+ import zipfile
4
+ import itertools
5
+ import contextlib
6
+ import sys
7
+ import pathlib
8
+
9
+ if sys.version_info < (3, 7):
10
+ from collections import OrderedDict
11
+ else:
12
+ OrderedDict = dict
13
+
14
+
15
+ __all__ = ['Path']
16
+
17
+
18
+ def _parents(path):
19
+ """
20
+ Given a path with elements separated by
21
+ posixpath.sep, generate all parents of that path.
22
+
23
+ >>> list(_parents('b/d'))
24
+ ['b']
25
+ >>> list(_parents('/b/d/'))
26
+ ['/b']
27
+ >>> list(_parents('b/d/f/'))
28
+ ['b/d', 'b']
29
+ >>> list(_parents('b'))
30
+ []
31
+ >>> list(_parents(''))
32
+ []
33
+ """
34
+ return itertools.islice(_ancestry(path), 1, None)
35
+
36
+
37
+ def _ancestry(path):
38
+ """
39
+ Given a path with elements separated by
40
+ posixpath.sep, generate all elements of that path
41
+
42
+ >>> list(_ancestry('b/d'))
43
+ ['b/d', 'b']
44
+ >>> list(_ancestry('/b/d/'))
45
+ ['/b/d', '/b']
46
+ >>> list(_ancestry('b/d/f/'))
47
+ ['b/d/f', 'b/d', 'b']
48
+ >>> list(_ancestry('b'))
49
+ ['b']
50
+ >>> list(_ancestry(''))
51
+ []
52
+ """
53
+ path = path.rstrip(posixpath.sep)
54
+ while path and path != posixpath.sep:
55
+ yield path
56
+ path, tail = posixpath.split(path)
57
+
58
+
59
+ _dedupe = OrderedDict.fromkeys
60
+ """Deduplicate an iterable in original order"""
61
+
62
+
63
+ def _difference(minuend, subtrahend):
64
+ """
65
+ Return items in minuend not in subtrahend, retaining order
66
+ with O(1) lookup.
67
+ """
68
+ return itertools.filterfalse(set(subtrahend).__contains__, minuend)
69
+
70
+
71
+ class CompleteDirs(zipfile.ZipFile):
72
+ """
73
+ A ZipFile subclass that ensures that implied directories
74
+ are always included in the namelist.
75
+ """
76
+
77
+ @staticmethod
78
+ def _implied_dirs(names):
79
+ parents = itertools.chain.from_iterable(map(_parents, names))
80
+ as_dirs = (p + posixpath.sep for p in parents)
81
+ return _dedupe(_difference(as_dirs, names))
82
+
83
+ def namelist(self):
84
+ names = super(CompleteDirs, self).namelist()
85
+ return names + list(self._implied_dirs(names))
86
+
87
+ def _name_set(self):
88
+ return set(self.namelist())
89
+
90
+ def resolve_dir(self, name):
91
+ """
92
+ If the name represents a directory, return that name
93
+ as a directory (with the trailing slash).
94
+ """
95
+ names = self._name_set()
96
+ dirname = name + '/'
97
+ dir_match = name not in names and dirname in names
98
+ return dirname if dir_match else name
99
+
100
+ @classmethod
101
+ def make(cls, source):
102
+ """
103
+ Given a source (filename or zipfile), return an
104
+ appropriate CompleteDirs subclass.
105
+ """
106
+ if isinstance(source, CompleteDirs):
107
+ return source
108
+
109
+ if not isinstance(source, zipfile.ZipFile):
110
+ return cls(_pathlib_compat(source))
111
+
112
+ # Only allow for FastLookup when supplied zipfile is read-only
113
+ if 'r' not in source.mode:
114
+ cls = CompleteDirs
115
+
116
+ source.__class__ = cls
117
+ return source
118
+
119
+
120
+ class FastLookup(CompleteDirs):
121
+ """
122
+ ZipFile subclass to ensure implicit
123
+ dirs exist and are resolved rapidly.
124
+ """
125
+
126
+ def namelist(self):
127
+ with contextlib.suppress(AttributeError):
128
+ return self.__names
129
+ self.__names = super(FastLookup, self).namelist()
130
+ return self.__names
131
+
132
+ def _name_set(self):
133
+ with contextlib.suppress(AttributeError):
134
+ return self.__lookup
135
+ self.__lookup = super(FastLookup, self)._name_set()
136
+ return self.__lookup
137
+
138
+
139
+ def _pathlib_compat(path):
140
+ """
141
+ For path-like objects, convert to a filename for compatibility
142
+ on Python 3.6.1 and earlier.
143
+ """
144
+ try:
145
+ return path.__fspath__()
146
+ except AttributeError:
147
+ return str(path)
148
+
149
+
150
+ class Path:
151
+ """
152
+ A pathlib-compatible interface for zip files.
153
+
154
+ Consider a zip file with this structure::
155
+
156
+ .
157
+ ├── a.txt
158
+ └── b
159
+ ├── c.txt
160
+ └── d
161
+ └── e.txt
162
+
163
+ >>> data = io.BytesIO()
164
+ >>> zf = zipfile.ZipFile(data, 'w')
165
+ >>> zf.writestr('a.txt', 'content of a')
166
+ >>> zf.writestr('b/c.txt', 'content of c')
167
+ >>> zf.writestr('b/d/e.txt', 'content of e')
168
+ >>> zf.filename = 'mem/abcde.zip'
169
+
170
+ Path accepts the zipfile object itself or a filename
171
+
172
+ >>> root = Path(zf)
173
+
174
+ From there, several path operations are available.
175
+
176
+ Directory iteration (including the zip file itself):
177
+
178
+ >>> a, b = root.iterdir()
179
+ >>> a
180
+ Path('mem/abcde.zip', 'a.txt')
181
+ >>> b
182
+ Path('mem/abcde.zip', 'b/')
183
+
184
+ name property:
185
+
186
+ >>> b.name
187
+ 'b'
188
+
189
+ join with divide operator:
190
+
191
+ >>> c = b / 'c.txt'
192
+ >>> c
193
+ Path('mem/abcde.zip', 'b/c.txt')
194
+ >>> c.name
195
+ 'c.txt'
196
+
197
+ Read text:
198
+
199
+ >>> c.read_text()
200
+ 'content of c'
201
+
202
+ existence:
203
+
204
+ >>> c.exists()
205
+ True
206
+ >>> (b / 'missing.txt').exists()
207
+ False
208
+
209
+ Coercion to string:
210
+
211
+ >>> import os
212
+ >>> str(c).replace(os.sep, posixpath.sep)
213
+ 'mem/abcde.zip/b/c.txt'
214
+
215
+ At the root, ``name``, ``filename``, and ``parent``
216
+ resolve to the zipfile. Note these attributes are not
217
+ valid and will raise a ``ValueError`` if the zipfile
218
+ has no filename.
219
+
220
+ >>> root.name
221
+ 'abcde.zip'
222
+ >>> str(root.filename).replace(os.sep, posixpath.sep)
223
+ 'mem/abcde.zip'
224
+ >>> str(root.parent)
225
+ 'mem'
226
+ """
227
+
228
+ __repr = "{self.__class__.__name__}({self.root.filename!r}, {self.at!r})"
229
+
230
+ def __init__(self, root, at=""):
231
+ """
232
+ Construct a Path from a ZipFile or filename.
233
+
234
+ Note: When the source is an existing ZipFile object,
235
+ its type (__class__) will be mutated to a
236
+ specialized type. If the caller wishes to retain the
237
+ original type, the caller should either create a
238
+ separate ZipFile object or pass a filename.
239
+ """
240
+ self.root = FastLookup.make(root)
241
+ self.at = at
242
+
243
+ def open(self, mode='r', *args, pwd=None, **kwargs):
244
+ """
245
+ Open this entry as text or binary following the semantics
246
+ of ``pathlib.Path.open()`` by passing arguments through
247
+ to io.TextIOWrapper().
248
+ """
249
+ if self.is_dir():
250
+ raise IsADirectoryError(self)
251
+ zip_mode = mode[0]
252
+ if not self.exists() and zip_mode == 'r':
253
+ raise FileNotFoundError(self)
254
+ stream = self.root.open(self.at, zip_mode, pwd=pwd)
255
+ if 'b' in mode:
256
+ if args or kwargs:
257
+ raise ValueError("encoding args invalid for binary operation")
258
+ return stream
259
+ return io.TextIOWrapper(stream, *args, **kwargs)
260
+
261
+ @property
262
+ def name(self):
263
+ return pathlib.Path(self.at).name or self.filename.name
264
+
265
+ @property
266
+ def suffix(self):
267
+ return pathlib.Path(self.at).suffix or self.filename.suffix
268
+
269
+ @property
270
+ def suffixes(self):
271
+ return pathlib.Path(self.at).suffixes or self.filename.suffixes
272
+
273
+ @property
274
+ def stem(self):
275
+ return pathlib.Path(self.at).stem or self.filename.stem
276
+
277
+ @property
278
+ def filename(self):
279
+ return pathlib.Path(self.root.filename).joinpath(self.at)
280
+
281
+ def read_text(self, *args, **kwargs):
282
+ with self.open('r', *args, **kwargs) as strm:
283
+ return strm.read()
284
+
285
+ def read_bytes(self):
286
+ with self.open('rb') as strm:
287
+ return strm.read()
288
+
289
+ def _is_child(self, path):
290
+ return posixpath.dirname(path.at.rstrip("/")) == self.at.rstrip("/")
291
+
292
+ def _next(self, at):
293
+ return self.__class__(self.root, at)
294
+
295
+ def is_dir(self):
296
+ return not self.at or self.at.endswith("/")
297
+
298
+ def is_file(self):
299
+ return self.exists() and not self.is_dir()
300
+
301
+ def exists(self):
302
+ return self.at in self.root._name_set()
303
+
304
+ def iterdir(self):
305
+ if not self.is_dir():
306
+ raise ValueError("Can't listdir a file")
307
+ subs = map(self._next, self.root.namelist())
308
+ return filter(self._is_child, subs)
309
+
310
+ def __str__(self):
311
+ return posixpath.join(self.root.filename, self.at)
312
+
313
+ def __repr__(self):
314
+ return self.__repr.format(self=self)
315
+
316
+ def joinpath(self, *other):
317
+ next = posixpath.join(self.at, *map(_pathlib_compat, other))
318
+ return self._next(self.root.resolve_dir(next))
319
+
320
+ __truediv__ = joinpath
321
+
322
+ @property
323
+ def parent(self):
324
+ if not self.at:
325
+ return self.filename.parent
326
+ parent_at = posixpath.dirname(self.at.rstrip('/'))
327
+ if parent_at:
328
+ parent_at += '/'
329
+ return self._next(parent_at)
@@ -24,7 +24,7 @@ def _check_stubs_supported():
24
24
  if _py_ver >= (3, 8):
25
25
  from importlib import metadata
26
26
  elif _py_ver >= (3, 7):
27
- from metaflow._vendor import importlib_metadata as metadata
27
+ from metaflow._vendor.v3_7 import importlib_metadata as metadata
28
28
  elif _py_ver >= (3, 6):
29
29
  from metaflow._vendor.v3_6 import importlib_metadata as metadata
30
30
  else:
@@ -321,7 +321,7 @@ if _py_ver >= (3, 4):
321
321
  if _py_ver >= (3, 8):
322
322
  from importlib import metadata
323
323
  elif _py_ver >= (3, 7):
324
- from metaflow._vendor import importlib_metadata as metadata
324
+ from metaflow._vendor.v3_7 import importlib_metadata as metadata
325
325
  elif _py_ver >= (3, 6):
326
326
  from metaflow._vendor.v3_6 import importlib_metadata as metadata
327
327
  else:
@@ -525,6 +525,7 @@ class ArgoWorkflows(object):
525
525
  default_value = json.dumps(default_value)
526
526
 
527
527
  parameters[param.name] = dict(
528
+ python_var_name=var,
528
529
  name=param.name,
529
530
  value=default_value,
530
531
  type=param_type,
@@ -839,7 +840,11 @@ class ArgoWorkflows(object):
839
840
  Arguments().parameters(
840
841
  [
841
842
  Parameter(parameter["name"])
842
- .value(parameter["value"])
843
+ .value(
844
+ "'%s'" % parameter["value"]
845
+ if parameter["type"] == "JSON"
846
+ else parameter["value"]
847
+ )
843
848
  .description(parameter.get("description"))
844
849
  # TODO: Better handle IncludeFile in Argo Workflows UI.
845
850
  for parameter in self.parameters.values()
@@ -1595,11 +1600,7 @@ class ArgoWorkflows(object):
1595
1600
  # {{foo.bar['param_name']}}.
1596
1601
  # https://argoproj.github.io/argo-events/tutorials/02-parameterization/
1597
1602
  # http://masterminds.github.io/sprig/strings.html
1598
- (
1599
- "--%s='{{workflow.parameters.%s}}'"
1600
- if parameter["type"] == "JSON"
1601
- else "--%s={{workflow.parameters.%s}}"
1602
- )
1603
+ "--%s={{workflow.parameters.%s}}"
1603
1604
  % (parameter["name"], parameter["name"])
1604
1605
  for parameter in self.parameters.values()
1605
1606
  ]
@@ -3197,15 +3198,37 @@ class ArgoWorkflows(object):
3197
3198
  # NOTE: We need the conditional logic in order to successfully fall back to the default value
3198
3199
  # when the event payload does not contain a key for a parameter.
3199
3200
  # NOTE: Keys might contain dashes, so use the safer 'get' for fetching the value
3200
- data_template='{{ if (hasKey $.Input.body.payload "%s") }}{{- (get $.Input.body.payload "%s" | toRawJson) -}}{{- else -}}{{ (fail "use-default-instead") }}{{- end -}}'
3201
- % (v, v),
3201
+ data_template='{{ if (hasKey $.Input.body.payload "%s") }}{{- (get $.Input.body.payload "%s" %s) -}}{{- else -}}{{ (fail "use-default-instead") }}{{- end -}}'
3202
+ % (
3203
+ v,
3204
+ v,
3205
+ (
3206
+ "| toRawJson | squote"
3207
+ if self.parameters[
3208
+ parameter_name
3209
+ ]["type"]
3210
+ == "JSON"
3211
+ else "| toRawJson"
3212
+ ),
3213
+ ),
3202
3214
  # Unfortunately the sensor needs to
3203
3215
  # record the default values for
3204
3216
  # the parameters - there doesn't seem
3205
3217
  # to be any way for us to skip
3206
- value=self.parameters[parameter_name][
3207
- "value"
3208
- ],
3218
+ value=(
3219
+ json.dumps(
3220
+ self.parameters[parameter_name][
3221
+ "value"
3222
+ ]
3223
+ )
3224
+ if self.parameters[parameter_name][
3225
+ "type"
3226
+ ]
3227
+ == "JSON"
3228
+ else self.parameters[
3229
+ parameter_name
3230
+ ]["value"]
3231
+ ),
3209
3232
  )
3210
3233
  .dest(
3211
3234
  # this undocumented (mis?)feature in
@@ -19,6 +19,7 @@ def generate_fake_flow_file_contents(
19
19
  ):
20
20
  params_code = ""
21
21
  for _, param_details in param_info.items():
22
+ param_python_var_name = param_details["python_var_name"]
22
23
  param_name = param_details["name"]
23
24
  param_type = param_details["type"]
24
25
  param_help = param_details["description"]
@@ -26,21 +27,21 @@ def generate_fake_flow_file_contents(
26
27
 
27
28
  if param_type == "JSON":
28
29
  params_code += (
29
- f" {param_name} = Parameter('{param_name}', "
30
- f"type=JSONType, help='{param_help}', required={param_required})\n"
30
+ f" {param_python_var_name} = Parameter('{param_name}', "
31
+ f"type=JSONType, help='''{param_help}''', required={param_required})\n"
31
32
  )
32
33
  elif param_type == "FilePath":
33
34
  is_text = param_details.get("is_text", True)
34
35
  encoding = param_details.get("encoding", "utf-8")
35
36
  params_code += (
36
- f" {param_name} = IncludeFile('{param_name}', "
37
- f"is_text={is_text}, encoding='{encoding}', help='{param_help}', "
37
+ f" {param_python_var_name} = IncludeFile('{param_name}', "
38
+ f"is_text={is_text}, encoding='{encoding}', help='''{param_help}''', "
38
39
  f"required={param_required})\n"
39
40
  )
40
41
  else:
41
42
  params_code += (
42
- f" {param_name} = Parameter('{param_name}', "
43
- f"type={param_type}, help='{param_help}', required={param_required})\n"
43
+ f" {param_python_var_name} = Parameter('{param_name}', "
44
+ f"type={param_type}, help='''{param_help}''', required={param_required})\n"
44
45
  )
45
46
 
46
47
  project_decorator = f"@project(name='{project_name}')\n" if project_name else ""
@@ -69,6 +69,8 @@ def pip_tags(python_version, mamba_platform):
69
69
  "_2_25",
70
70
  "_2_26",
71
71
  "_2_27",
72
+ "_2_28",
73
+ "_2_29",
72
74
  )
73
75
  ]
74
76
  platforms.append("linux_x86_64")
@@ -87,6 +89,8 @@ def pip_tags(python_version, mamba_platform):
87
89
  "_2_25",
88
90
  "_2_26",
89
91
  "_2_27",
92
+ "_2_28",
93
+ "_2_29",
90
94
  )
91
95
  ]
92
96
  platforms.append("linux_aarch64")
@@ -1,7 +1,13 @@
1
1
  import os
2
2
  import sys
3
3
 
4
- if sys.version_info < (3, 7):
4
+ _py_ver = sys.version_info[:2]
5
+
6
+ if _py_ver >= (3, 8):
7
+ from metaflow._vendor.typeguard import TypeCheckError, check_type
8
+ elif _py_ver >= (3, 7):
9
+ from metaflow._vendor.v3_7.typeguard import TypeCheckError, check_type
10
+ else:
5
11
  raise RuntimeError(
6
12
  """
7
13
  The Metaflow Programmatic API is not supported for versions of Python less than 3.7
@@ -35,7 +41,6 @@ from metaflow._vendor.click.types import (
35
41
  Tuple,
36
42
  UUIDParameterType,
37
43
  )
38
- from metaflow._vendor.typeguard import TypeCheckError, check_type
39
44
  from metaflow.decorators import add_decorator_options
40
45
  from metaflow.exception import MetaflowException
41
46
  from metaflow.includefile import FilePathClass
metaflow/vendor.py CHANGED
@@ -13,6 +13,7 @@ WHITELIST = {
13
13
  "vendor_any.txt",
14
14
  "vendor_v3_5.txt",
15
15
  "vendor_v3_6.txt",
16
+ "vendor_v3_7.txt",
16
17
  "pip.LICENSE",
17
18
  }
18
19
 
metaflow/version.py CHANGED
@@ -1 +1 @@
1
- metaflow_version = "2.15.5.1"
1
+ metaflow_version = "2.15.7.1"
@@ -75,7 +75,7 @@ check-docker:
75
75
  @if [ "$(shell uname)" = "Darwin" ]; then \
76
76
  open -a Docker || (echo "❌ Please start Docker Desktop" && exit 1); \
77
77
  else \
78
- systemctl is-active --quiet docker || (echo "❌ Docker daemon is not running. Start with 'sudo systemctl start docker'" && exit 1); \
78
+ docker info >/dev/null 2>&1 || (echo "❌ Docker daemon is not running." && exit 1); \
79
79
  fi
80
80
  @echo "✅ Docker is running"
81
81
 
@@ -339,4 +339,4 @@ ui: setup-tilt
339
339
 
340
340
  .PHONY: install-helm setup-minikube setup-tilt teardown-minikube tunnel up down check-docker install-curl install-gum install-brew up down dashboard shell ui all-up help
341
341
 
342
- .DEFAULT_GOAL := help
342
+ .DEFAULT_GOAL := help
@@ -1,6 +1,6 @@
1
- Metadata-Version: 2.2
1
+ Metadata-Version: 2.4
2
2
  Name: ob-metaflow
3
- Version: 2.15.5.1
3
+ Version: 2.15.7.1
4
4
  Summary: Metaflow: More AI and ML, Less Engineering
5
5
  Author: Netflix, Outerbounds & the Metaflow Community
6
6
  Author-email: help@outerbounds.co
@@ -12,12 +12,13 @@ Requires-Dist: boto3
12
12
  Requires-Dist: pylint
13
13
  Requires-Dist: kubernetes
14
14
  Provides-Extra: stubs
15
- Requires-Dist: metaflow-stubs==2.15.5.1; extra == "stubs"
15
+ Requires-Dist: metaflow-stubs==2.15.7.1; extra == "stubs"
16
16
  Dynamic: author
17
17
  Dynamic: author-email
18
18
  Dynamic: description
19
19
  Dynamic: description-content-type
20
20
  Dynamic: license
21
+ Dynamic: license-file
21
22
  Dynamic: provides-extra
22
23
  Dynamic: requires-dist
23
24
  Dynamic: summary