dycw-utilities 0.160.1__tar.gz → 0.161.0__tar.gz

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.
Files changed (211) hide show
  1. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/PKG-INFO +1 -1
  2. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/pyproject.toml +2 -2
  3. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/conftest.py +3 -1
  4. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_polars.py +32 -43
  5. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_types.py +2 -3
  6. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_whenever.py +4 -9
  7. dycw_utilities-0.161.0/src/tests/test_zoneinfo.py +138 -0
  8. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/__init__.py +1 -1
  9. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/hypothesis.py +2 -2
  10. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/polars.py +8 -12
  11. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/types.py +5 -0
  12. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/whenever.py +8 -25
  13. dycw_utilities-0.161.0/src/utilities/zoneinfo.py +133 -0
  14. dycw_utilities-0.160.1/src/tests/test_zoneinfo.py +0 -103
  15. dycw_utilities-0.160.1/src/utilities/zoneinfo.py +0 -75
  16. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/.gitignore +0 -0
  17. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/LICENSE +0 -0
  18. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/README.md +0 -0
  19. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/__init__.py +0 -0
  20. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/__init__.py +0 -0
  21. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/__init__.py +0 -0
  22. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/module.py +0 -0
  23. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/__init__.py +0 -0
  24. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_1.py +0 -0
  25. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_2.py +0 -0
  26. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/__init__.py +0 -0
  27. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_1.py +0 -0
  28. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_2.py +0 -0
  29. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_3.py +0 -0
  30. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_without/__init__.py +0 -0
  31. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_1.py +0 -0
  32. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_2.py +0 -0
  33. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/standalone.py +0 -0
  34. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/modules/with_imports.py +0 -0
  35. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__obj.json +0 -0
  36. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__series.json +0 -0
  37. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_int.json +0 -0
  38. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__false.json +0 -0
  39. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__true.json +0 -0
  40. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_nested.json +0 -0
  41. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_dataframe.json +0 -0
  42. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_series.json +0 -0
  43. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_altair.py +0 -0
  44. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_asyncio.py +0 -0
  45. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_atomicwrites.py +0 -0
  46. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_atools.py +0 -0
  47. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_cachetools.py +0 -0
  48. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_click.py +0 -0
  49. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_concurrent.py +0 -0
  50. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_contextlib.py +0 -0
  51. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_contextvars.py +0 -0
  52. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_cryptography.py +0 -0
  53. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_cvxpy.py +0 -0
  54. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_dataclasses.py +0 -0
  55. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_enum.py +0 -0
  56. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_errors.py +0 -0
  57. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_eventkit.py +0 -0
  58. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_fastapi.py +0 -0
  59. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_fpdf2.py +0 -0
  60. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_functions.py +0 -0
  61. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_functools.py +0 -0
  62. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_getpass.py +0 -0
  63. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_gzip.py +0 -0
  64. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_hashlib.py +0 -0
  65. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_http.py +0 -0
  66. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_hypothesis.py +0 -0
  67. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_importlib.py +0 -0
  68. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_inflect.py +0 -0
  69. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_ipython.py +0 -0
  70. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_iterables.py +0 -0
  71. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_json.py +0 -0
  72. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_jupyter.py +0 -0
  73. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_libcst.py +0 -0
  74. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_lightweight_charts.py +0 -0
  75. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_logging.py +0 -0
  76. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_math.py +0 -0
  77. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_memory_profiler.py +0 -0
  78. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_modules.py +0 -0
  79. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_more_itertools.py +0 -0
  80. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_numpy.py +0 -0
  81. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_objects/__init__.py +0 -0
  82. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_objects/objects.py +0 -0
  83. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_operator.py +0 -0
  84. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_optuna.py +0 -0
  85. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_orjson.py +0 -0
  86. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_os.py +0 -0
  87. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_parse.py +0 -0
  88. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pathlib.py +0 -0
  89. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pickle.py +0 -0
  90. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_platform.py +0 -0
  91. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_polars_ols.py +0 -0
  92. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_postgres.py +0 -0
  93. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pottery.py +0 -0
  94. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pqdm.py +0 -0
  95. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_psutil.py +0 -0
  96. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pyinstrument.py +0 -0
  97. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pytest.py +0 -0
  98. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pytest_randomly.py +0 -0
  99. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_pytest_regressions.py +0 -0
  100. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_random.py +0 -0
  101. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_re.py +0 -0
  102. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_redis.py +0 -0
  103. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_reprlib.py +0 -0
  104. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_scipy.py +0 -0
  105. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_sentinel.py +0 -0
  106. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_shelve.py +0 -0
  107. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_slack_sdk.py +0 -0
  108. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_socket.py +0 -0
  109. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy.py +0 -0
  110. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy_polars.py +0 -0
  111. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_statsmodels.py +0 -0
  112. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_string.py +0 -0
  113. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_tempfile.py +0 -0
  114. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_text.py +0 -0
  115. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_threading.py +0 -0
  116. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_timer.py +0 -0
  117. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_traceback.py +0 -0
  118. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_typed_settings.py +0 -0
  119. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_typing.py +0 -0
  120. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/__init__.py +0 -0
  121. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/no_future.py +0 -0
  122. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/with_future.py +0 -0
  123. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_tzdata.py +0 -0
  124. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_tzlocal.py +0 -0
  125. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_uuid.py +0 -0
  126. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_version.py +0 -0
  127. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_warnings.py +0 -0
  128. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/tests/test_zipfile.py +0 -0
  129. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/altair.py +0 -0
  130. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/asyncio.py +0 -0
  131. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/atomicwrites.py +0 -0
  132. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/atools.py +0 -0
  133. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/cachetools.py +0 -0
  134. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/click.py +0 -0
  135. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/concurrent.py +0 -0
  136. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/contextlib.py +0 -0
  137. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/contextvars.py +0 -0
  138. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/cryptography.py +0 -0
  139. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/cvxpy.py +0 -0
  140. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/dataclasses.py +0 -0
  141. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/enum.py +0 -0
  142. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/errors.py +0 -0
  143. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/eventkit.py +0 -0
  144. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/fastapi.py +0 -0
  145. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/fpdf2.py +0 -0
  146. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/functions.py +0 -0
  147. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/functools.py +0 -0
  148. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/getpass.py +0 -0
  149. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/gzip.py +0 -0
  150. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/hashlib.py +0 -0
  151. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/http.py +0 -0
  152. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/importlib.py +0 -0
  153. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/inflect.py +0 -0
  154. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/ipython.py +0 -0
  155. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/iterables.py +0 -0
  156. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/json.py +0 -0
  157. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/jupyter.py +0 -0
  158. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/libcst.py +0 -0
  159. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/lightweight_charts.py +0 -0
  160. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/logging.py +0 -0
  161. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/math.py +0 -0
  162. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/memory_profiler.py +0 -0
  163. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/modules.py +0 -0
  164. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/more_itertools.py +0 -0
  165. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/numpy.py +0 -0
  166. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/operator.py +0 -0
  167. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/optuna.py +0 -0
  168. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/orjson.py +0 -0
  169. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/os.py +0 -0
  170. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/parse.py +0 -0
  171. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pathlib.py +0 -0
  172. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pickle.py +0 -0
  173. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/platform.py +0 -0
  174. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/polars_ols.py +0 -0
  175. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/postgres.py +0 -0
  176. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pottery.py +0 -0
  177. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pqdm.py +0 -0
  178. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/psutil.py +0 -0
  179. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/py.typed +0 -0
  180. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pyinstrument.py +0 -0
  181. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pytest.py +0 -0
  182. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/__init__.py +0 -0
  183. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_randomly.py +0 -0
  184. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_regressions.py +0 -0
  185. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/pytest_regressions.py +0 -0
  186. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/random.py +0 -0
  187. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/re.py +0 -0
  188. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/redis.py +0 -0
  189. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/reprlib.py +0 -0
  190. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/scipy.py +0 -0
  191. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/sentinel.py +0 -0
  192. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/shelve.py +0 -0
  193. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/slack_sdk.py +0 -0
  194. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/socket.py +0 -0
  195. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy.py +0 -0
  196. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy_polars.py +0 -0
  197. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/statsmodels.py +0 -0
  198. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/string.py +0 -0
  199. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/tempfile.py +0 -0
  200. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/text.py +0 -0
  201. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/threading.py +0 -0
  202. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/timer.py +0 -0
  203. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/traceback.py +0 -0
  204. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/typed_settings.py +0 -0
  205. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/typing.py +0 -0
  206. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/tzdata.py +0 -0
  207. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/tzlocal.py +0 -0
  208. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/uuid.py +0 -0
  209. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/version.py +0 -0
  210. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/warnings.py +0 -0
  211. {dycw_utilities-0.160.1 → dycw_utilities-0.161.0}/src/utilities/zipfile.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: dycw-utilities
