dcicutils 8.13.3.1b24__tar.gz → 8.13.3.1b26__tar.gz

Sign up to get free protection for your applications and to get access to all the features.
Files changed (81) hide show
  1. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/PKG-INFO +1 -1
  2. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/misc_utils.py +71 -60
  3. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/structured_data.py +7 -7
  4. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/pyproject.toml +1 -1
  5. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/LICENSE.txt +0 -0
  6. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/README.rst +0 -0
  7. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/__init__.py +0 -0
  8. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/base.py +0 -0
  9. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/beanstalk_utils.py +0 -0
  10. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/bundle_utils.py +0 -0
  11. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/captured_output.py +0 -0
  12. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/cloudformation_utils.py +0 -0
  13. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/codebuild_utils.py +0 -0
  14. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/command_utils.py +0 -0
  15. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/common.py +0 -0
  16. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/contribution_scripts.py +0 -0
  17. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/contribution_utils.py +0 -0
  18. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/creds_utils.py +0 -0
  19. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/data_readers.py +0 -0
  20. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/data_utils.py +0 -0
  21. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/datetime_utils.py +0 -0
  22. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/deployment_utils.py +0 -0
  23. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/diff_utils.py +0 -0
  24. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/docker_utils.py +0 -0
  25. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ecr_scripts.py +0 -0
  26. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ecr_utils.py +0 -0
  27. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ecs_utils.py +0 -0
  28. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/env_base.py +0 -0
  29. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/env_manager.py +0 -0
  30. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/env_scripts.py +0 -0
  31. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/env_utils.py +0 -0
  32. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/env_utils_legacy.py +0 -0
  33. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/es_utils.py +0 -0
  34. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/exceptions.py +0 -0
  35. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ff_mocks.py +0 -0
  36. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ff_utils.py +0 -0
  37. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/file_utils.py +0 -0
  38. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/function_cache_decorator.py +0 -0
  39. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/glacier_utils.py +0 -0
  40. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/hack_for_elasticsearch_numpy_usage.py +0 -0
  41. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/http_utils.py +0 -0
  42. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/jh_utils.py +0 -0
  43. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/kibana/dashboards.json +0 -0
  44. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/kibana/readme.md +0 -0
  45. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/lang_utils.py +0 -0
  46. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/c4-infrastructure.jsonc +0 -0
  47. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/c4-python-infrastructure.jsonc +0 -0
  48. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/park-lab-common-server.jsonc +0 -0
  49. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/park-lab-common.jsonc +0 -0
  50. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/park-lab-gpl-pipeline.jsonc +0 -0
  51. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_policies/park-lab-pipeline.jsonc +0 -0
  52. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/license_utils.py +0 -0
  53. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/log_utils.py +0 -0
  54. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/obfuscation_utils.py +0 -0
  55. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/opensearch_utils.py +0 -0
  56. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/portal_object_utils.py +0 -0
  57. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/portal_utils.py +0 -0
  58. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/progress_bar.py +0 -0
  59. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/project_utils.py +0 -0
  60. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/qa_checkers.py +0 -0
  61. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/qa_utils.py +0 -0
  62. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/redis_tools.py +0 -0
  63. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/redis_utils.py +0 -0
  64. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/s3_utils.py +0 -0
  65. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/schema_utils.py +0 -0
  66. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/scripts/publish_to_pypi.py +0 -0
  67. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/scripts/run_license_checker.py +0 -0
  68. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/scripts/update_portal_object.py +0 -0
  69. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/scripts/view_portal_object.py +0 -0
  70. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/secrets_utils.py +0 -0
  71. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/sheet_utils.py +0 -0
  72. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/snapshot_utils.py +0 -0
  73. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/ssl_certificate_utils.py +0 -0
  74. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/submitr/progress_constants.py +0 -0
  75. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/submitr/ref_lookup_strategy.py +0 -0
  76. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/task_utils.py +0 -0
  77. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/tmpfile_utils.py +0 -0
  78. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/trace_utils.py +0 -0
  79. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/validation_utils.py +0 -0
  80. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/variant_utils.py +0 -0
  81. {dcicutils-8.13.3.1b24 → dcicutils-8.13.3.1b26}/dcicutils/zip_utils.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: dcicutils
