dycw-utilities 0.160.2__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.2 → dycw_utilities-0.161.0}/PKG-INFO +1 -1
  2. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/pyproject.toml +2 -2
  3. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/conftest.py +3 -1
  4. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_polars.py +12 -12
  5. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_types.py +2 -3
  6. {dycw_utilities-0.160.2 → 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.2 → dycw_utilities-0.161.0}/src/utilities/__init__.py +1 -1
  9. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/hypothesis.py +2 -2
  10. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/polars.py +6 -6
  11. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/types.py +5 -0
  12. {dycw_utilities-0.160.2 → 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.2/src/tests/test_zoneinfo.py +0 -103
  15. dycw_utilities-0.160.2/src/utilities/zoneinfo.py +0 -75
  16. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/.gitignore +0 -0
  17. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/LICENSE +0 -0
  18. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/README.md +0 -0
  19. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/__init__.py +0 -0
  20. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/__init__.py +0 -0
  21. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/__init__.py +0 -0
  22. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/module.py +0 -0
  23. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/__init__.py +0 -0
  24. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_1.py +0 -0
  25. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_2.py +0 -0
  26. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/__init__.py +0 -0
  27. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_1.py +0 -0
  28. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_2.py +0 -0
  29. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_3.py +0 -0
  30. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/__init__.py +0 -0
  31. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_1.py +0 -0
  32. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_2.py +0 -0
  33. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/standalone.py +0 -0
  34. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/with_imports.py +0 -0
  35. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__obj.json +0 -0
  36. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__series.json +0 -0
  37. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_int.json +0 -0
  38. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__false.json +0 -0
  39. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__true.json +0 -0
  40. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_nested.json +0 -0
  41. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_dataframe.json +0 -0
  42. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_series.json +0 -0
  43. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_altair.py +0 -0
  44. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_asyncio.py +0 -0
  45. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_atomicwrites.py +0 -0
  46. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_atools.py +0 -0
  47. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cachetools.py +0 -0
  48. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_click.py +0 -0
  49. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_concurrent.py +0 -0
  50. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_contextlib.py +0 -0
  51. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_contextvars.py +0 -0
  52. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cryptography.py +0 -0
  53. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cvxpy.py +0 -0
  54. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_dataclasses.py +0 -0
  55. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_enum.py +0 -0
  56. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_errors.py +0 -0
  57. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_eventkit.py +0 -0
  58. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_fastapi.py +0 -0
  59. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_fpdf2.py +0 -0
  60. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_functions.py +0 -0
  61. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_functools.py +0 -0
  62. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_getpass.py +0 -0
  63. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_gzip.py +0 -0
  64. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_hashlib.py +0 -0
  65. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_http.py +0 -0
  66. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_hypothesis.py +0 -0
  67. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_importlib.py +0 -0
  68. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_inflect.py +0 -0
  69. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_ipython.py +0 -0
  70. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_iterables.py +0 -0
  71. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_json.py +0 -0
  72. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_jupyter.py +0 -0
  73. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_libcst.py +0 -0
  74. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_lightweight_charts.py +0 -0
  75. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_logging.py +0 -0
  76. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_math.py +0 -0
  77. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_memory_profiler.py +0 -0
  78. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_modules.py +0 -0
  79. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_more_itertools.py +0 -0
  80. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_numpy.py +0 -0
  81. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_objects/__init__.py +0 -0
  82. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_objects/objects.py +0 -0
  83. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_operator.py +0 -0
  84. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_optuna.py +0 -0
  85. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_orjson.py +0 -0
  86. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_os.py +0 -0
  87. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_parse.py +0 -0
  88. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pathlib.py +0 -0
  89. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pickle.py +0 -0
  90. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_platform.py +0 -0
  91. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_polars_ols.py +0 -0
  92. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_postgres.py +0 -0
  93. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pottery.py +0 -0
  94. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pqdm.py +0 -0
  95. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_psutil.py +0 -0
  96. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pyinstrument.py +0 -0
  97. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest.py +0 -0
  98. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest_randomly.py +0 -0
  99. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest_regressions.py +0 -0
  100. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_random.py +0 -0
  101. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_re.py +0 -0
  102. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_redis.py +0 -0
  103. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_reprlib.py +0 -0
  104. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_scipy.py +0 -0
  105. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sentinel.py +0 -0
  106. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_shelve.py +0 -0
  107. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_slack_sdk.py +0 -0
  108. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_socket.py +0 -0
  109. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy.py +0 -0
  110. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy_polars.py +0 -0
  111. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_statsmodels.py +0 -0
  112. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_string.py +0 -0
  113. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tempfile.py +0 -0
  114. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_text.py +0 -0
  115. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_threading.py +0 -0
  116. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_timer.py +0 -0
  117. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_traceback.py +0 -0
  118. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typed_settings.py +0 -0
  119. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing.py +0 -0
  120. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/__init__.py +0 -0
  121. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/no_future.py +0 -0
  122. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/with_future.py +0 -0
  123. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tzdata.py +0 -0
  124. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tzlocal.py +0 -0
  125. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_uuid.py +0 -0
  126. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_version.py +0 -0
  127. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_warnings.py +0 -0
  128. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_zipfile.py +0 -0
  129. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/altair.py +0 -0
  130. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/asyncio.py +0 -0
  131. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/atomicwrites.py +0 -0
  132. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/atools.py +0 -0
  133. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cachetools.py +0 -0
  134. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/click.py +0 -0
  135. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/concurrent.py +0 -0
  136. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/contextlib.py +0 -0
  137. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/contextvars.py +0 -0
  138. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cryptography.py +0 -0
  139. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cvxpy.py +0 -0
  140. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/dataclasses.py +0 -0
  141. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/enum.py +0 -0
  142. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/errors.py +0 -0
  143. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/eventkit.py +0 -0
  144. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/fastapi.py +0 -0
  145. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/fpdf2.py +0 -0
  146. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/functions.py +0 -0
  147. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/functools.py +0 -0
  148. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/getpass.py +0 -0
  149. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/gzip.py +0 -0
  150. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/hashlib.py +0 -0
  151. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/http.py +0 -0
  152. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/importlib.py +0 -0
  153. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/inflect.py +0 -0
  154. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/ipython.py +0 -0
  155. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/iterables.py +0 -0
  156. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/json.py +0 -0
  157. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/jupyter.py +0 -0
  158. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/libcst.py +0 -0
  159. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/lightweight_charts.py +0 -0
  160. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/logging.py +0 -0
  161. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/math.py +0 -0
  162. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/memory_profiler.py +0 -0
  163. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/modules.py +0 -0
  164. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/more_itertools.py +0 -0
  165. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/numpy.py +0 -0
  166. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/operator.py +0 -0
  167. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/optuna.py +0 -0
  168. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/orjson.py +0 -0
  169. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/os.py +0 -0
  170. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/parse.py +0 -0
  171. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pathlib.py +0 -0
  172. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pickle.py +0 -0
  173. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/platform.py +0 -0
  174. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/polars_ols.py +0 -0
  175. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/postgres.py +0 -0
  176. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pottery.py +0 -0
  177. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pqdm.py +0 -0
  178. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/psutil.py +0 -0
  179. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/py.typed +0 -0
  180. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pyinstrument.py +0 -0
  181. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest.py +0 -0
  182. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/__init__.py +0 -0
  183. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_randomly.py +0 -0
  184. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_regressions.py +0 -0
  185. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_regressions.py +0 -0
  186. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/random.py +0 -0
  187. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/re.py +0 -0
  188. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/redis.py +0 -0
  189. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/reprlib.py +0 -0
  190. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/scipy.py +0 -0
  191. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sentinel.py +0 -0
  192. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/shelve.py +0 -0
  193. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/slack_sdk.py +0 -0
  194. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/socket.py +0 -0
  195. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy.py +0 -0
  196. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy_polars.py +0 -0
  197. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/statsmodels.py +0 -0
  198. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/string.py +0 -0
  199. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tempfile.py +0 -0
  200. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/text.py +0 -0
  201. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/threading.py +0 -0
  202. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/timer.py +0 -0
  203. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/traceback.py +0 -0
  204. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/typed_settings.py +0 -0
  205. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/typing.py +0 -0
  206. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tzdata.py +0 -0
  207. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tzlocal.py +0 -0
  208. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/uuid.py +0 -0
  209. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/version.py +0 -0
  210. {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/warnings.py +0 -0
  211. {dycw_utilities-0.160.2 → 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.2
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.2"
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.2"
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
@@ -222,7 +222,7 @@ from utilities.whenever import (
222
222
  get_now_plain,
223
223
  get_today,
224
224
  )
225
- from utilities.zoneinfo import UTC, get_time_zone_name
225
+ from utilities.zoneinfo import UTC, to_time_zone_name
226
226
 
227
227
  if TYPE_CHECKING:
228
228
  from collections.abc import Callable, Iterable, Mapping, Sequence
@@ -1426,18 +1426,18 @@ class TestDataClassToSchema:
1426
1426
 
1427
1427
 
1428
1428
  class TestDatetimeDTypes:
1429
- @given(
1430
- case=sampled_from([
1431
- (HongKong, DatetimeHongKong),
1432
- (Tokyo, DatetimeTokyo),
1433
- (USCentral, DatetimeUSCentral),
1434
- (USEastern, DatetimeUSEastern),
1435
- (UTC, DatetimeUTC),
1436
- ])
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
+ ],
1437
1438
  )
1438
- def test_main(self, *, case: tuple[ZoneInfo, Datetime]) -> None:
1439
- time_zone, dtype = case
1440
- 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)
1441
1441
  expected = dtype.time_zone
1442
1442
  assert name == expected
1443
1443
 
@@ -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.2"
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
 
@@ -2110,7 +2110,7 @@ def period_range(
2110
2110
  eager: bool = False,
2111
2111
  ) -> Series | Expr:
2112
2112
  """Construct a period range."""
2113
- 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)
2114
2114
  match end_or_length:
