dcicutils 8.8.6.1b1__tar.gz → 8.8.6.1b3__tar.gz
Sign up to get free protection for your applications and to get access to all the features.
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/PKG-INFO +1 -1
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/structured_data.py +26 -11
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/pyproject.toml +1 -1
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/LICENSE.txt +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/README.rst +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/__init__.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/base.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/beanstalk_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/bundle_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/captured_output.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/cloudformation_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/codebuild_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/command_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/common.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/contribution_scripts.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/contribution_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/creds_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/data_readers.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/data_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/datetime_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/deployment_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/diff_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/docker_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ecr_scripts.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ecr_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ecs_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/env_base.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/env_manager.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/env_scripts.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/env_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/env_utils_legacy.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/es_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/exceptions.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ff_mocks.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ff_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/file_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/function_cache_decorator.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/glacier_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/http_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/jh_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/kibana/dashboards.json +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/kibana/readme.md +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/lang_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/c4-infrastructure.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/c4-python-infrastructure.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-common-server.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-common.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-gpl-pipeline.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-pipeline.jsonc +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/log_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/misc_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/obfuscation_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/opensearch_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/portal_object_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/portal_utils.py +41 -41
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/progress_bar.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/project_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/qa_checkers.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/qa_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/redis_tools.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/redis_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/s3_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/schema_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/scripts/publish_to_pypi.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/scripts/run_license_checker.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/scripts/view_portal_object.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/secrets_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/sheet_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/snapshot_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/ssl_certificate_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/submitr/progress_constants.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/submitr/ref_lookup_strategy.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/task_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/tmpfile_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/trace_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/validation_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/variant_utils.py +0 -0
- {dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/zip_utils.py +0 -0
@@ -351,18 +351,23 @@ class StructuredDataSet:
|
|
351
351
|
|
352
352
|
def _load_json_file(self, file: str) -> None:
|
353
353
|
with open(file) as f:
|
354
|
-
|
355
|
-
|
356
|
-
|
354
|
+
item = json.load(f)
|
355
|
+
if ((schema_name_inferred_from_file_name := Schema.type_name(file)) and
|
356
|
+
(self._portal.get_schema(schema_name_inferred_from_file_name) is not None)): # noqa
|
357
357
|
# If the JSON file name looks like a schema name then assume it
|
358
358
|
# contains an object or an array of object of that schema type.
|
359
|
-
self.
|
360
|
-
|
359
|
+
if self._merge:
|
360
|
+
item = self._merge_with_existing_portal_object(item, schema_name_inferred_from_file_name)
|
361
|
+
self._add(Schema.type_name(file), item)
|
362
|
+
elif isinstance(item, dict):
|
361
363
|
# Otherwise if the JSON file name does not look like a schema name then
|
362
364
|
# assume it a dictionary where each property is the name of a schema, and
|
363
365
|
# which (each property) contains a list of object of that schema type.
|
364
|
-
for schema_name in
|
365
|
-
|
366
|
+
for schema_name in item:
|
367
|
+
item = item[schema_name]
|
368
|
+
if self._merge:
|
369
|
+
item = self._merge_with_existing_portal_object(item, schema_name)
|
370
|
+
self._add(schema_name, item)
|
366
371
|
|
367
372
|
def _load_reader(self, reader: RowReader, type_name: str) -> None:
|
368
373
|
schema = None
|
@@ -386,14 +391,12 @@ class StructuredDataSet:
|
|
386
391
|
self._add_properties(structured_row, self._autoadd_properties, schema)
|
387
392
|
# New merge functionality (2024-05-25).
|
388
393
|
if self._merge:
|
389
|
-
|
390
|
-
if existing_portal_object := self._portal.get_metadata(identifying_path):
|
391
|
-
structured_row = merge_objects(existing_portal_object, structured_row)
|
394
|
+
structured_row = self._merge_with_existing_portal_object(structured_row, schema_name)
|
392
395
|
if (prune_error := self._prune_structured_row(structured_row)) is not None:
|
393
396
|
self._note_error({"src": create_dict(type=schema_name, row=reader.row_number),
|
394
397
|
"error": prune_error}, "validation")
|
395
398
|
else:
|
396
|
-
self._add(type_name, structured_row)
|
399
|
+
self._add(type_name, structured_row) # TODO: why type_name and not schema_name?
|
397
400
|
if self._progress:
|
398
401
|
self._progress({
|
399
402
|
PROGRESS.LOAD_ITEM: self._nrows,
|
@@ -434,6 +437,18 @@ class StructuredDataSet:
|
|
434
437
|
if name not in structured_row and (not schema or schema.data.get("properties", {}).get(name)):
|
435
438
|
structured_row[name] = properties[name]
|
436
439
|
|
440
|
+
def _merge_with_existing_portal_object(self, portal_object: dict, portal_type: str) -> dict:
|
441
|
+
"""
|
442
|
+
Given a Portal object (presumably/in-practice from the given metadata), if there is
|
443
|
+
an existing Portal item, identified by the identifying properties for the given object,
|
444
|
+
then merges the given object into the existing one and returns the result; otherwise
|
445
|
+
just returns the given object. Note that the given object may be CHANGED in place.
|
446
|
+
"""
|
447
|
+
for identifying_path in self._portal.get_identifying_paths(portal_object, portal_type):
|
448
|
+
if existing_portal_object := self._portal.get_metadata(identifying_path, raw=True, raise_exception=False):
|
449
|
+
return merge_objects(existing_portal_object, portal_object)
|
450
|
+
return portal_object
|
451
|
+
|
437
452
|
def _is_ref_lookup_specified_type(ref_lookup_flags: int) -> bool:
|
438
453
|
return (ref_lookup_flags &
|
439
454
|
Portal.LOOKUP_SPECIFIED_TYPE) == Portal.LOOKUP_SPECIFIED_TYPE
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
{dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/c4-infrastructure.jsonc
RENAMED
File without changes
|
File without changes
|
{dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-common-server.jsonc
RENAMED
File without changes
|
{dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-common.jsonc
RENAMED
File without changes
|
{dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-gpl-pipeline.jsonc
RENAMED
File without changes
|
{dcicutils-8.8.6.1b1 → dcicutils-8.8.6.1b3}/dcicutils/license_policies/park-lab-pipeline.jsonc
RENAMED
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
@@ -418,31 +418,6 @@ class Portal:
|
|
418
418
|
return []
|
419
419
|
return schemas_super_type_map.get(type_name, [])
|
420
420
|
|
421
|
-
def url(self, url: str, raw: bool = False, database: bool = False) -> str:
|
422
|
-
if not isinstance(url, str) or not url:
|
423
|
-
return "/"
|
424
|
-
elif (lowercase_url := url.lower()).startswith("http://") or lowercase_url.startswith("https://"):
|
425
|
-
return url
|
426
|
-
elif not (url := re.sub(r"/+", "/", url)).startswith("/"):
|
427
|
-
url = "/"
|
428
|
-
url = self.server + url if self.server else url
|
429
|
-
if isinstance(raw, bool) and raw:
|
430
|
-
url += ("&" if "?" in url else "?") + "frame=raw"
|
431
|
-
if isinstance(database, bool) and database:
|
432
|
-
url += ("&" if "?" in url else "?") + "datastore=database"
|
433
|
-
return url
|
434
|
-
|
435
|
-
def _kwargs(self, **kwargs) -> dict:
|
436
|
-
if "headers" in kwargs:
|
437
|
-
result_kwargs = {"headers": kwargs["headers"]}
|
438
|
-
else:
|
439
|
-
result_kwargs = {"headers": {"Content-type": Portal.MIME_TYPE_JSON, "Accept": Portal.MIME_TYPE_JSON}}
|
440
|
-
if self.key_pair:
|
441
|
-
result_kwargs["auth"] = self.key_pair
|
442
|
-
if isinstance(timeout := kwargs.get("timeout"), int):
|
443
|
-
result_kwargs["timeout"] = timeout
|
444
|
-
return result_kwargs
|
445
|
-
|
446
421
|
@function_cache(maxsize=100, serialize_key=True)
|
447
422
|
def get_identifying_paths(self, portal_object: dict, portal_type: Optional[str] = None) -> List[str]:
|
448
423
|
"""
|
@@ -491,6 +466,31 @@ class Portal:
|
|
491
466
|
identifying_properties.insert(0, favored_identifying_property)
|
492
467
|
return identifying_properties
|
493
468
|
|
469
|
+
def url(self, url: str, raw: bool = False, database: bool = False) -> str:
|
470
|
+
if not isinstance(url, str) or not url:
|
471
|
+
return "/"
|
472
|
+
elif (lowercase_url := url.lower()).startswith("http://") or lowercase_url.startswith("https://"):
|
473
|
+
return url
|
474
|
+
elif not (url := re.sub(r"/+", "/", url)).startswith("/"):
|
475
|
+
url = "/"
|
476
|
+
url = self.server + url if self.server else url
|
477
|
+
if isinstance(raw, bool) and raw:
|
478
|
+
url += ("&" if "?" in url else "?") + "frame=raw"
|
479
|
+
if isinstance(database, bool) and database:
|
480
|
+
url += ("&" if "?" in url else "?") + "datastore=database"
|
481
|
+
return url
|
482
|
+
|
483
|
+
def _kwargs(self, **kwargs) -> dict:
|
484
|
+
if "headers" in kwargs:
|
485
|
+
result_kwargs = {"headers": kwargs["headers"]}
|
486
|
+
else:
|
487
|
+
result_kwargs = {"headers": {"Content-type": Portal.MIME_TYPE_JSON, "Accept": Portal.MIME_TYPE_JSON}}
|
488
|
+
if self.key_pair:
|
489
|
+
result_kwargs["auth"] = self.key_pair
|
490
|
+
if isinstance(timeout := kwargs.get("timeout"), int):
|
491
|
+
result_kwargs["timeout"] = timeout
|
492
|
+
return result_kwargs
|
493
|
+
|
494
494
|
@staticmethod
|
495
495
|
def _default_keys_file(app: Optional[str], env: Optional[str], server: Optional[str]) -> Optional[str]:
|
496
496
|
def infer_app_from_env(env: str) -> Optional[str]: # noqa
|
@@ -566,6 +566,22 @@ class Portal:
|
|
566
566
|
response = TestResponseWrapper(response)
|
567
567
|
return response
|
568
568
|
|
569
|
+
@staticmethod
|
570
|
+
def _create_vapp(arg: Union[TestApp, VirtualApp, PyramidRouter, str] = None) -> TestApp:
|
571
|
+
if isinstance(arg, TestApp):
|
572
|
+
return arg
|
573
|
+
elif isinstance(arg, VirtualApp):
|
574
|
+
if not isinstance(arg.wrapped_app, TestApp):
|
575
|
+
raise Exception("Portal._create_vapp VirtualApp argument error.")
|
576
|
+
return arg.wrapped_app
|
577
|
+
if isinstance(arg, PyramidRouter):
|
578
|
+
router = arg
|
579
|
+
elif isinstance(arg, str) or not arg:
|
580
|
+
router = pyramid_get_app(arg or "development.ini", "app")
|
581
|
+
else:
|
582
|
+
raise Exception("Portal._create_vapp argument error.")
|
583
|
+
return TestApp(router, {"HTTP_ACCEPT": Portal.MIME_TYPE_JSON, "REMOTE_USER": "TEST"})
|
584
|
+
|
569
585
|
@staticmethod
|
570
586
|
def create_for_testing(arg: Optional[Union[str, bool, List[dict], dict, Callable]] = None) -> Portal:
|
571
587
|
if isinstance(arg, list) or isinstance(arg, dict) or isinstance(arg, Callable):
|
@@ -597,22 +613,6 @@ class Portal:
|
|
597
613
|
with temporary_file(content=minimal_ini_for_testing, suffix=".ini") as ini_file:
|
598
614
|
return Portal(ini_file)
|
599
615
|
|
600
|
-
@staticmethod
|
601
|
-
def _create_vapp(arg: Union[TestApp, VirtualApp, PyramidRouter, str] = None) -> TestApp:
|
602
|
-
if isinstance(arg, TestApp):
|
603
|
-
return arg
|
604
|
-
elif isinstance(arg, VirtualApp):
|
605
|
-
if not isinstance(arg.wrapped_app, TestApp):
|
606
|
-
raise Exception("Portal._create_vapp VirtualApp argument error.")
|
607
|
-
return arg.wrapped_app
|
608
|
-
if isinstance(arg, PyramidRouter):
|
609
|
-
router = arg
|
610
|
-
elif isinstance(arg, str) or not arg:
|
611
|
-
router = pyramid_get_app(arg or "development.ini", "app")
|
612
|
-
else:
|
613
|
-
raise Exception("Portal._create_vapp argument error.")
|
614
|
-
return TestApp(router, {"HTTP_ACCEPT": Portal.MIME_TYPE_JSON, "REMOTE_USER": "TEST"})
|
615
|
-
|
616
616
|
@staticmethod
|
617
617
|
def _create_router_for_testing(endpoints: Optional[List[Dict[str, Union[str, Callable]]]] = None) -> PyramidRouter:
|
618
618
|
if isinstance(endpoints, dict):
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|