3
- Version: 0.160.1
3
+ Version: 0.161.0
4
4
  Author-email: Derek Wan <d.wan@icloud.com>
5
5
  License-File: LICENSE
6
6
  Requires-Python: >=3.12
@@ -227,7 +227,7 @@ dependencies = [
227
227
  name = "dycw-utilities"
228
228
  readme = "README.md"
229
229
  requires-python = ">= 3.12"
230
- version = "0.160.1"
230
+ version = "0.161.0"
231
231
 
232
232
  [project.entry-points.pytest11]
233
233
  pytest-randomly = "utilities.pytest_plugins.pytest_randomly"
@@ -259,7 +259,7 @@ test = [
259
259
  # bump-my-version
260
260
  [tool.bumpversion]
261
261
  allow_dirty = true
262
- current_version = "0.160.1"
262
+ current_version = "0.161.0"
263
263
 
264
264
  [[tool.bumpversion.files]]
265
265
  filename = "src/utilities/__init__.py"
@@ -9,7 +9,7 @@ from hypothesis import HealthCheck
9
9
  from pytest import fixture, mark, param, skip
10
10
 
11
11
  from utilities.contextlib import enhanced_context_manager
12
- from utilities.platform import IS_MAC, IS_NOT_LINUX, IS_WINDOWS
12
+ from utilities.platform import IS_LINUX, IS_MAC, IS_NOT_LINUX, IS_WINDOWS
13
13
  from utilities.re import ExtractGroupError, extract_group
14
14
  from utilities.whenever import MINUTE, get_now, parse_plain_local
15
15
 
@@ -29,11 +29,13 @@ SKIPIF_CI = mark.skipif(IS_CI, reason="Skipped for CI")
29
29
  IS_CI_AND_NOT_LINUX = IS_CI and IS_NOT_LINUX
30
30
  IS_CI_AND_WINDOWS = IS_CI and IS_WINDOWS
31
31
  IS_CI_AND_MAC = IS_CI and IS_MAC
32
+ IS_CI_AND_LINUX = IS_CI and IS_LINUX
32
33
  SKIPIF_CI_AND_NOT_LINUX = mark.skipif(
33
34
  IS_CI_AND_NOT_LINUX, reason="Skipped for CI/non-Linux"
34
35
  )
35
36
  SKIPIF_CI_AND_WINDOWS = mark.skipif(IS_CI_AND_WINDOWS, reason="Skipped for CI/Windows")
36
37
  SKIPIF_CI_AND_MAC = mark.skipif(IS_CI_AND_MAC, reason="Skipped for CI/Mac")
38
+ SKIPIF_CI_AND_LINUX = mark.skipif(IS_CI_AND_LINUX, reason="Skipped for CI/Linux")
37
39
 
38
40
 
39
41
  # hypothesis
@@ -60,7 +60,6 @@ from pytest import mark, param, raises
60
60
  from whenever import DateDelta, DateTimeDelta, PlainDateTime, TimeDelta, ZonedDateTime
61
61
 
62
62
  import tests.test_math
63
- import utilities.math
64
63
  import utilities.polars
65
64
  from utilities.hypothesis import (
66
65
  date_deltas,
@@ -223,7 +222,7 @@ from utilities.whenever import (
223
222
  get_now_plain,
224
223
  get_today,
225
224
  )
226
- from utilities.zoneinfo import UTC, get_time_zone_name
225
+ from utilities.zoneinfo import UTC, to_time_zone_name
227
226
 
228
227
  if TYPE_CHECKING:
229
228
  from collections.abc import Callable, Iterable, Mapping, Sequence
@@ -232,7 +231,6 @@ if TYPE_CHECKING:
232
231
 
233
232
  from _pytest.mark import ParameterSet
234
233
  from polars._typing import IntoExprColumn, PolarsDataType, SchemaDict
235
- from polars.datatypes import DataTypeClass
236
234
 
237
235
  from utilities.types import MaybeType, StrMapping, WeekDay
238
236
 
@@ -1428,18 +1426,18 @@ class TestDataClassToSchema:
1428
1426
 
1429
1427
 
1430
1428
  class TestDatetimeDTypes:
1431
- @given(
1432
- case=sampled_from([
1433
- (HongKong, DatetimeHongKong),
1434
- (Tokyo, DatetimeTokyo),
1435
- (USCentral, DatetimeUSCentral),
1436
- (USEastern, DatetimeUSEastern),
1437
- (UTC, DatetimeUTC),
1438
- ])
1429
+ @mark.parametrize(
1430
+ ("time_zone", "dtype"),
1431
+ [
1432
+ param(HongKong, DatetimeHongKong),
1433
+ param(Tokyo, DatetimeTokyo),
1434
+ param(USCentral, DatetimeUSCentral),
1435
+ param(USEastern, DatetimeUSEastern),
1436
+ param(UTC, DatetimeUTC),
1437
+ ],
1439
1438
  )
1440
- def test_main(self, *, case: tuple[ZoneInfo, Datetime]) -> None:
1441
- time_zone, dtype = case
1442
- name = get_time_zone_name(time_zone)
1439
+ def test_main(self, *, time_zone: ZoneInfo, dtype: Datetime) -> None:
1440
+ name = to_time_zone_name(time_zone)
1443
1441
  expected = dtype.time_zone
1444
1442
  assert name == expected
1445
1443
 
@@ -2156,36 +2154,27 @@ class TestMapOverColumns:
2156
2154
 
2157
2155
 
2158
2156
  class TestNanSumAgg:
2159
- @given(
2160
- case=sampled_from([
2161
- ([None], None),
2162
- ([None, None], None),
2163
- ([0], 0),
2164
- ([0, None], 0),
2165
- ([0, None, None], 0),
2166
- ([1, 2], 3),
2167
- ([1, 2, None], 3),
2168
- ([1, 2, None, None], 3),
2169
- ]),
2170
- dtype=sampled_from([Int64, Float64]),
2171
- mode=sampled_from(["str", "column"]),
2157
+ @mark.parametrize(
2158
+ ("values", "expected"),
2159
+ [
2160
+ param([None], None),
2161
+ param([None, None], None),
2162
+ param([0], 0),
2163
+ param([0, None], 0),
2164
+ param([0, None, None], 0),
2165
+ param([1, 2], 3),
2166
+ param([1, 2, None], 3),
2167
+ param([1, 2, None, None], 3),
2168
+ ],
2172
2169
  )
2173
- def test_main(
2174
- self,
2175
- *,
2176
- case: tuple[list[Any], int | None],
2177
- dtype: DataTypeClass,
2178
- mode: Literal["str", "column"],
2179
- ) -> None:
2180
- values, expected = case
2181
- df = DataFrame(data=values, schema={"value": dtype}).with_columns(id=lit("id"))
2182
- match mode:
2183
- case "str":
2184
- agg = "value"
2185
- case "column":
2186
- agg = col("value")
2187
- result = df.group_by("id").agg(nan_sum_agg(agg))
2188
- assert result["value"].item() == expected
2170
+ def test_main(self, *, values: list[int | None], expected: int | None) -> None:
2171
+ df = (
2172
+ Series(name="x", values=values, dtype=Int64)
2173
+ .to_frame()
2174
+ .with_columns(id=lit("id"))
2175
+ )
2176
+ result = df.group_by("id").agg(nan_sum_agg("x"))
2177
+ assert result.item(0, "x") == expected
2189
2178
 
2190
2179
 
2191
2180
  class TestNanSumHorizontal:
@@ -9,8 +9,7 @@ from hypothesis.strategies import sampled_from
9
9
  from pytest import mark, param
10
10
 
11
11
  from utilities.platform import SYSTEM
12
- from utilities.types import Dataclass, Number, PathLike, TimeZone
13
- from utilities.typing import get_literal_elements
12
+ from utilities.types import TIME_ZONES, Dataclass, Number, PathLike
14
13
 
15
14
 
16
15
  class TestDataClassProtocol:
@@ -45,7 +44,7 @@ class TestPathLike:
45
44
 
46
45
  class TestTimeZone:
47
46
  def test_main(self) -> None:
48
- result = set(get_literal_elements(TimeZone))
47
+ result = set(TIME_ZONES)
49
48
  expected = available_timezones()
50
49
  match SYSTEM:
51
50
  case "windows" | "mac":
@@ -150,7 +150,7 @@ from utilities.whenever import (
150
150
  to_zoned_date_time,
151
151
  two_digit_year_month,
152
152
  )
153
- from utilities.zoneinfo import UTC, get_time_zone_name
153
+ from utilities.zoneinfo import UTC
154
154
 
155
155
  if TYPE_CHECKING:
156
156
  from collections.abc import Callable
@@ -418,7 +418,8 @@ class TestGetNow:
418
418
 
419
419
  def test_constant(self) -> None:
420
420
  assert isinstance(NOW_UTC, ZonedDateTime)
421
- assert NOW_UTC.tz == "UTC"
421
+ expected = UTC.key
422
+ assert NOW_UTC.tz == expected
422
423
 
423
424
 
424
425
  class TestGetNowLocal:
@@ -1466,12 +1467,6 @@ class TestWheneverLogRecord:
1466
1467
  def test_get_length(self) -> None:
1467
1468
  assert isinstance(WheneverLogRecord._get_length(), int)
1468
1469
 
1469
- def test_get_time_zone(self) -> None:
1470
- assert isinstance(WheneverLogRecord._get_time_zone(), ZoneInfo)
1471
-
1472
- def test_get_time_zone_key(self) -> None:
1473
- assert isinstance(WheneverLogRecord._get_time_zone_key(), str)
1474
-
1475
1470
 
1476
1471
  class TestZonedDateTimePeriod:
1477
1472
  @given(period=zoned_date_time_periods(), delta=time_deltas())
@@ -1669,7 +1664,7 @@ class TestZonedDateTimePeriod:
1669
1664
  with assume_does_not_raise(OverflowError, match="date value out of range"):
1670
1665
  result = period.to_tz(UTC)
1671
1666
  assert result.time_zone == UTC
1672
- name = get_time_zone_name(UTC)
1667
+ name = UTC.key
1673
1668
  expected = ZonedDateTimePeriod(period.start.to_tz(name), period.end.to_tz(name))
1674
1669
  assert result == expected
1675
1670
 
@@ -0,0 +1,138 @@
1
+ from __future__ import annotations
2
+
3
+ import datetime as dt
4
+ from typing import TYPE_CHECKING, cast
5
+ from zoneinfo import ZoneInfo
6
+
7
+ from hypothesis import given
8
+ from hypothesis.strategies import (
9
+ DataObject,
10
+ data,
11
+ datetimes,
12
+ just,
13
+ sampled_from,
14
+ timezones,
15
+ )
16
+ from pytest import raises
17
+
18
+ from tests.conftest import SKIPIF_CI_AND_LINUX
19
+ from utilities.hypothesis import zoned_date_times
20
+ from utilities.tzdata import HongKong, Tokyo
21
+ from utilities.tzlocal import LOCAL_TIME_ZONE, LOCAL_TIME_ZONE_NAME
22
+ from utilities.zoneinfo import (
23
+ UTC,
24
+ _ToTimeZoneNameInvalidKeyError,
25
+ _ToTimeZoneNameInvalidTZInfoError,
26
+ _ToTimeZoneNamePlainDateTimeError,
27
+ _ToZoneInfoInvalidTZInfoError,
28
+ _ToZoneInfoPlainDateTimeError,
29
+ to_time_zone_name,
30
+ to_zone_info,
31
+ )
32
+
33
+ if TYPE_CHECKING:
34
+ from utilities.types import TimeZoneLike
35
+
36
+
37
+ class TestToZoneInfo:
38
+ @given(time_zone=timezones())
39
+ def test_zone_info(self, *, time_zone: ZoneInfo) -> None:
40
+ result = to_zone_info(time_zone)
41
+ assert result is time_zone
42
+
43
+ @given(data=data(), time_zone=timezones())
44
+ def test_zoned_date_time(self, *, data: DataObject, time_zone: ZoneInfo) -> None:
45
+ date_time = data.draw(zoned_date_times(time_zone=time_zone))
46
+ result = to_zone_info(date_time)
47
+ assert result is time_zone
48
+
49
+ def test_local(self) -> None:
50
+ result = to_zone_info("local")
51
+ assert result is LOCAL_TIME_ZONE
52
+
53
+ @given(time_zone=timezones())
54
+ def test_str(self, *, time_zone: ZoneInfo) -> None:
55
+ result = to_zone_info(cast("TimeZoneLike", time_zone.key))
56
+ assert result is time_zone
57
+
58
+ def test_tz_info(self) -> None:
59
+ result = to_zone_info(dt.UTC)
60
+ assert result is UTC
61
+
62
+ @given(data=data(), time_zone=timezones())
63
+ def test_py_zoned_date_time(self, *, data: DataObject, time_zone: ZoneInfo) -> None:
64
+ date_time = data.draw(datetimes(timezones=just(time_zone)))
65
+ result = to_zone_info(date_time)
66
+ assert result is time_zone
67
+
68
+ def test_error_invalid_tz_info(self) -> None:
69
+ time_zone = dt.timezone(dt.timedelta(hours=12))
70
+ with raises(
71
+ _ToZoneInfoInvalidTZInfoError, match=r"Invalid time-zone: UTC\+12:00"
72
+ ):
73
+ _ = to_zone_info(time_zone)
74
+
75
+ @given(date_time=datetimes())
76
+ def test_error_plain_date_time(self, *, date_time: dt.datetime) -> None:
77
+ with raises(_ToZoneInfoPlainDateTimeError, match="Plain date-time: .*"):
78
+ _ = to_zone_info(date_time)
79
+
80
+
81
+ class TestToTimeZoneName:
82
+ @given(time_zone=timezones())
83
+ def test_zone_info(self, *, time_zone: ZoneInfo) -> None:
84
+ result = to_time_zone_name(time_zone)
85
+ expected = time_zone.key
86
+ assert result == expected
87
+
88
+ @given(data=data(), time_zone=timezones())
89
+ def test_zoned_date_time(self, *, data: DataObject, time_zone: ZoneInfo) -> None:
90
+ date_time = data.draw(zoned_date_times(time_zone=time_zone))
91
+ result = to_time_zone_name(date_time)
92
+ expected = time_zone.key
93
+ assert result == expected
94
+
95
+ def test_local(self) -> None:
96
+ result = to_time_zone_name("local")
97
+ assert result == LOCAL_TIME_ZONE_NAME
98
+
99
+ @given(time_zone=timezones())
100
+ @SKIPIF_CI_AND_LINUX
101
+ def test_str(self, *, time_zone: ZoneInfo) -> None:
102
+ result = to_time_zone_name(cast("TimeZoneLike", time_zone.key))
103
+ expected = time_zone.key
104
+ assert result == expected
105
+
106
+ def test_tz_info(self) -> None:
107
+ result = to_time_zone_name(dt.UTC)
108
+ expected = UTC.key
109
+ assert result == expected
110
+
111
+ @given(data=data(), time_zone=timezones())
112
+ def test_py_zoned_date_time(self, *, data: DataObject, time_zone: ZoneInfo) -> None:
113
+ date_time = data.draw(datetimes(timezones=just(time_zone)))
114
+ result = to_time_zone_name(date_time)
115
+ expected = time_zone.key
116
+ assert result == expected
117
+
118
+ def test_error_invalid_key(self) -> None:
119
+ with raises(
120
+ _ToTimeZoneNameInvalidKeyError, match="Invalid time-zone: 'invalid'"
121
+ ):
122
+ _ = to_time_zone_name(cast("TimeZoneLike", "invalid"))
123
+
124
+ def test_error_invalid_tz_info(self) -> None:
125
+ time_zone = dt.timezone(dt.timedelta(hours=12))
126
+ with raises(_ToTimeZoneNameInvalidTZInfoError, match="Invalid time-zone: .*"):
127
+ _ = to_time_zone_name(time_zone)
128
+
129
+ @given(date_time=datetimes())
130
+ def test_error_plain_date_time(self, *, date_time: dt.datetime) -> None:
131
+ with raises(_ToTimeZoneNamePlainDateTimeError, match="Plain date-time: .*"):
132
+ _ = to_time_zone_name(date_time)
133
+
134
+
135
+ class TestTimeZones:
136
+ @given(time_zone=sampled_from([HongKong, Tokyo, UTC]))
137
+ def test_main(self, *, time_zone: ZoneInfo) -> None:
138
+ assert isinstance(time_zone, ZoneInfo)
@@ -1,3 +1,3 @@
1
1
  from __future__ import annotations
2
2
 
3
- __version__ = "0.160.1"
3
+ __version__ = "0.161.0"
@@ -101,7 +101,7 @@ from utilities.whenever import (
101
101
  to_days,
102
102
  to_nanoseconds,
103
103
  )
104
- from utilities.zoneinfo import UTC, ensure_time_zone
104
+ from utilities.zoneinfo import UTC, to_zone_info
105
105
 
106
106
  if TYPE_CHECKING:
107
107
  from collections.abc import Collection, Hashable, Iterable, Iterator
@@ -1509,7 +1509,7 @@ def zoned_date_times(
1509
1509
  ) -> ZonedDateTime:
1510
1510
  """Strategy for generating zoned date-times."""
1511
1511
  min_value_, max_value_ = [draw2(draw, v) for v in [min_value, max_value]]
1512
- time_zone_ = ensure_time_zone(draw2(draw, time_zone))
1512
+ time_zone_ = to_zone_info(draw2(draw, time_zone))
1513
1513
  match min_value_:
1514
1514
  case None | PlainDateTime():
1515
1515
  ...
@@ -104,7 +104,7 @@ from utilities.whenever import (
104
104
  ZonedDateTimePeriod,
105
105
  to_py_time_delta,
106
106
  )
107
- from utilities.zoneinfo import UTC, ensure_time_zone, get_time_zone_name
107
+ from utilities.zoneinfo import UTC, to_time_zone_name
108
108
 
109
109
  if TYPE_CHECKING:
110
110
  from collections.abc import Callable, Iterable, Iterator, Mapping, Sequence
@@ -834,7 +834,7 @@ def convert_time_zone(
834
834
 
835
835
  def _convert_time_zone_one(sr: Series, /, *, time_zone: TimeZoneLike = UTC) -> Series:
836
836
  if isinstance(sr.dtype, Datetime):
837
- return sr.dt.convert_time_zone(get_time_zone_name(time_zone))
837
+ return sr.dt.convert_time_zone(to_time_zone_name(time_zone))
838
838
  return sr
839
839
 
840
840
 
@@ -1906,14 +1906,10 @@ def _map_over_series_one(func: Callable[[Series], Series], series: Series, /) ->
1906
1906
  ##
1907
1907
 
1908
1908
 
1909
- def nan_sum_agg(column: str | Expr, /, *, dtype: PolarsDataType | None = None) -> Expr:
1909
+ def nan_sum_agg(column: str | Expr, /) -> Expr:
1910
1910
  """Nan sum aggregation."""
1911
1911
  col_use = col(column) if isinstance(column, str) else column
1912
- return (
1913
- when(col_use.is_not_null().any())
1914
- .then(col_use.sum())
1915
- .otherwise(lit(None, dtype=dtype))
1916
- )
1912
+ return when(col_use.is_not_null().any()).then(col_use.sum())
1917
1913
 
1918
1914
 
1919
1915
  ##
@@ -2114,7 +2110,7 @@ def period_range(
2114
2110
  eager: bool = False,
2115
2111
  ) -> Series | Expr:
2116
2112
  """Construct a period range."""
2117
- time_zone_use = None if time_zone is None else ensure_time_zone(time_zone).key
2113
+ time_zone_use = None if time_zone is None else to_time_zone_name(time_zone)
2118
2114
  match end_or_length:
2119
2115
  case ZonedDateTime() as end:
2120
2116
  ...
@@ -2131,7 +2127,7 @@ def period_range(
2131
2127
  time_zone=time_zone_use,
2132
2128
  eager=eager,
2133
2129
  ).alias("start")
2134
- ends = (starts.dt.offset_by(interval)).alias("end")
2130
+ ends = starts.dt.offset_by(interval).alias("end")
2135
2131
  period = struct(starts, ends)
2136
2132
  return try_reify_expr(period, starts, ends)
2137
2133
 
@@ -2224,7 +2220,7 @@ def _replace_time_zone_one(
2224
2220
  sr: Series, /, *, time_zone: TimeZoneLike | None = UTC
2225
2221
  ) -> Series:
2226
2222
  if isinstance(sr.dtype, Datetime):
2227
- time_zone_use = None if time_zone is None else get_time_zone_name(time_zone)
2223
+ time_zone_use = None if time_zone is None else to_time_zone_name(time_zone)
2228
2224
  return sr.dt.replace_time_zone(time_zone_use)
2229
2225
  return sr
2230
2226
 
@@ -2615,7 +2611,7 @@ def zoned_date_time_dtype(
2615
2611
  *, time_unit: TimeUnit = "us", time_zone: TimeZoneLike = UTC
2616
2612
  ) -> Datetime:
2617
2613
  """Create a zoned date-time data type."""
2618
- return Datetime(time_unit=time_unit, time_zone=get_time_zone_name(time_zone))
2614
+ return Datetime(time_unit=time_unit, time_zone=to_time_zone_name(time_zone))
2619
2615
 
2620
2616
 
2621
2617
  def zoned_date_time_period_dtype(
@@ -15,6 +15,7 @@ from typing import (
15
15
  Literal,
16
16
  Protocol,
17
17
  TypeVar,
18
+ get_args,
18
19
  overload,
19
20
  runtime_checkable,
20
21
  )
@@ -271,12 +272,16 @@ type TimeZone = Literal[
271
272
  "Africa/Abidjan", "Africa/Accra", "Africa/Addis_Ababa", "Africa/Algiers", "Africa/Asmara", "Africa/Asmera", "Africa/Bamako", "Africa/Bangui", "Africa/Banjul", "Africa/Bissau", "Africa/Blantyre", "Africa/Brazzaville", "Africa/Bujumbura", "Africa/Cairo", "Africa/Casablanca", "Africa/Ceuta", "Africa/Conakry", "Africa/Dakar", "Africa/Dar_es_Salaam", "Africa/Djibouti", "Africa/Douala", "Africa/El_Aaiun", "Africa/Freetown", "Africa/Gaborone", "Africa/Harare", "Africa/Johannesburg", "Africa/Juba", "Africa/Kampala", "Africa/Khartoum", "Africa/Kigali", "Africa/Kinshasa", "Africa/Lagos", "Africa/Libreville", "Africa/Lome", "Africa/Luanda", "Africa/Lubumbashi", "Africa/Lusaka", "Africa/Malabo", "Africa/Maputo", "Africa/Maseru", "Africa/Mbabane", "Africa/Mogadishu", "Africa/Monrovia", "Africa/Nairobi", "Africa/Ndjamena", "Africa/Niamey", "Africa/Nouakchott", "Africa/Ouagadougou", "Africa/Porto-Novo", "Africa/Sao_Tome", "Africa/Timbuktu", "Africa/Tripoli", "Africa/Tunis", "Africa/Windhoek", "America/Adak", "America/Anchorage", "America/Anguilla", "America/Antigua", "America/Araguaina", "America/Argentina/Buenos_Aires", "America/Argentina/Catamarca", "America/Argentina/ComodRivadavia", "America/Argentina/Cordoba", "America/Argentina/Jujuy", "America/Argentina/La_Rioja", "America/Argentina/Mendoza", "America/Argentina/Rio_Gallegos", "America/Argentina/Salta", "America/Argentina/San_Juan", "America/Argentina/San_Luis", "America/Argentina/Tucuman", "America/Argentina/Ushuaia", "America/Aruba", "America/Asuncion", "America/Atikokan", "America/Atka", "America/Bahia", "America/Bahia_Banderas", "America/Barbados", "America/Belem", "America/Belize", "America/Blanc-Sablon", "America/Boa_Vista", "America/Bogota", "America/Boise", "America/Buenos_Aires", "America/Cambridge_Bay", "America/Campo_Grande", "America/Cancun", "America/Caracas", "America/Catamarca", "America/Cayenne", "America/Cayman", "America/Chicago", "America/Chihuahua", "America/Ciudad_Juarez", "America/Coral_Harbour", "America/Cordoba", "America/Costa_Rica", "America/Coyhaique", "America/Creston", "America/Cuiaba", "America/Curacao", "America/Danmarkshavn", "America/Dawson", "America/Dawson_Creek", "America/Denver", "America/Detroit", "America/Dominica", "America/Edmonton", "America/Eirunepe", "America/El_Salvador", "America/Ensenada", "America/Fort_Nelson", "America/Fort_Wayne", "America/Fortaleza", "America/Glace_Bay", "America/Godthab", "America/Goose_Bay", "America/Grand_Turk", "America/Grenada", "America/Guadeloupe", "America/Guatemala", "America/Guayaquil", "America/Guyana", "America/Halifax", "America/Havana", "America/Hermosillo", "America/Indiana/Indianapolis", "America/Indiana/Knox", "America/Indiana/Marengo", "America/Indiana/Petersburg", "America/Indiana/Tell_City", "America/Indiana/Vevay", "America/Indiana/Vincennes", "America/Indiana/Winamac", "America/Indianapolis", "America/Inuvik", "America/Iqaluit", "America/Jamaica", "America/Jujuy", "America/Juneau", "America/Kentucky/Louisville", "America/Kentucky/Monticello", "America/Knox_IN", "America/Kralendijk", "America/La_Paz", "America/Lima", "America/Los_Angeles", "America/Louisville", "America/Lower_Princes", "America/Maceio", "America/Managua", "America/Manaus", "America/Marigot", "America/Martinique", "America/Matamoros", "America/Mazatlan", "America/Mendoza", "America/Menominee", "America/Merida", "America/Metlakatla", "America/Mexico_City", "America/Miquelon", "America/Moncton", "America/Monterrey", "America/Montevideo", "America/Montreal", "America/Montserrat", "America/Nassau", "America/New_York", "America/Nipigon", "America/Nome", "America/Noronha", "America/North_Dakota/Beulah", "America/North_Dakota/Center", "America/North_Dakota/New_Salem", "America/Nuuk", "America/Ojinaga", "America/Panama", "America/Pangnirtung", "America/Paramaribo", "America/Phoenix", "America/Port-au-Prince", "America/Port_of_Spain", "America/Porto_Acre", "America/Porto_Velho", "America/Puerto_Rico", "America/Punta_Arenas", "America/Rainy_River", "America/Rankin_Inlet", "America/Recife", "America/Regina", "America/Resolute", "America/Rio_Branco", "America/Rosario", "America/Santa_Isabel", "America/Santarem", "America/Santiago", "America/Santo_Domingo", "America/Sao_Paulo", "America/Scoresbysund", "America/Shiprock", "America/Sitka", "America/St_Barthelemy", "America/St_Johns", "America/St_Kitts", "America/St_Lucia", "America/St_Thomas", "America/St_Vincent", "America/Swift_Current", "America/Tegucigalpa", "America/Thule", "America/Thunder_Bay", "America/Tijuana", "America/Toronto", "America/Tortola", "America/Vancouver", "America/Virgin", "America/Whitehorse", "America/Winnipeg", "America/Yakutat", "America/Yellowknife", "Antarctica/Casey", "Antarctica/Davis", "Antarctica/DumontDUrville", "Antarctica/Macquarie", "Antarctica/Mawson", "Antarctica/McMurdo", "Antarctica/Palmer", "Antarctica/Rothera", "Antarctica/South_Pole", "Antarctica/Syowa", "Antarctica/Troll", "Antarctica/Vostok", "Arctic/Longyearbyen", "Asia/Aden", "Asia/Almaty", "Asia/Amman", "Asia/Anadyr", "Asia/Aqtau", "Asia/Aqtobe", "Asia/Ashgabat", "Asia/Ashkhabad", "Asia/Atyrau", "Asia/Baghdad", "Asia/Bahrain", "Asia/Baku", "Asia/Bangkok", "Asia/Barnaul", "Asia/Beirut", "Asia/Bishkek", "Asia/Brunei", "Asia/Calcutta", "Asia/Chita", "Asia/Choibalsan", "Asia/Chongqing", "Asia/Chungking", "Asia/Colombo", "Asia/Dacca", "Asia/Damascus", "Asia/Dhaka", "Asia/Dili", "Asia/Dubai", "Asia/Dushanbe", "Asia/Famagusta", "Asia/Gaza", "Asia/Harbin", "Asia/Hebron", "Asia/Ho_Chi_Minh", "Asia/Hong_Kong", "Asia/Hovd", "Asia/Irkutsk", "Asia/Istanbul", "Asia/Jakarta", "Asia/Jayapura", "Asia/Jerusalem", "Asia/Kabul", "Asia/Kamchatka", "Asia/Karachi", "Asia/Kashgar", "Asia/Kathmandu", "Asia/Katmandu", "Asia/Khandyga", "Asia/Kolkata", "Asia/Krasnoyarsk", "Asia/Kuala_Lumpur", "Asia/Kuching", "Asia/Kuwait", "Asia/Macao", "Asia/Macau", "Asia/Magadan", "Asia/Makassar", "Asia/Manila", "Asia/Muscat", "Asia/Nicosia", "Asia/Novokuznetsk", "Asia/Novosibirsk", "Asia/Omsk", "Asia/Oral", "Asia/Phnom_Penh", "Asia/Pontianak", "Asia/Pyongyang", "Asia/Qatar", "Asia/Qostanay", "Asia/Qyzylorda", "Asia/Rangoon", "Asia/Riyadh", "Asia/Saigon", "Asia/Sakhalin", "Asia/Samarkand", "Asia/Seoul", "Asia/Shanghai", "Asia/Singapore", "Asia/Srednekolymsk", "Asia/Taipei", "Asia/Tashkent", "Asia/Tbilisi", "Asia/Tehran", "Asia/Tel_Aviv", "Asia/Thimbu", "Asia/Thimphu", "Asia/Tokyo", "Asia/Tomsk", "Asia/Ujung_Pandang", "Asia/Ulaanbaatar", "Asia/Ulan_Bator", "Asia/Urumqi", "Asia/Ust-Nera", "Asia/Vientiane", "Asia/Vladivostok", "Asia/Yakutsk", "Asia/Yangon", "Asia/Yekaterinburg", "Asia/Yerevan", "Atlantic/Azores", "Atlantic/Bermuda", "Atlantic/Canary", "Atlantic/Cape_Verde", "Atlantic/Faeroe", "Atlantic/Faroe", "Atlantic/Jan_Mayen", "Atlantic/Madeira", "Atlantic/Reykjavik", "Atlantic/South_Georgia", "Atlantic/St_Helena", "Atlantic/Stanley", "Australia/ACT", "Australia/Adelaide", "Australia/Brisbane", "Australia/Broken_Hill", "Australia/Canberra", "Australia/Currie", "Australia/Darwin", "Australia/Eucla", "Australia/Hobart", "Australia/LHI", "Australia/Lindeman", "Australia/Lord_Howe", "Australia/Melbourne", "Australia/NSW", "Australia/North", "Australia/Perth", "Australia/Queensland", "Australia/South", "Australia/Sydney", "Australia/Tasmania", "Australia/Victoria", "Australia/West", "Australia/Yancowinna", "Brazil/Acre", "Brazil/DeNoronha", "Brazil/East", "Brazil/West", "CET", "CST6CDT", "Canada/Atlantic", "Canada/Central", "Canada/Eastern", "Canada/Mountain", "Canada/Newfoundland", "Canada/Pacific", "Canada/Saskatchewan", "Canada/Yukon", "Chile/Continental", "Chile/EasterIsland", "Cuba", "EET", "EST", "EST5EDT", "Egypt", "Eire", "Etc/GMT", "Etc/GMT+0", "Etc/GMT+1", "Etc/GMT+10", "Etc/GMT+11", "Etc/GMT+12", "Etc/GMT+2", "Etc/GMT+3", "Etc/GMT+4", "Etc/GMT+5", "Etc/GMT+6", "Etc/GMT+7", "Etc/GMT+8", "Etc/GMT+9", "Etc/GMT-0", "Etc/GMT-1", "Etc/GMT-10", "Etc/GMT-11", "Etc/GMT-12", "Etc/GMT-13", "Etc/GMT-14", "Etc/GMT-2", "Etc/GMT-3", "Etc/GMT-4", "Etc/GMT-5", "Etc/GMT-6", "Etc/GMT-7", "Etc/GMT-8", "Etc/GMT-9", "Etc/GMT0", "Etc/Greenwich", "Etc/UCT", "Etc/UTC", "Etc/Universal", "Etc/Zulu", "Europe/Amsterdam", "Europe/Andorra", "Europe/Astrakhan", "Europe/Athens", "Europe/Belfast", "Europe/Belgrade", "Europe/Berlin", "Europe/Bratislava", "Europe/Brussels", "Europe/Bucharest", "Europe/Budapest", "Europe/Busingen", "Europe/Chisinau", "Europe/Copenhagen", "Europe/Dublin", "Europe/Gibraltar", "Europe/Guernsey", "Europe/Helsinki", "Europe/Isle_of_Man", "Europe/Istanbul", "Europe/Jersey", "Europe/Kaliningrad", "Europe/Kiev", "Europe/Kirov", "Europe/Kyiv", "Europe/Lisbon", "Europe/Ljubljana", "Europe/London", "Europe/Luxembourg", "Europe/Madrid", "Europe/Malta", "Europe/Mariehamn", "Europe/Minsk", "Europe/Monaco", "Europe/Moscow", "Europe/Nicosia", "Europe/Oslo", "Europe/Paris", "Europe/Podgorica", "Europe/Prague", "Europe/Riga", "Europe/Rome", "Europe/Samara", "Europe/San_Marino", "Europe/Sarajevo", "Europe/Saratov", "Europe/Simferopol", "Europe/Skopje", "Europe/Sofia", "Europe/Stockholm", "Europe/Tallinn", "Europe/Tirane", "Europe/Tiraspol", "Europe/Ulyanovsk", "Europe/Uzhgorod", "Europe/Vaduz", "Europe/Vatican", "Europe/Vienna", "Europe/Vilnius", "Europe/Volgograd", "Europe/Warsaw", "Europe/Zagreb", "Europe/Zaporozhye", "Europe/Zurich", "Factory", "GB", "GB-Eire", "GMT", "GMT+0", "GMT-0", "GMT0", "Greenwich", "HST", "Hongkong", "Iceland", "Indian/Antananarivo", "Indian/Chagos", "Indian/Christmas", "Indian/Cocos", "Indian/Comoro", "Indian/Kerguelen", "Indian/Mahe", "Indian/Maldives", "Indian/Mauritius", "Indian/Mayotte", "Indian/Reunion", "Iran", "Israel", "Jamaica", "Japan", "Kwajalein", "Libya", "MET", "MST", "MST7MDT", "Mexico/BajaNorte", "Mexico/BajaSur", "Mexico/General", "NZ", "NZ-CHAT", "Navajo", "PRC", "PST8PDT", "Pacific/Apia", "Pacific/Auckland", "Pacific/Bougainville", "Pacific/Chatham", "Pacific/Chuuk", "Pacific/Easter", "Pacific/Efate", "Pacific/Enderbury", "Pacific/Fakaofo", "Pacific/Fiji", "Pacific/Funafuti", "Pacific/Galapagos", "Pacific/Gambier", "Pacific/Guadalcanal", "Pacific/Guam", "Pacific/Honolulu", "Pacific/Johnston", "Pacific/Kanton", "Pacific/Kiritimati", "Pacific/Kosrae", "Pacific/Kwajalein", "Pacific/Majuro", "Pacific/Marquesas", "Pacific/Midway", "Pacific/Nauru", "Pacific/Niue", "Pacific/Norfolk", "Pacific/Noumea", "Pacific/Pago_Pago", "Pacific/Palau", "Pacific/Pitcairn", "Pacific/Pohnpei", "Pacific/Ponape", "Pacific/Port_Moresby", "Pacific/Rarotonga", "Pacific/Saipan", "Pacific/Samoa", "Pacific/Tahiti", "Pacific/Tarawa", "Pacific/Tongatapu", "Pacific/Truk", "Pacific/Wake", "Pacific/Wallis", "Pacific/Yap", "Poland", "Portugal", "ROC", "ROK", "Singapore", "Turkey", "UCT", "US/Alaska", "US/Aleutian", "US/Arizona", "US/Central", "US/East-Indiana", "US/Eastern", "US/Hawaii", "US/Indiana-Starke", "US/Michigan", "US/Mountain", "US/Pacific", "US/Samoa", "UTC", "Universal", "W-SU", "WET", "Zulu"
272
273
  ]
273
274
  # fmt: on
275
+ TIME_ZONES: list[TimeZone] = list(get_args(TimeZone.__value__))
276
+
277
+
274
278
  type TimeZoneLike = (
275
279
  ZoneInfo | ZonedDateTime | Literal["local"] | TimeZone | dt.tzinfo | dt.datetime
276
280
  )
277
281
 
278
282
 
279
283
  __all__ = [
284
+ "TIME_ZONES",
280
285
  "Coro",
281
286
  "Dataclass",
282
287
  "DateDeltaLike",
@@ -39,7 +39,7 @@ from utilities.math import sign
39
39
  from utilities.platform import get_strftime
40
40
  from utilities.sentinel import Sentinel, sentinel
41
41
  from utilities.tzlocal import LOCAL_TIME_ZONE, LOCAL_TIME_ZONE_NAME
42
- from utilities.zoneinfo import UTC, ensure_time_zone, get_time_zone_name
42
+ from utilities.zoneinfo import UTC, to_time_zone_name
43
43
 
44
44
  if TYPE_CHECKING:
45
45
  from utilities.types import (
@@ -188,7 +188,7 @@ class DatePeriod:
188
188
  ...
189
189
  case never:
190
190
  assert_never(never)
191
- tz = ensure_time_zone(time_zone).key
191
+ tz = to_time_zone_name(time_zone)
192
192
  return ZonedDateTimePeriod(
193
193
  self.start.at(start).assume_tz(tz), self.end.at(end).assume_tz(tz)
194
194
  )
@@ -335,17 +335,17 @@ def format_compact(
335
335
 
336
336
  def from_timestamp(i: float, /, *, time_zone: TimeZoneLike = UTC) -> ZonedDateTime:
337
337
  """Get a zoned datetime from a timestamp."""
338
- return ZonedDateTime.from_timestamp(i, tz=get_time_zone_name(time_zone))
338
+ return ZonedDateTime.from_timestamp(i, tz=to_time_zone_name(time_zone))
339
339
 
340
340
 
341
341
  def from_timestamp_millis(i: int, /, *, time_zone: TimeZoneLike = UTC) -> ZonedDateTime:
342
342
  """Get a zoned datetime from a timestamp (in milliseconds)."""
343
- return ZonedDateTime.from_timestamp_millis(i, tz=get_time_zone_name(time_zone))
343
+ return ZonedDateTime.from_timestamp_millis(i, tz=to_time_zone_name(time_zone))
344
344
 
345
345
 
346
346
  def from_timestamp_nanos(i: int, /, *, time_zone: TimeZoneLike = UTC) -> ZonedDateTime:
347
347
  """Get a zoned datetime from a timestamp (in nanoseconds)."""
348
- return ZonedDateTime.from_timestamp_nanos(i, tz=get_time_zone_name(time_zone))
348
+ return ZonedDateTime.from_timestamp_nanos(i, tz=to_time_zone_name(time_zone))
349
349
 
350
350
 
351
351
  ##
@@ -353,7 +353,7 @@ def from_timestamp_nanos(i: int, /, *, time_zone: TimeZoneLike = UTC) -> ZonedDa
353
353
 
354
354
  def get_now(time_zone: TimeZoneLike = UTC, /) -> ZonedDateTime:
355
355
  """Get the current zoned datetime."""
356
- return ZonedDateTime.now(get_time_zone_name(time_zone))
356
+ return ZonedDateTime.now(to_time_zone_name(time_zone))
357
357
 
358
358
 
359
359
  NOW_UTC = get_now(UTC)
@@ -1703,24 +1703,7 @@ class WheneverLogRecord(LogRecord):
1703
1703
  )
1704
1704
  length = self._get_length()
1705
1705
  plain = format(get_now_local().to_plain().format_common_iso(), f"{length}s")
1706
- time_zone = self._get_time_zone_key()
1707
- self.zoned_datetime = f"{plain}[{time_zone}]"
1708
-
1709
- @classmethod
1710
- @cache
1711
- def _get_time_zone(cls) -> ZoneInfo:
1712
- """Get the local timezone."""
1713
- try:
1714
- from utilities.tzlocal import get_local_time_zone
1715
- except ModuleNotFoundError: # pragma: no cover
1716
- return UTC
1717
- return get_local_time_zone()
1718
-
1719
- @classmethod
1720
- @cache
1721
- def _get_time_zone_key(cls) -> str:
1722
- """Get the local timezone as a string."""
1723
- return cls._get_time_zone().key
1706
+ self.zoned_datetime = f"{plain}[{LOCAL_TIME_ZONE_NAME}]"
1724
1707
 
1725
1708
  @classmethod
1726
1709
  @cache
@@ -1878,7 +1861,7 @@ class ZonedDateTimePeriod:
1878
1861
 
1879
1862
  def to_tz(self, time_zone: TimeZoneLike, /) -> Self:
1880
1863
  """Convert the time zone."""
1881
- tz = get_time_zone_name(time_zone)
1864
+ tz = to_time_zone_name(time_zone)
1882
1865
  return self.replace(start=self.start.to_tz(tz), end=self.end.to_tz(tz))
1883
1866
 
1884
1867