2115
2115
  case ZonedDateTime() as end:
2116
2116
  ...
@@ -2127,7 +2127,7 @@ def period_range(
2127
2127
  time_zone=time_zone_use,
2128
2128
  eager=eager,
2129
2129
  ).alias("start")
2130
- ends = (starts.dt.offset_by(interval)).alias("end")
2130
+ ends = starts.dt.offset_by(interval).alias("end")
2131
2131
  period = struct(starts, ends)
2132
2132
  return try_reify_expr(period, starts, ends)
2133
2133
 
@@ -2220,7 +2220,7 @@ def _replace_time_zone_one(
2220
2220
  sr: Series, /, *, time_zone: TimeZoneLike | None = UTC
2221
2221
  ) -> Series:
2222
2222
  if isinstance(sr.dtype, Datetime):
2223
- 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)
2224
2224
  return sr.dt.replace_time_zone(time_zone_use)
2225
2225
  return sr
2226
2226
 
@@ -2611,7 +2611,7 @@ def zoned_date_time_dtype(
2611
2611
  *, time_unit: TimeUnit = "us", time_zone: TimeZoneLike = UTC
2612
2612
  ) -> Datetime:
2613
2613
  """Create a zoned date-time data type."""
2614
- 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))
2615
2615
 
2616
2616
 
