dcicutils 8.5.0.1b2__tar.gz → 8.5.0.1b4__tar.gz

Sign up to get free protection for your applications and to get access to all the features.
Files changed (68) hide show
  1. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/PKG-INFO +1 -1
  2. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/portal_utils.py +23 -8
  3. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/structured_data.py +20 -8
  4. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/pyproject.toml +1 -1
  5. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/LICENSE.txt +0 -0
  6. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/README.rst +0 -0
  7. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/__init__.py +0 -0
  8. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/base.py +0 -0
  9. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/beanstalk_utils.py +0 -0
  10. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/bundle_utils.py +0 -0
  11. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/cloudformation_utils.py +0 -0
  12. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/codebuild_utils.py +0 -0
  13. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/command_utils.py +0 -0
  14. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/common.py +0 -0
  15. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/contribution_scripts.py +0 -0
  16. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/contribution_utils.py +0 -0
  17. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/creds_utils.py +0 -0
  18. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/data_readers.py +0 -0
  19. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/data_utils.py +0 -0
  20. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/deployment_utils.py +0 -0
  21. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/diff_utils.py +0 -0
  22. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/docker_utils.py +0 -0
  23. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ecr_scripts.py +0 -0
  24. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ecr_utils.py +0 -0
  25. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ecs_utils.py +0 -0
  26. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/env_base.py +0 -0
  27. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/env_manager.py +0 -0
  28. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/env_scripts.py +0 -0
  29. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/env_utils.py +0 -0
  30. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/env_utils_legacy.py +0 -0
  31. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/es_utils.py +0 -0
  32. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/exceptions.py +0 -0
  33. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ff_mocks.py +0 -0
  34. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ff_utils.py +0 -0
  35. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/function_cache_decorator.py +0 -0
  36. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/glacier_utils.py +0 -0
  37. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/jh_utils.py +0 -0
  38. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/kibana/dashboards.json +0 -0
  39. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/kibana/readme.md +0 -0
  40. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/lang_utils.py +0 -0
  41. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/c4-infrastructure.jsonc +0 -0
  42. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/c4-python-infrastructure.jsonc +0 -0
  43. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/park-lab-common-server.jsonc +0 -0
  44. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/park-lab-common.jsonc +0 -0
  45. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/park-lab-gpl-pipeline.jsonc +0 -0
  46. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_policies/park-lab-pipeline.jsonc +0 -0
  47. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/license_utils.py +0 -0
  48. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/log_utils.py +0 -0
  49. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/misc_utils.py +0 -0
  50. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/obfuscation_utils.py +0 -0
  51. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/opensearch_utils.py +0 -0
  52. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/project_utils.py +0 -0
  53. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/qa_checkers.py +0 -0
  54. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/qa_utils.py +0 -0
  55. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/redis_tools.py +0 -0
  56. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/redis_utils.py +0 -0
  57. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/s3_utils.py +0 -0
  58. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/scripts/publish_to_pypi.py +0 -0
  59. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/scripts/run_license_checker.py +0 -0
  60. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/secrets_utils.py +0 -0
  61. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/sheet_utils.py +0 -0
  62. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/snapshot_utils.py +0 -0
  63. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/ssl_certificate_utils.py +0 -0
  64. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/task_utils.py +0 -0
  65. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/trace_utils.py +0 -0
  66. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/validation_utils.py +0 -0
  67. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/variant_utils.py +0 -0
  68. {dcicutils-8.5.0.1b2 → dcicutils-8.5.0.1b4}/dcicutils/zip_utils.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: dcicutils
3
- Version: 8.5.0.1b2
3
+ Version: 8.5.0.1b4
4
4
  Summary: Utility package for interacting with the 4DN Data Portal and other 4DN resources
5
5
  Home-page: https://github.com/4dn-dcic/utils
6
6
  License: MIT
@@ -23,15 +23,14 @@ class Portal:
23
23
  2. From a key dictionary, containing "key" and "secret" property values.
24
24
  3. From a key tuple, containing (in order) a key and secret values.
25
25
  4. From a keys file assumed to reside in ~/.{app}-keys.json where the given "app" value is either "smaht", "cgap",