3
- Version: 8.13.3.1b24
3
+ Version: 8.13.3.1b26
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
@@ -986,53 +986,61 @@ def str_to_bool(x: Optional[str]) -> Optional[bool]:
986
986
  raise ValueError(f"An argument to str_or_bool must be a string or None: {x!r}")
987
987
 
988
988
 
989
- def to_integer(value: str, fallback: Optional[Any] = None, strict: bool = False) -> Optional[Any]:
990
- try:
991
- return int(value)
992
- except Exception:
993
- if strict is not True:
994
- try:
995
- return int(float(value))
996
- except Exception:
997
- pass
998
- return fallback
999
-
1000
-
1001
- _MULTIPLIER_K = 1000
1002
- _MULTIPLIER_M = 1000 * _MULTIPLIER_K
1003
- _MULTIPLIER_G = 1000 * _MULTIPLIER_M
1004
- _MULTIPLIER_T = 1000 * _MULTIPLIER_G
1005
-
1006
- _MULTIPLIER_SUFFIXES = {
1007
- "K": _MULTIPLIER_K,
1008
- "KB": _MULTIPLIER_K,
1009
- "M": _MULTIPLIER_M,
1010
- "MB": _MULTIPLIER_M,
1011
- "G": _MULTIPLIER_G,
1012
- "GB": _MULTIPLIER_G,
1013
- "T": _MULTIPLIER_T,
1014
- "TB": _MULTIPLIER_T
989
+ def to_integer(value: str,
990
+ allow_commas: bool = False,
991
+ allow_multiplier_suffix: bool = False,
992
+ fallback: Optional[Union[int, float]] = None) -> Optional[int]:
993
+ return to_number(value, fallback=fallback, as_float=False,
994
+ allow_commas=allow_commas,
995
+ allow_multiplier_suffix=allow_multiplier_suffix)
996
+
997
+
998
+ def to_float(value: str,
999
+ allow_commas: bool = False,
1000
+ allow_multiplier_suffix: bool = False,
1001
+ fallback: Optional[Union[int, float]] = None) -> Optional[int]:
1002
+ return to_number(value, fallback=fallback, as_float=True,
1003
+ allow_commas=allow_commas,
1004
+ allow_multiplier_suffix=allow_multiplier_suffix)
1005
+
1006
+
1007
+ _TO_NUMBER_MULTIPLIER_K = 1000
1008
+ _TO_NUMBER_MULTIPLIER_M = 1000 * _TO_NUMBER_MULTIPLIER_K
1009
+ _TO_NUMBER_MULTIPLIER_G = 1000 * _TO_NUMBER_MULTIPLIER_M
1010
+ _TO_NUMBER_MULTIPLIER_T = 1000 * _TO_NUMBER_MULTIPLIER_G
1011
+
1012
+ _TO_NUMBER_MULTIPLIER_SUFFIXES = {
1013
+ "K": _TO_NUMBER_MULTIPLIER_K,
1014
+ "KB": _TO_NUMBER_MULTIPLIER_K,
1015
+ "M": _TO_NUMBER_MULTIPLIER_M,
1016
+ "MB": _TO_NUMBER_MULTIPLIER_M,
1017
+ "G": _TO_NUMBER_MULTIPLIER_G,
1018
+ "GB": _TO_NUMBER_MULTIPLIER_G,
1019
+ "T": _TO_NUMBER_MULTIPLIER_T,
1020
+ "TB": _TO_NUMBER_MULTIPLIER_T
1015
1021
  }
1016
1022
 
1017
1023
 
1018
1024
  def to_number(value: str,
1025
+ as_float: bool = False,
1019
1026
  allow_commas: bool = False,
1020
1027
  allow_multiplier_suffix: bool = False,
1021
- allow_float: bool = False,
1022
1028
  fallback: Optional[Union[int, float]] = None) -> Optional[Union[int, float]]:
1023
-
1024
1029
  """
1025
- Converts the give string value to an int, or possibly float if allow_float is True.
1026
- If allow_commas is True (default: False) then allow commas (only) every three digits.
1027
- If allow_multiplier_suffix is True (default: False) allow any of K, Kb, KB; or M, Mb, MB;
1028
- or G, Gb, or GB; or T, Tb, TB, to mean multiply value by one thousand; one million;
1029
- one billion; or one trillion; respectively. If allow_float is True (default: False)
1030
- allow the value to be floating point (i.e. with a decimal point and a fractional part),
1031
- in which case the returned value will be of type float, if it needs to be, and not int.
1032
- If the string is not well formated then returns None.
1030
+ Converts the given string value to an int, or float if as_float is True, or None if malformed.
1031
+ If allow_commas is True then allow commas (only) every three digits. If allow_multiplier_suffix
1032
+ is True allow any of K, KB; or M, MB; or G, or GB; or T, TB (case-insensitive), to mean multiply
1033
+ value by one thousand; one million; one billion; or one trillion; respectively. If as_float is True
1034
+ then value is parsed and returned as a float rather than int. Note that even if as_float is False,
1035
+ values that might look like a float, can be an int, for example, "1.5K", returns 1500 as an int;
1036
+ but "1.5002K" returns None, i.e. since 1.5002K is 1500.2 which is not an int.
1033
1037
  """
1038
+
1034
1039
  if not (isinstance(value, str) and (value := value.strip())):
1035
- return value if isinstance(value, (int, float)) else fallback
1040
+ if as_float is True:
1041
+ return float(value) if isinstance(value, (float, int)) else fallback
1042
+ else:
1043
+ return value if isinstance(value, int) else fallback
1036
1044
 
1037
1045
  value_multiplier = 1
1038
1046
  value_negative = False
@@ -1048,22 +1056,28 @@ def to_number(value: str,
1048
1056
 
1049
1057
  if allow_multiplier_suffix is True:
1050
1058
  value_upper = value.upper()
1051
- for suffix in _MULTIPLIER_SUFFIXES:
1059
+ for suffix in _TO_NUMBER_MULTIPLIER_SUFFIXES:
1052
1060
  if value_upper.endswith(suffix):
1053
- value_multiplier *= _MULTIPLIER_SUFFIXES[suffix]
1061
+ value_multiplier *= _TO_NUMBER_MULTIPLIER_SUFFIXES[suffix]
1054
1062
  if not (value := value[:-len(suffix)].strip()):
1055
1063
  return fallback
1056
1064
  break
1057
1065
 
1058
- if allow_float is True:
1066
+ if (allow_multiplier_suffix is True) or (as_float is True):
1067
+ # Allow for example "1.5K" to mean 1500 (integer).
1059
1068
  if (dot_index := value.rfind(".")) >= 0:
1060
1069
  if value_fraction := value[dot_index + 1:].strip():
1061
1070
  try:
1062
- # value_fraction = float(f"0.{value_fraction}")
1063
1071
  value_fraction = float(f"0.{value_fraction}")
1064
1072
  except Exception:
1065
1073
  return fallback
1066
- value = value[:dot_index].strip()
1074
+ if not (value := value[:dot_index].strip()):
1075
+ if not value_fraction:
1076
+ return fallback
1077
+ value = "0"
1078
+ elif (as_float is not True) and (value_dot_zero_suffix := re.search(r"\.0*$", value)):
1079
+ # Allow for example "123.00" to mean 123 (integer).
1080
+ value = value[:value_dot_zero_suffix.start()]
1067
1081
 
1068
1082
  if (allow_commas is True) and ("," in value):
1069
1083
  if not re.fullmatch(r"(-?\d{1,3}(,\d{3})*)", value):
@@ -1073,28 +1087,23 @@ def to_number(value: str,
1073
1087
  if not value.isdigit():
1074
1088
  return fallback
1075
1089
 
1076
- result = int(value)
1090
+ value = float(value) if as_float is True else int(value)
1077
1091
 
1078
1092
  if value_fraction:
1079
- result += value_fraction
1080
-
1081
- result *= value_multiplier
1093
+ value_float = (float(value) + value_fraction) * float(value_multiplier)
1094
+ if as_float is True:
1095
+ value = value_float
1096
+ else:
1097
+ value = int(value_float)
1098
+ if value_float != value:
1099
+ return fallback
1100
+ else:
1101
+ value *= value_multiplier
1082
1102
 
1083
1103
  if value_negative:
1084
- result = -result
1085
-
1086
- if allow_float is True:
1087
- if result == int(result):
1088
- result = int(result)
1089
-
1090
- return result
1104
+ value = -value
1091
1105
 
1092
-
1093
- def to_float(value: str, fallback: Optional[Any] = None) -> Optional[Any]:
1094
- try:
1095
- return float(value)
1096
- except Exception:
1097
- return fallback
1106
+ return value
1098
1107
 
1099
1108
 
1100
1109
  def to_boolean(value: str, fallback: Optional[Any]) -> Optional[Any]:
@@ -1106,12 +1115,14 @@ def to_boolean(value: str, fallback: Optional[Any]) -> Optional[Any]:
1106
1115
  return fallback
1107
1116
 
1108
1117
 
1109
- def to_enum(value: str, enumerators: List[str]) -> Optional[str]:
1118
+ def to_enum(value: str, enumerators: List[str], fuzzy: bool = False) -> Optional[str]:
1110
1119
  matches = []
1111
1120
  if isinstance(value, str) and (value := value.strip()) and isinstance(enumerators, List):
1112
1121
  enum_specifiers = {str(enum).lower(): enum for enum in enumerators}
1113
1122
  if (enum_value := enum_specifiers.get(lower_value := value.lower())) is not None:
1114
1123
  return enum_value
1124
+ if fuzzy is not True:
1125
+ return value
1115
1126
  for enum_canonical, _ in enum_specifiers.items():
1116
1127
  if enum_canonical.startswith(lower_value):
1117
1128
  matches.append(enum_canonical)
@@ -13,7 +13,7 @@ from dcicutils.data_readers import CsvReader, Excel, RowReader
13
13
  from dcicutils.datetime_utils import normalize_date_string, normalize_datetime_string
14
14
  from dcicutils.misc_utils import (create_dict, create_readonly_object, is_uuid, load_json_if,
15
15
  merge_objects, remove_empty_properties, right_trim, split_string,
16
- to_boolean, to_enum, to_integer, to_number, VirtualApp)
16
+ to_boolean, to_enum, to_float, to_integer, VirtualApp)
17
17
  from dcicutils.portal_object_utils import PortalObject
18
18
  from dcicutils.portal_utils import Portal as PortalBase
19
19
  from dcicutils.submitr.progress_constants import PROGRESS_PARSE as PROGRESS
@@ -727,9 +727,9 @@ class Schema(SchemaBase):
727
727
  allow_multiplier_suffix = typeinfo.get("allow_multiplier_suffix") is True
728
728
  def map_integer(value: str, src: Optional[str]) -> Any: # noqa
729
729
  nonlocal allow_commas, allow_multiplier_suffix
730
- return to_number(value, fallback=value, allow_float=False,
731
- allow_commas=allow_commas,
732
- allow_multiplier_suffix=allow_multiplier_suffix)
730
+ return to_integer(value, fallback=value,
731
+ allow_commas=allow_commas,
732
+ allow_multiplier_suffix=allow_multiplier_suffix)
733
733
  return map_integer
734
734
 
735
735
  def _map_function_number(self, typeinfo: dict) -> Callable:
@@ -737,9 +737,9 @@ class Schema(SchemaBase):
737
737
  allow_multiplier_suffix = typeinfo.get("allow_multiplier_suffix") is True
738
738
  def map_number(value: str, src: Optional[str]) -> Any: # noqa
739
739
  nonlocal allow_commas, allow_multiplier_suffix
740
- return to_number(value, fallback=value, allow_float=True,
741
- allow_commas=allow_commas,
742
- allow_multiplier_suffix=allow_multiplier_suffix)
740
+ return to_float(value, fallback=value,
741
+ allow_commas=allow_commas,
742
+ allow_multiplier_suffix=allow_multiplier_suffix)
743
743
  return map_number
744
744
 
745
745
  def _map_function_string(self, typeinfo: dict) -> Callable:
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "dcicutils"
3
- version = "8.13.3.1b24" # TODO: To become 8.14.0
3
+ version = "8.13.3.1b26" # TODO: To become 8.14.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"