2617
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
 
@@ -0,0 +1,133 @@
1
+ from __future__ import annotations
2
+
3
+ import datetime as dt
4
+ from dataclasses import dataclass
5
+ from typing import TYPE_CHECKING, assert_never, cast, override
6
+ from zoneinfo import ZoneInfo
7
+
8
+ from whenever import ZonedDateTime
9
+
10
+ from utilities.types import TIME_ZONES
11
+ from utilities.tzlocal import LOCAL_TIME_ZONE, LOCAL_TIME_ZONE_NAME
12
+
13
+ if TYPE_CHECKING:
14
+ from utilities.types import TimeZone, TimeZoneLike
15
+
16
+
17
+ UTC = ZoneInfo("UTC")
18
+
19
+
20
+ ##
21
+
22
+
23
+ def to_zone_info(obj: TimeZoneLike, /) -> ZoneInfo:
24
+ """Convert to a time-zone."""
25
+ match obj:
26
+ case ZoneInfo() as zone_info:
27
+ return zone_info
28
+ case ZonedDateTime() as date_time:
29
+ return ZoneInfo(date_time.tz)
30
+ case "local":
31
+ return LOCAL_TIME_ZONE
32
+ case str() as key:
33
+ return ZoneInfo(key)
34
+ case dt.tzinfo() as tzinfo:
35
+ if tzinfo is dt.UTC:
36
+ return UTC
37
+ raise _ToZoneInfoInvalidTZInfoError(time_zone=obj)
38
+ case dt.datetime() as date_time:
39
+ if date_time.tzinfo is None:
40
+ raise _ToZoneInfoPlainDateTimeError(date_time=date_time)
41
+ return to_zone_info(date_time.tzinfo)
42
+ case never:
43
+ assert_never(never)
44
+
45
+
46
+ @dataclass(kw_only=True, slots=True)
47
+ class ToTimeZoneError(Exception): ...
48
+
49
+
50
+ @dataclass(kw_only=True, slots=True)
51
+ class _ToZoneInfoInvalidTZInfoError(ToTimeZoneError):
52
+ time_zone: dt.tzinfo
53
+
54
+ @override
55
+ def __str__(self) -> str:
56
+ return f"Invalid time-zone: {self.time_zone}"
57
+
58
+
59
+ @dataclass(kw_only=True, slots=True)
60
+ class _ToZoneInfoPlainDateTimeError(ToTimeZoneError):
61
+ date_time: dt.datetime
62
+
63
+ @override
64
+ def __str__(self) -> str:
65
+ return f"Plain date-time: {self.date_time}"
66
+
67
+
68
+ ##
69
+
70
+
71
+ def to_time_zone_name(obj: TimeZoneLike, /) -> TimeZone:
72
+ """Convert to a time zone name."""
73
+ match obj:
74
+ case ZoneInfo() as zone_info:
75
+ return cast("TimeZone", zone_info.key)
76
+ case ZonedDateTime() as date_time:
77
+ return cast("TimeZone", date_time.tz)
78
+ case "local":
79
+ return LOCAL_TIME_ZONE_NAME
80
+ case str() as time_zone:
81
+ if time_zone in TIME_ZONES:
82
+ return time_zone
83
+ raise _ToTimeZoneNameInvalidKeyError(time_zone=time_zone)
84
+ case dt.tzinfo() as tzinfo:
85
+ if tzinfo is dt.UTC:
86
+ return cast("TimeZone", UTC.key)
87
+ raise _ToTimeZoneNameInvalidTZInfoError(time_zone=obj)
88
+ case dt.datetime() as date_time:
89
+ if date_time.tzinfo is None:
90
+ raise _ToTimeZoneNamePlainDateTimeError(date_time=date_time)
91
+ return to_time_zone_name(date_time.tzinfo)
92
+ case never:
93
+ assert_never(never)
94
+
95
+
96
+ @dataclass(kw_only=True, slots=True)
97
+ class ToTimeZoneNameError(Exception): ...
98
+
99
+
100
+ @dataclass(kw_only=True, slots=True)
101
+ class _ToTimeZoneNameInvalidKeyError(ToTimeZoneNameError):
102
+ time_zone: str
103
+
104
+ @override
105
+ def __str__(self) -> str:
106
+ return f"Invalid time-zone: {self.time_zone!r}"
107
+
108
+
109
+ @dataclass(kw_only=True, slots=True)
110
+ class _ToTimeZoneNameInvalidTZInfoError(ToTimeZoneNameError):
111
+ time_zone: dt.tzinfo
112
+
113
+ @override
114
+ def __str__(self) -> str:
115
+ return f"Invalid time-zone: {self.time_zone}"
116
+
117
+
118
+ @dataclass(kw_only=True, slots=True)
119
+ class _ToTimeZoneNamePlainDateTimeError(ToTimeZoneNameError):
120
+ date_time: dt.datetime
121
+
122
+ @override
123
+ def __str__(self) -> str:
124
+ return f"Plain date-time: {self.date_time}"
125
+
126
+
127
+ __all__ = [
128
+ "UTC",
129
+ "ToTimeZoneError",
130
+ "ToTimeZoneNameError",
131
+ "to_time_zone_name",
132
+ "to_zone_info",
133
+ ]