26
- or "fourfront"; and where this file is assumed to contain a dictionary with a key equal to the given "env"
27
- value (e.g. smaht-localhost) and with a dictionary value containing "key" and "secret" property values; if
28
- an "app" value is not specified but the given "env" value begins with one of the app values then that value
29
- will be used, i.e. e.g. if env is "smaht-localhost" and app is unspecified than it is assumed to be "smaht".
26
+ or "fourfront"; where is assumed to contain a dictionary with a key for the given "env" value, e.g. smaht-local;
27
+ and with a dictionary value containing "key" and "secret" property values, and an optional "server" property;
28
+ if an "app" value is not specified but the given "env" value begins with one of the app values then that value
29
+ will be used, i.e. e.g. if "env" is "smaht-local" and app is unspecified than it is assumed to be "smaht".
30
30
  5. From a keys file as described above (#4) but rather than be identified by the given "env" value it
31
- is looked up by the given "server" name and the "server" key dictionary value in the key file.
31
+ is looked up via the given "server" name and the "server" key dictionary value in the key file.
32
32
  6. From a given "vapp" value (which is assumed to be a TestApp or VirtualApp).
33
- 7. From another Portal object.
34
- 8. From a a pyramid Router object.
33
+ 7. From another Portal object; or from a a pyramid Router object.
35
34
  """
36
35
  def __init__(self,
37
36
  arg: Optional[Union[VirtualApp, TestApp, Router, Portal, dict, tuple, str]] = None,
@@ -62,6 +61,7 @@ class Portal:
62
61
  self._server = portal._server
63
62
  self._key = portal._key
64
63
  self._key_pair = portal._key_pair
64
+ self._key_id = portal._key_id
65
65
  self._key_file = portal._key_file
66
66
  return
67
67
  self._vapp = None
@@ -70,6 +70,7 @@ class Portal:
70
70
  self._server = server
71
71
  self._key = None
72
72
  self._key_pair = None
73
+ self._key_id = None
73
74
  self._key_file = None
74
75
  if isinstance(portal, (VirtualApp, TestApp)):
75
76
  self._vapp = portal
@@ -95,6 +96,10 @@ class Portal:
95
96
  self._key = key_manager.get_keydict_for_server(self._server)
96
97
  self._key_pair = key_manager.keydict_to_keypair(self._key) if self._key else None
97
98
  self._key_file = key_manager.keys_file
99
+ if self._key and (key_id := self._key.get("key")):
100
+ self._key_id = key_id
101
+ elif self._key_pair and (key_id := self._key_pair[1]):
102
+ self._key_id = key_id
98
103
 
99
104
  @property
100
105
  def env(self):
@@ -116,6 +121,10 @@ class Portal:
116
121
  def key_pair(self):
117
122
  return self._key_pair
118
123
 
124
+ @property
125
+ def key_id(self):
126
+ return self._key_id
127
+
119
128
  @property
120
129
  def key_file(self):
121
130
  return self._key_file
@@ -206,13 +215,19 @@ class Portal:
206
215
  super_type_map_flattened[super_type_name] = breadth_first(super_type_map, super_type_name)
207
216
  return super_type_map_flattened
208
217
 
218
+ def ping(self) -> bool:
219
+ try:
220
+ return self.get("/health").status_code == 200
221
+ except Exception:
222
+ return False
223
+
209
224
  def _uri(self, uri: str) -> str:
210
225
  if not isinstance(uri, str) or not uri:
211
226
  return "/"
212
227
  if uri.lower().startswith("http://") or uri.lower().startswith("https://"):
213
228
  return uri
214
229
  uri = re.sub(r"/+", "/", uri)
215
- return (self._server + ("/" if uri.startswith("/") else "") + uri) if self._server else uri
230
+ return (self._server + ("/" if not uri.startswith("/") else "") + uri) if self._server else uri
216
231
 
217
232
  def _kwargs(self, **kwargs) -> dict:
218
233
  result_kwargs = {"headers":
@@ -42,8 +42,8 @@ StructuredDataSet = Type["StructuredDataSet"]
42
42
  class StructuredDataSet:
43
43
 
44
44
  def __init__(self, file: Optional[str] = None, portal: Optional[Union[VirtualApp, TestApp, Portal]] = None,
45
- schemas: Optional[List[dict]] = None, data: Optional[List[dict]] = None,
46
- order: Optional[List[str]] = None, prune: bool = True) -> None:
45
+ schemas: Optional[List[dict]] = None, autoadd: Optional[dict] = None,
46
+ data: Optional[List[dict]] = None, order: Optional[List[str]] = None, prune: bool = True) -> None:
47
47
  self.data = {} if not data else data # If portal is None then no schemas nor refs.
48
48
  self._portal = Portal(portal, data=self.data, schemas=schemas) if portal else None
49
49
  self._order = order
@@ -52,13 +52,14 @@ class StructuredDataSet:
52
52
  self._errors = {}
53
53
  self._resolved_refs = set()
54
54
  self._validated = False
55
+ self._autoadd_properties = autoadd if isinstance(autoadd, dict) and autoadd else None
55
56
  self._load_file(file) if file else None
56
57
 
57
58
  @staticmethod
58
59
  def load(file: str, portal: Optional[Union[VirtualApp, TestApp, Portal]] = None,
59
- schemas: Optional[List[dict]] = None,
60
+ schemas: Optional[List[dict]] = None, autoadd: Optional[dict] = None,
60
61
  order: Optional[List[str]] = None, prune: bool = True) -> StructuredDataSet:
61
- return StructuredDataSet(file=file, portal=portal, schemas=schemas, order=order, prune=prune)
62
+ return StructuredDataSet(file=file, portal=portal, schemas=schemas, autoadd=autoadd, order=order, prune=prune)
62
63
 
63
64
  def validate(self, force: bool = False) -> None:
64
65
  if self._validated and not force:
@@ -163,6 +164,8 @@ class StructuredDataSet:
163
164
  structured_row = structured_row_template.create_row()
164
165
  for column_name, value in row.items():
165
166
  structured_row_template.set_value(structured_row, column_name, value, reader.file, reader.row_number)
167
+ if self._autoadd_properties:
168
+ self._add_properties(structured_row, self._autoadd_properties, schema)
166
169
  self._add(type_name, structured_row)
167
170
  self._note_warning(reader.warnings, "reader")
168
171
  if schema:
@@ -177,6 +180,11 @@ class StructuredDataSet:
177
180
  else:
178
181
  self.data[type_name] = [data] if isinstance(data, dict) else data
179
182
 
183
+ def _add_properties(self, structured_row: dict, properties: dict, schema: Optional[dict] = None) -> None:
184
+ for name in properties:
185
+ if name not in structured_row and (not schema or schema.data.get("properties", {}).get(name)):
186
+ structured_row[name] = properties[name]
187
+
180
188
  def _note_warning(self, item: Optional[Union[dict, List[dict]]], group: str) -> None:
181
189
  self._note_issue(self._warnings, item, group)
182
190
 
@@ -317,7 +325,8 @@ class Schema:
317
325
 
318
326
  @staticmethod
319
327
  def load_by_name(name: str, portal: Portal) -> Optional[dict]:
320
- return Schema(portal.get_schema(Schema.type_name(name)), portal) if portal else None
328
+ schema_json = portal.get_schema(Schema.type_name(name)) if portal else None
329
+ return Schema(schema_json, portal) if schema_json else None
321
330
 
322
331
  def validate(self, data: dict) -> List[str]:
323
332
  errors = []
@@ -565,7 +574,9 @@ class Portal(PortalBase):
565
574
 
566
575
  @lru_cache(maxsize=256)
567
576
  def get_schema(self, schema_name: str) -> Optional[dict]:
568
- if (schemas := self.get_schemas()) and (schema := schemas.get(schema_name := Schema.type_name(schema_name))):
577
+ if not (schemas := self.get_schemas()):
578
+ return None
579
+ if schema := schemas.get(schema_name := Schema.type_name(schema_name)):
569
580
  return schema
570
581
  if schema_name == schema_name.upper() and (schema := schemas.get(schema_name.lower().title())):
571
582
  return schema
@@ -573,8 +584,9 @@ class Portal(PortalBase):
573
584
  return schema
574
585
 
575
586
  @lru_cache(maxsize=1)
576
- def get_schemas(self) -> dict:
577
- schemas = super().get_schemas()
587
+ def get_schemas(self) -> Optional[dict]:
588
+ if not (schemas := super().get_schemas()) or (schemas.get("status") == "error"):
589
+ return None
578
590
  if self._schemas:
579
591
  schemas = copy.deepcopy(schemas)
580
592
  for user_specified_schema in self._schemas:
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "dcicutils"
3
- version = "8.5.0.1b2" # TODO: To become 8.6.0
3
+ version = "8.5.0.1b4" # TODO: To become 8.6.0
4
4
  description = "Utility package for interacting with the 4DN Data Portal and other 4DN resources"
5
5
  authors = ["4DN-DCIC Team <support@4dnucleome.org>"]
6
6
  license = "MIT"
File without changes
File without changes