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.
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/PKG-INFO +1 -1
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/pyproject.toml +2 -2
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/conftest.py +3 -1
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_polars.py +12 -12
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_types.py +2 -3
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_whenever.py +4 -9
- dycw_utilities-0.161.0/src/tests/test_zoneinfo.py +138 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/__init__.py +1 -1
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/hypothesis.py +2 -2
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/polars.py +6 -6
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/types.py +5 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/whenever.py +8 -25
- dycw_utilities-0.161.0/src/utilities/zoneinfo.py +133 -0
- dycw_utilities-0.160.2/src/tests/test_zoneinfo.py +0 -103
- dycw_utilities-0.160.2/src/utilities/zoneinfo.py +0 -75
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/.gitignore +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/LICENSE +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/README.md +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_missing/module.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_1.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/outer_2.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_1.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_2.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_with/subpackage/inner_3.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_1.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/package_without/module_2.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/standalone.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/modules/with_imports.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__obj.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestMultipleRegressionFixtures__test_main__series.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_int.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__false.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_literal__true.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestOrjsonRegressionFixture__test_dataclass_nested.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_dataframe.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/regressions/test_pytest_regressions/TestPolarsRegressionFixture__test_series.json +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_altair.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_asyncio.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_atomicwrites.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_atools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cachetools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_click.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_concurrent.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_contextlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_contextvars.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cryptography.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_cvxpy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_dataclasses.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_enum.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_errors.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_eventkit.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_fastapi.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_fpdf2.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_functions.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_functools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_getpass.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_gzip.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_hashlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_http.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_hypothesis.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_importlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_inflect.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_ipython.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_iterables.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_json.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_jupyter.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_libcst.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_lightweight_charts.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_logging.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_math.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_memory_profiler.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_modules.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_more_itertools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_numpy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_objects/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_objects/objects.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_operator.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_optuna.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_orjson.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_os.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_parse.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pathlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pickle.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_platform.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_polars_ols.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_postgres.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pottery.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pqdm.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_psutil.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pyinstrument.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest_randomly.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_pytest_regressions.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_random.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_re.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_redis.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_reprlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_scipy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sentinel.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_shelve.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_slack_sdk.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_socket.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_sqlalchemy_polars.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_statsmodels.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_string.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tempfile.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_text.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_threading.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_timer.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_traceback.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typed_settings.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/no_future.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_typing_funcs/with_future.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tzdata.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_tzlocal.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_uuid.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_version.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_warnings.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/tests/test_zipfile.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/altair.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/asyncio.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/atomicwrites.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/atools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cachetools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/click.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/concurrent.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/contextlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/contextvars.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cryptography.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/cvxpy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/dataclasses.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/enum.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/errors.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/eventkit.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/fastapi.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/fpdf2.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/functions.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/functools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/getpass.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/gzip.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/hashlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/http.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/importlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/inflect.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/ipython.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/iterables.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/json.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/jupyter.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/libcst.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/lightweight_charts.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/logging.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/math.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/memory_profiler.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/modules.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/more_itertools.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/numpy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/operator.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/optuna.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/orjson.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/os.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/parse.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pathlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pickle.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/platform.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/polars_ols.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/postgres.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pottery.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pqdm.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/psutil.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/py.typed +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pyinstrument.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/__init__.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_randomly.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_plugins/pytest_regressions.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/pytest_regressions.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/random.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/re.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/redis.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/reprlib.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/scipy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sentinel.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/shelve.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/slack_sdk.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/socket.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/sqlalchemy_polars.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/statsmodels.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/string.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tempfile.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/text.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/threading.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/timer.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/traceback.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/typed_settings.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/typing.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tzdata.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/tzlocal.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/uuid.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/version.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/warnings.py +0 -0
- {dycw_utilities-0.160.2 → dycw_utilities-0.161.0}/src/utilities/zipfile.py +0 -0
@@ -227,7 +227,7 @@ dependencies = [
|
|
227
227
|
name = "dycw-utilities"
|
228
228
|
readme = "README.md"
|
229
229
|
requires-python = ">= 3.12"
|
230
|
-
version = "0.
|
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.
|
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,
|
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
|
-
@
|
1430
|
-
|
1431
|
-
|
1432
|
-
(
|
1433
|
-
(
|
1434
|
-
(
|
1435
|
-
(
|
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, *,
|
1439
|
-
|
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
|
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(
|
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
|
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
|
-
|
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 =
|
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)
|
@@ -101,7 +101,7 @@ from utilities.whenever import (
|
|
101
101
|
to_days,
|
102
102
|
to_nanoseconds,
|
103
103
|
)
|
104
|
-
from utilities.zoneinfo import UTC,
|
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_ =
|
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,
|
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(
|
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
|
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 =
|
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
|
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=
|
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,
|
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 =
|
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=
|
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=
|
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=
|
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(
|
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
|
-
|
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 =
|
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
|
+
]
|