From ea4b66d8ef217c6ffe996f5fcd532ccc1f5102b1 Mon Sep 17 00:00:00 2001 From: "Juan M. Ley" Date: Sat, 7 Feb 2026 02:22:32 -0600 Subject: [PATCH] changed .env from windows to UNIX --- .../site/python3.12/greenlet/greenlet.h | 164 - .../typing_extensions.cpython-312.pyc | Bin 163639 -> 0 bytes .../annotated_doc-0.0.4.dist-info/INSTALLER | 1 - .../annotated_doc-0.0.4.dist-info/METADATA | 145 - .../annotated_doc-0.0.4.dist-info/RECORD | 11 - .../annotated_doc-0.0.4.dist-info/WHEEL | 4 - .../entry_points.txt | 4 - .../licenses/LICENSE | 21 - .../site-packages/annotated_doc/__init__.py | 3 - .../__pycache__/__init__.cpython-312.pyc | Bin 250 -> 0 bytes .../__pycache__/main.cpython-312.pyc | Bin 1898 -> 0 bytes .env/Lib/site-packages/annotated_doc/main.py | 36 - .env/Lib/site-packages/annotated_doc/py.typed | 0 .../annotated_types-0.7.0.dist-info/INSTALLER | 1 - .../annotated_types-0.7.0.dist-info/METADATA | 295 - .../annotated_types-0.7.0.dist-info/RECORD | 10 - .../annotated_types-0.7.0.dist-info/WHEEL | 4 - .../licenses/LICENSE | 21 - .../site-packages/annotated_types/__init__.py | 432 - .../__pycache__/__init__.cpython-312.pyc | Bin 18619 -> 0 bytes .../__pycache__/test_cases.cpython-312.pyc | Bin 13232 -> 0 bytes .../site-packages/annotated_types/py.typed | 0 .../annotated_types/test_cases.py | 151 - .../anyio-4.12.1.dist-info/INSTALLER | 1 - .../anyio-4.12.1.dist-info/METADATA | 96 - .../anyio-4.12.1.dist-info/RECORD | 92 - .../anyio-4.12.1.dist-info/WHEEL | 5 - .../anyio-4.12.1.dist-info/entry_points.txt | 2 - .../anyio-4.12.1.dist-info/licenses/LICENSE | 20 - .../anyio-4.12.1.dist-info/top_level.txt | 1 - .env/Lib/site-packages/anyio/__init__.py | 111 - .../__pycache__/__init__.cpython-312.pyc | Bin 4622 -> 0 bytes .../__pycache__/from_thread.cpython-312.pyc | Bin 25828 -> 0 bytes .../__pycache__/functools.cpython-312.pyc | Bin 15646 -> 0 bytes .../__pycache__/lowlevel.cpython-312.pyc | Bin 7993 -> 0 bytes .../__pycache__/pytest_plugin.cpython-312.pyc | Bin 14284 -> 0 bytes .../to_interpreter.cpython-312.pyc | Bin 10325 -> 0 bytes .../__pycache__/to_process.cpython-312.pyc | Bin 11955 -> 0 bytes .../__pycache__/to_thread.cpython-312.pyc | Bin 3196 -> 0 bytes .../site-packages/anyio/_backends/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 188 -> 0 bytes .../__pycache__/_asyncio.cpython-312.pyc | Bin 138339 -> 0 bytes .../__pycache__/_trio.cpython-312.pyc | Bin 71575 -> 0 bytes .../site-packages/anyio/_backends/_asyncio.py | 2980 ------ .../site-packages/anyio/_backends/_trio.py | 1346 --- .../Lib/site-packages/anyio/_core/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 184 -> 0 bytes .../_asyncio_selector_thread.cpython-312.pyc | Bin 8404 -> 0 bytes .../_contextmanagers.cpython-312.pyc | Bin 8994 -> 0 bytes .../__pycache__/_eventloop.cpython-312.pyc | Bin 8178 -> 0 bytes .../__pycache__/_exceptions.cpython-312.pyc | Bin 7421 -> 0 bytes .../_core/__pycache__/_fileio.cpython-312.pyc | Bin 43371 -> 0 bytes .../__pycache__/_resources.cpython-312.pyc | Bin 926 -> 0 bytes .../__pycache__/_signals.cpython-312.pyc | Bin 1380 -> 0 bytes .../__pycache__/_sockets.cpython-312.pyc | Bin 40591 -> 0 bytes .../__pycache__/_streams.cpython-312.pyc | Bin 2333 -> 0 bytes .../__pycache__/_subprocesses.cpython-312.pyc | Bin 9624 -> 0 bytes .../_synchronization.cpython-312.pyc | Bin 32934 -> 0 bytes .../_core/__pycache__/_tasks.cpython-312.pyc | Bin 7683 -> 0 bytes .../__pycache__/_tempfile.cpython-312.pyc | Bin 28128 -> 0 bytes .../__pycache__/_testing.cpython-312.pyc | Bin 3786 -> 0 bytes .../__pycache__/_typedattr.cpython-312.pyc | Bin 3829 -> 0 bytes .../anyio/_core/_asyncio_selector_thread.py | 167 - .../anyio/_core/_contextmanagers.py | 200 - .../site-packages/anyio/_core/_eventloop.py | 234 - .../site-packages/anyio/_core/_exceptions.py | 156 - .env/Lib/site-packages/anyio/_core/_fileio.py | 797 -- .../site-packages/anyio/_core/_resources.py | 18 - .../Lib/site-packages/anyio/_core/_signals.py | 29 - .../Lib/site-packages/anyio/_core/_sockets.py | 1003 -- .../Lib/site-packages/anyio/_core/_streams.py | 52 - .../anyio/_core/_subprocesses.py | 202 - .../anyio/_core/_synchronization.py | 753 -- .env/Lib/site-packages/anyio/_core/_tasks.py | 173 - .../site-packages/anyio/_core/_tempfile.py | 616 -- .../Lib/site-packages/anyio/_core/_testing.py | 82 - .../site-packages/anyio/_core/_typedattr.py | 81 - .env/Lib/site-packages/anyio/abc/__init__.py | 58 - .../abc/__pycache__/__init__.cpython-312.pyc | Bin 2359 -> 0 bytes .../__pycache__/_eventloop.cpython-312.pyc | Bin 16631 -> 0 bytes .../__pycache__/_resources.cpython-312.pyc | Bin 1631 -> 0 bytes .../abc/__pycache__/_sockets.cpython-312.pyc | Bin 18424 -> 0 bytes .../abc/__pycache__/_streams.cpython-312.pyc | Bin 9885 -> 0 bytes .../__pycache__/_subprocesses.cpython-312.pyc | Bin 3239 -> 0 bytes .../abc/__pycache__/_tasks.cpython-312.pyc | Bin 5138 -> 0 bytes .../abc/__pycache__/_testing.cpython-312.pyc | Bin 2834 -> 0 bytes .../Lib/site-packages/anyio/abc/_eventloop.py | 414 - .../Lib/site-packages/anyio/abc/_resources.py | 33 - .env/Lib/site-packages/anyio/abc/_sockets.py | 405 - .env/Lib/site-packages/anyio/abc/_streams.py | 239 - .../site-packages/anyio/abc/_subprocesses.py | 79 - .env/Lib/site-packages/anyio/abc/_tasks.py | 117 - .env/Lib/site-packages/anyio/abc/_testing.py | 65 - .env/Lib/site-packages/anyio/from_thread.py | 578 -- .env/Lib/site-packages/anyio/functools.py | 375 - .env/Lib/site-packages/anyio/lowlevel.py | 196 - .env/Lib/site-packages/anyio/py.typed | 0 .env/Lib/site-packages/anyio/pytest_plugin.py | 302 - .../site-packages/anyio/streams/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 186 -> 0 bytes .../__pycache__/buffered.cpython-312.pyc | Bin 9065 -> 0 bytes .../streams/__pycache__/file.cpython-312.pyc | Bin 7560 -> 0 bytes .../__pycache__/memory.cpython-312.pyc | Bin 15007 -> 0 bytes .../__pycache__/stapled.cpython-312.pyc | Bin 7536 -> 0 bytes .../streams/__pycache__/text.cpython-312.pyc | Bin 9332 -> 0 bytes .../streams/__pycache__/tls.cpython-312.pyc | Bin 20217 -> 0 bytes .../site-packages/anyio/streams/buffered.py | 188 - .env/Lib/site-packages/anyio/streams/file.py | 154 - .../Lib/site-packages/anyio/streams/memory.py | 325 - .../site-packages/anyio/streams/stapled.py | 147 - .env/Lib/site-packages/anyio/streams/text.py | 176 - .env/Lib/site-packages/anyio/streams/tls.py | 424 - .../Lib/site-packages/anyio/to_interpreter.py | 246 - .env/Lib/site-packages/anyio/to_process.py | 266 - .env/Lib/site-packages/anyio/to_thread.py | 78 - .../click-8.3.1.dist-info/INSTALLER | 1 - .../click-8.3.1.dist-info/METADATA | 84 - .../click-8.3.1.dist-info/RECORD | 40 - .../site-packages/click-8.3.1.dist-info/WHEEL | 4 - .../licenses/LICENSE.txt | 28 - .env/Lib/site-packages/click/__init__.py | 123 - .../__pycache__/__init__.cpython-312.pyc | Bin 4064 -> 0 bytes .../click/__pycache__/_compat.cpython-312.pyc | Bin 24183 -> 0 bytes .../__pycache__/_termui_impl.cpython-312.pyc | Bin 31557 -> 0 bytes .../__pycache__/_textwrap.cpython-312.pyc | Bin 2412 -> 0 bytes .../click/__pycache__/_utils.cpython-312.pyc | Bin 1192 -> 0 bytes .../__pycache__/_winconsole.cpython-312.pyc | Bin 11757 -> 0 bytes .../click/__pycache__/core.cpython-312.pyc | Bin 134412 -> 0 bytes .../__pycache__/decorators.cpython-312.pyc | Bin 22129 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 14769 -> 0 bytes .../__pycache__/formatting.cpython-312.pyc | Bin 13590 -> 0 bytes .../click/__pycache__/globals.cpython-312.pyc | Bin 2957 -> 0 bytes .../click/__pycache__/parser.cpython-312.pyc | Bin 20416 -> 0 bytes .../shell_completion.cpython-312.pyc | Bin 23295 -> 0 bytes .../click/__pycache__/termui.cpython-312.pyc | Bin 34644 -> 0 bytes .../click/__pycache__/testing.cpython-312.pyc | Bin 27386 -> 0 bytes .../click/__pycache__/types.cpython-312.pyc | Bin 50056 -> 0 bytes .../click/__pycache__/utils.cpython-312.pyc | Bin 24861 -> 0 bytes .env/Lib/site-packages/click/_compat.py | 622 -- .env/Lib/site-packages/click/_termui_impl.py | 852 -- .env/Lib/site-packages/click/_textwrap.py | 51 - .env/Lib/site-packages/click/_utils.py | 36 - .env/Lib/site-packages/click/_winconsole.py | 296 - .env/Lib/site-packages/click/core.py | 3415 ------- .env/Lib/site-packages/click/decorators.py | 551 - .env/Lib/site-packages/click/exceptions.py | 308 - .env/Lib/site-packages/click/formatting.py | 301 - .env/Lib/site-packages/click/globals.py | 67 - .env/Lib/site-packages/click/parser.py | 532 - .env/Lib/site-packages/click/py.typed | 0 .../site-packages/click/shell_completion.py | 667 -- .env/Lib/site-packages/click/termui.py | 883 -- .env/Lib/site-packages/click/testing.py | 577 -- .env/Lib/site-packages/click/types.py | 1209 --- .env/Lib/site-packages/click/utils.py | 627 -- .../colorama-0.4.6.dist-info/INSTALLER | 1 - .../colorama-0.4.6.dist-info/METADATA | 441 - .../colorama-0.4.6.dist-info/RECORD | 31 - .../colorama-0.4.6.dist-info/WHEEL | 5 - .../licenses/LICENSE.txt | 27 - .env/Lib/site-packages/colorama/__init__.py | 7 - .../__pycache__/__init__.cpython-312.pyc | Bin 481 -> 0 bytes .../colorama/__pycache__/ansi.cpython-312.pyc | Bin 3934 -> 0 bytes .../__pycache__/ansitowin32.cpython-312.pyc | Bin 16401 -> 0 bytes .../__pycache__/initialise.cpython-312.pyc | Bin 3539 -> 0 bytes .../__pycache__/win32.cpython-312.pyc | Bin 8124 -> 0 bytes .../__pycache__/winterm.cpython-312.pyc | Bin 9077 -> 0 bytes .env/Lib/site-packages/colorama/ansi.py | 102 - .../Lib/site-packages/colorama/ansitowin32.py | 277 - .env/Lib/site-packages/colorama/initialise.py | 121 - .../site-packages/colorama/tests/__init__.py | 1 - .../__pycache__/__init__.cpython-312.pyc | Bin 187 -> 0 bytes .../__pycache__/ansi_test.cpython-312.pyc | Bin 5456 -> 0 bytes .../ansitowin32_test.cpython-312.pyc | Bin 17995 -> 0 bytes .../initialise_test.cpython-312.pyc | Bin 11643 -> 0 bytes .../__pycache__/isatty_test.cpython-312.pyc | Bin 4770 -> 0 bytes .../tests/__pycache__/utils.cpython-312.pyc | Bin 2459 -> 0 bytes .../__pycache__/winterm_test.cpython-312.pyc | Bin 6601 -> 0 bytes .../site-packages/colorama/tests/ansi_test.py | 76 - .../colorama/tests/ansitowin32_test.py | 294 - .../colorama/tests/initialise_test.py | 189 - .../colorama/tests/isatty_test.py | 57 - .../Lib/site-packages/colorama/tests/utils.py | 49 - .../colorama/tests/winterm_test.py | 131 - .env/Lib/site-packages/colorama/win32.py | 180 - .env/Lib/site-packages/colorama/winterm.py | 195 - .../fastapi-0.128.3.dist-info/INSTALLER | 1 - .../fastapi-0.128.3.dist-info/METADATA | 640 -- .../fastapi-0.128.3.dist-info/RECORD | 103 - .../fastapi-0.128.3.dist-info/REQUESTED | 0 .../fastapi-0.128.3.dist-info/WHEEL | 4 - .../entry_points.txt | 5 - .../licenses/LICENSE | 21 - .env/Lib/site-packages/fastapi/__init__.py | 25 - .env/Lib/site-packages/fastapi/__main__.py | 3 - .../__pycache__/__init__.cpython-312.pyc | Bin 1095 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 242 -> 0 bytes .../__pycache__/applications.cpython-312.pyc | Bin 86867 -> 0 bytes .../__pycache__/background.cpython-312.pyc | Bin 2396 -> 0 bytes .../fastapi/__pycache__/cli.cpython-312.pyc | Bin 656 -> 0 bytes .../__pycache__/concurrency.cpython-312.pyc | Bin 1717 -> 0 bytes .../datastructures.cpython-312.pyc | Bin 7207 -> 0 bytes .../__pycache__/encoders.cpython-312.pyc | Bin 11055 -> 0 bytes .../exception_handlers.cpython-312.pyc | Bin 2022 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 10481 -> 0 bytes .../__pycache__/logger.cpython-312.pyc | Bin 282 -> 0 bytes .../param_functions.cpython-312.pyc | Bin 41548 -> 0 bytes .../__pycache__/params.cpython-312.pyc | Bin 24335 -> 0 bytes .../__pycache__/requests.cpython-312.pyc | Bin 271 -> 0 bytes .../__pycache__/responses.cpython-312.pyc | Bin 2399 -> 0 bytes .../__pycache__/routing.cpython-312.pyc | Bin 92417 -> 0 bytes .../__pycache__/staticfiles.cpython-312.pyc | Bin 243 -> 0 bytes .../__pycache__/templating.cpython-312.pyc | Bin 245 -> 0 bytes .../__pycache__/testclient.cpython-312.pyc | Bin 240 -> 0 bytes .../fastapi/__pycache__/types.cpython-312.pyc | Bin 783 -> 0 bytes .../fastapi/__pycache__/utils.cpython-312.pyc | Bin 6425 -> 0 bytes .../__pycache__/websockets.cpython-312.pyc | Bin 320 -> 0 bytes .../site-packages/fastapi/_compat/__init__.py | 41 - .../__pycache__/__init__.cpython-312.pyc | Bin 1760 -> 0 bytes .../__pycache__/shared.cpython-312.pyc | Bin 9387 -> 0 bytes .../_compat/__pycache__/v2.cpython-312.pyc | Bin 23005 -> 0 bytes .../site-packages/fastapi/_compat/shared.py | 206 - .env/Lib/site-packages/fastapi/_compat/v2.py | 568 -- .../Lib/site-packages/fastapi/applications.py | 4672 --------- .env/Lib/site-packages/fastapi/background.py | 60 - .env/Lib/site-packages/fastapi/cli.py | 13 - .env/Lib/site-packages/fastapi/concurrency.py | 41 - .../site-packages/fastapi/datastructures.py | 183 - .../fastapi/dependencies/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 193 -> 0 bytes .../__pycache__/models.cpython-312.pyc | Bin 9210 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 40861 -> 0 bytes .../fastapi/dependencies/models.py | 193 - .../fastapi/dependencies/utils.py | 1039 -- .env/Lib/site-packages/fastapi/encoders.py | 346 - .../fastapi/exception_handlers.py | 34 - .env/Lib/site-packages/fastapi/exceptions.py | 256 - .env/Lib/site-packages/fastapi/logger.py | 3 - .../fastapi/middleware/__init__.py | 1 - .../__pycache__/__init__.cpython-312.pyc | Bin 249 -> 0 bytes .../asyncexitstack.cpython-312.pyc | Bin 1484 -> 0 bytes .../__pycache__/cors.cpython-312.pyc | Bin 254 -> 0 bytes .../__pycache__/gzip.cpython-312.pyc | Bin 254 -> 0 bytes .../__pycache__/httpsredirect.cpython-312.pyc | Bin 283 -> 0 bytes .../__pycache__/trustedhost.cpython-312.pyc | Bin 277 -> 0 bytes .../__pycache__/wsgi.cpython-312.pyc | Bin 256 -> 0 bytes .../fastapi/middleware/asyncexitstack.py | 18 - .../site-packages/fastapi/middleware/cors.py | 1 - .../site-packages/fastapi/middleware/gzip.py | 1 - .../fastapi/middleware/httpsredirect.py | 3 - .../fastapi/middleware/trustedhost.py | 3 - .../site-packages/fastapi/middleware/wsgi.py | 3 - .../site-packages/fastapi/openapi/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 188 -> 0 bytes .../__pycache__/constants.cpython-312.pyc | Bin 358 -> 0 bytes .../openapi/__pycache__/docs.cpython-312.pyc | Bin 12411 -> 0 bytes .../__pycache__/models.cpython-312.pyc | Bin 22795 -> 0 bytes .../openapi/__pycache__/utils.cpython-312.pyc | Bin 21273 -> 0 bytes .../fastapi/openapi/constants.py | 3 - .../Lib/site-packages/fastapi/openapi/docs.py | 375 - .../site-packages/fastapi/openapi/models.py | 438 - .../site-packages/fastapi/openapi/utils.py | 570 -- .../site-packages/fastapi/param_functions.py | 2461 ----- .env/Lib/site-packages/fastapi/params.py | 755 -- .env/Lib/site-packages/fastapi/py.typed | 0 .env/Lib/site-packages/fastapi/requests.py | 2 - .env/Lib/site-packages/fastapi/responses.py | 48 - .env/Lib/site-packages/fastapi/routing.py | 4657 --------- .../fastapi/security/__init__.py | 15 - .../__pycache__/__init__.cpython-312.pyc | Bin 845 -> 0 bytes .../__pycache__/api_key.cpython-312.pyc | Bin 10482 -> 0 bytes .../security/__pycache__/base.cpython-312.pyc | Bin 507 -> 0 bytes .../security/__pycache__/http.cpython-312.pyc | Bin 14535 -> 0 bytes .../__pycache__/oauth2.cpython-312.pyc | Bin 20667 -> 0 bytes .../open_id_connect_url.cpython-312.pyc | Bin 3761 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 633 -> 0 bytes .../site-packages/fastapi/security/api_key.py | 318 - .../site-packages/fastapi/security/base.py | 6 - .../site-packages/fastapi/security/http.py | 423 - .../site-packages/fastapi/security/oauth2.py | 693 -- .../fastapi/security/open_id_connect_url.py | 94 - .../site-packages/fastapi/security/utils.py | 10 - .env/Lib/site-packages/fastapi/staticfiles.py | 1 - .env/Lib/site-packages/fastapi/templating.py | 1 - .env/Lib/site-packages/fastapi/testclient.py | 1 - .env/Lib/site-packages/fastapi/types.py | 11 - .env/Lib/site-packages/fastapi/utils.py | 164 - .env/Lib/site-packages/fastapi/websockets.py | 3 - .../greenlet-3.3.1.dist-info/INSTALLER | 1 - .../greenlet-3.3.1.dist-info/METADATA | 98 - .../greenlet-3.3.1.dist-info/RECORD | 121 - .../greenlet-3.3.1.dist-info/WHEEL | 5 - .../greenlet-3.3.1.dist-info/licenses/LICENSE | 30 - .../licenses/LICENSE.PSF | 47 - .../greenlet-3.3.1.dist-info/top_level.txt | 1 - .env/Lib/site-packages/greenlet/CObjects.cpp | 157 - .../Lib/site-packages/greenlet/PyGreenlet.cpp | 774 -- .../Lib/site-packages/greenlet/PyGreenlet.hpp | 35 - .../greenlet/PyGreenletUnswitchable.cpp | 147 - .env/Lib/site-packages/greenlet/PyModule.cpp | 292 - .../greenlet/TBrokenGreenlet.cpp | 45 - .../greenlet/TExceptionState.cpp | 62 - .env/Lib/site-packages/greenlet/TGreenlet.cpp | 725 -- .env/Lib/site-packages/greenlet/TGreenlet.hpp | 837 -- .../greenlet/TGreenletGlobals.cpp | 94 - .../site-packages/greenlet/TMainGreenlet.cpp | 160 - .../site-packages/greenlet/TPythonState.cpp | 439 - .../site-packages/greenlet/TStackState.cpp | 265 - .../site-packages/greenlet/TThreadState.hpp | 523 - .../greenlet/TThreadStateCreator.hpp | 102 - .../greenlet/TThreadStateDestroy.cpp | 223 - .../site-packages/greenlet/TUserGreenlet.cpp | 662 -- .env/Lib/site-packages/greenlet/__init__.py | 71 - .../__pycache__/__init__.cpython-312.pyc | Bin 1067 -> 0 bytes .../greenlet/_greenlet.cp312-win_amd64.pyd | Bin 73216 -> 0 bytes .env/Lib/site-packages/greenlet/greenlet.cpp | 323 - .env/Lib/site-packages/greenlet/greenlet.h | 164 - .../greenlet/greenlet_allocator.hpp | 76 - .../greenlet/greenlet_compiler_compat.hpp | 98 - .../greenlet/greenlet_cpython_compat.hpp | 156 - .../greenlet/greenlet_exceptions.hpp | 171 - .../greenlet/greenlet_internal.hpp | 107 - .../greenlet/greenlet_msvc_compat.hpp | 100 - .../site-packages/greenlet/greenlet_refs.hpp | 1118 --- .../greenlet/greenlet_slp_switch.hpp | 103 - .../greenlet/greenlet_thread_support.hpp | 31 - .../greenlet/platform/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 190 -> 0 bytes .../platform/setup_switch_x64_masm.cmd | 2 - .../greenlet/platform/switch_aarch64_gcc.h | 124 - .../greenlet/platform/switch_alpha_unix.h | 30 - .../greenlet/platform/switch_amd64_unix.h | 87 - .../greenlet/platform/switch_arm32_gcc.h | 79 - .../greenlet/platform/switch_arm32_ios.h | 67 - .../greenlet/platform/switch_arm64_masm.asm | 53 - .../greenlet/platform/switch_arm64_masm.obj | Bin 746 -> 0 bytes .../greenlet/platform/switch_arm64_msvc.h | 17 - .../greenlet/platform/switch_csky_gcc.h | 48 - .../platform/switch_loongarch64_linux.h | 31 - .../greenlet/platform/switch_m68k_gcc.h | 38 - .../greenlet/platform/switch_mips_unix.h | 65 - .../greenlet/platform/switch_ppc64_aix.h | 103 - .../greenlet/platform/switch_ppc64_linux.h | 105 - .../greenlet/platform/switch_ppc_aix.h | 87 - .../greenlet/platform/switch_ppc_linux.h | 84 - .../greenlet/platform/switch_ppc_macosx.h | 82 - .../greenlet/platform/switch_ppc_unix.h | 82 - .../greenlet/platform/switch_riscv_unix.h | 41 - .../greenlet/platform/switch_s390_unix.h | 87 - .../greenlet/platform/switch_sh_gcc.h | 36 - .../greenlet/platform/switch_sparc_sun_gcc.h | 92 - .../greenlet/platform/switch_x32_unix.h | 63 - .../greenlet/platform/switch_x64_masm.asm | 111 - .../greenlet/platform/switch_x64_masm.obj | Bin 1078 -> 0 bytes .../greenlet/platform/switch_x64_msvc.h | 60 - .../greenlet/platform/switch_x86_msvc.h | 326 - .../greenlet/platform/switch_x86_unix.h | 105 - .../greenlet/slp_platformselect.h | 77 - .../site-packages/greenlet/tests/__init__.py | 248 - .../__pycache__/__init__.cpython-312.pyc | Bin 9226 -> 0 bytes ...fail_clearing_run_switches.cpython-312.pyc | Bin 2075 -> 0 bytes .../fail_cpp_exception.cpython-312.pyc | Bin 1610 -> 0 bytes ...nitialstub_already_started.cpython-312.pyc | Bin 3478 -> 0 bytes .../fail_slp_switch.cpython-312.pyc | Bin 1303 -> 0 bytes ...ail_switch_three_greenlets.cpython-312.pyc | Bin 1719 -> 0 bytes ...il_switch_three_greenlets2.cpython-312.pyc | Bin 2573 -> 0 bytes .../fail_switch_two_greenlets.cpython-312.pyc | Bin 1690 -> 0 bytes .../__pycache__/leakcheck.cpython-312.pyc | Bin 11682 -> 0 bytes .../test_contextvars.cpython-312.pyc | Bin 15399 -> 0 bytes .../__pycache__/test_cpp.cpython-312.pyc | Bin 4087 -> 0 bytes .../test_extension_interface.cpython-312.pyc | Bin 7447 -> 0 bytes .../tests/__pycache__/test_gc.cpython-312.pyc | Bin 4913 -> 0 bytes .../test_generator.cpython-312.pyc | Bin 3063 -> 0 bytes .../test_generator_nested.cpython-312.pyc | Bin 7757 -> 0 bytes .../__pycache__/test_greenlet.cpython-312.pyc | Bin 76147 -> 0 bytes .../test_greenlet_trash.cpython-312.pyc | Bin 6770 -> 0 bytes .../__pycache__/test_leaks.cpython-312.pyc | Bin 19895 -> 0 bytes .../test_stack_saved.cpython-312.pyc | Bin 1342 -> 0 bytes .../__pycache__/test_throw.cpython-312.pyc | Bin 7350 -> 0 bytes .../__pycache__/test_tracing.cpython-312.pyc | Bin 13836 -> 0 bytes .../__pycache__/test_version.cpython-312.pyc | Bin 2549 -> 0 bytes .../__pycache__/test_weakref.cpython-312.pyc | Bin 2733 -> 0 bytes .../greenlet/tests/_test_extension.c | 258 - .../tests/_test_extension.cp312-win_amd64.pyd | Bin 14336 -> 0 bytes .../_test_extension_cpp.cp312-win_amd64.pyd | Bin 15872 -> 0 bytes .../greenlet/tests/_test_extension_cpp.cpp | 229 - .../tests/fail_clearing_run_switches.py | 47 - .../greenlet/tests/fail_cpp_exception.py | 33 - .../tests/fail_initialstub_already_started.py | 78 - .../greenlet/tests/fail_slp_switch.py | 29 - .../tests/fail_switch_three_greenlets.py | 44 - .../tests/fail_switch_three_greenlets2.py | 55 - .../tests/fail_switch_two_greenlets.py | 41 - .../site-packages/greenlet/tests/leakcheck.py | 336 - .../greenlet/tests/test_contextvars.py | 312 - .../site-packages/greenlet/tests/test_cpp.py | 73 - .../tests/test_extension_interface.py | 115 - .../site-packages/greenlet/tests/test_gc.py | 86 - .../greenlet/tests/test_generator.py | 59 - .../greenlet/tests/test_generator_nested.py | 168 - .../greenlet/tests/test_greenlet.py | 1365 --- .../greenlet/tests/test_greenlet_trash.py | 187 - .../greenlet/tests/test_leaks.py | 457 - .../greenlet/tests/test_stack_saved.py | 19 - .../greenlet/tests/test_throw.py | 128 - .../greenlet/tests/test_tracing.py | 299 - .../greenlet/tests/test_version.py | 41 - .../greenlet/tests/test_weakref.py | 35 - .../h11-0.16.0.dist-info/INSTALLER | 1 - .../h11-0.16.0.dist-info/METADATA | 202 - .../site-packages/h11-0.16.0.dist-info/RECORD | 29 - .../site-packages/h11-0.16.0.dist-info/WHEEL | 5 - .../h11-0.16.0.dist-info/licenses/LICENSE.txt | 22 - .../h11-0.16.0.dist-info/top_level.txt | 1 - .env/Lib/site-packages/h11/__init__.py | 62 - .../h11/__pycache__/__init__.cpython-312.pyc | Bin 1067 -> 0 bytes .../h11/__pycache__/_abnf.cpython-312.pyc | Bin 1773 -> 0 bytes .../__pycache__/_connection.cpython-312.pyc | Bin 23123 -> 0 bytes .../h11/__pycache__/_events.cpython-312.pyc | Bin 13213 -> 0 bytes .../h11/__pycache__/_headers.cpython-312.pyc | Bin 7975 -> 0 bytes .../h11/__pycache__/_readers.cpython-312.pyc | Bin 9640 -> 0 bytes .../_receivebuffer.cpython-312.pyc | Bin 4691 -> 0 bytes .../h11/__pycache__/_state.cpython-312.pyc | Bin 8455 -> 0 bytes .../h11/__pycache__/_util.cpython-312.pyc | Bin 4711 -> 0 bytes .../h11/__pycache__/_version.cpython-312.pyc | Bin 205 -> 0 bytes .../h11/__pycache__/_writers.cpython-312.pyc | Bin 6277 -> 0 bytes .env/Lib/site-packages/h11/_abnf.py | 132 - .env/Lib/site-packages/h11/_connection.py | 659 -- .env/Lib/site-packages/h11/_events.py | 369 - .env/Lib/site-packages/h11/_headers.py | 282 - .env/Lib/site-packages/h11/_readers.py | 250 - .env/Lib/site-packages/h11/_receivebuffer.py | 153 - .env/Lib/site-packages/h11/_state.py | 365 - .env/Lib/site-packages/h11/_util.py | 135 - .env/Lib/site-packages/h11/_version.py | 16 - .env/Lib/site-packages/h11/_writers.py | 145 - .env/Lib/site-packages/h11/py.typed | 1 - .../idna-3.11.dist-info/INSTALLER | 1 - .../idna-3.11.dist-info/METADATA | 209 - .../site-packages/idna-3.11.dist-info/RECORD | 22 - .../site-packages/idna-3.11.dist-info/WHEEL | 4 - .../idna-3.11.dist-info/licenses/LICENSE.md | 31 - .env/Lib/site-packages/idna/__init__.py | 45 - .../idna/__pycache__/__init__.cpython-312.pyc | Bin 875 -> 0 bytes .../idna/__pycache__/codec.cpython-312.pyc | Bin 4965 -> 0 bytes .../idna/__pycache__/compat.cpython-312.pyc | Bin 879 -> 0 bytes .../idna/__pycache__/core.cpython-312.pyc | Bin 16132 -> 0 bytes .../idna/__pycache__/idnadata.cpython-312.pyc | Bin 100904 -> 0 bytes .../__pycache__/intranges.cpython-312.pyc | Bin 2622 -> 0 bytes .../__pycache__/package_data.cpython-312.pyc | Bin 206 -> 0 bytes .../__pycache__/uts46data.cpython-312.pyc | Bin 161834 -> 0 bytes .env/Lib/site-packages/idna/codec.py | 122 - .env/Lib/site-packages/idna/compat.py | 15 - .env/Lib/site-packages/idna/core.py | 437 - .env/Lib/site-packages/idna/idnadata.py | 4309 -------- .env/Lib/site-packages/idna/intranges.py | 57 - .env/Lib/site-packages/idna/package_data.py | 1 - .env/Lib/site-packages/idna/py.typed | 0 .env/Lib/site-packages/idna/uts46data.py | 8841 ----------------- .../pip-24.2.dist-info/AUTHORS.txt | 796 -- .../pip-24.2.dist-info/INSTALLER | 1 - .../pip-24.2.dist-info/LICENSE.txt | 20 - .../site-packages/pip-24.2.dist-info/METADATA | 89 - .../site-packages/pip-24.2.dist-info/RECORD | 853 -- .../pip-24.2.dist-info/REQUESTED | 0 .../site-packages/pip-24.2.dist-info/WHEEL | 5 - .../pip-24.2.dist-info/entry_points.txt | 3 - .../pip-24.2.dist-info/top_level.txt | 1 - .env/Lib/site-packages/pip/__init__.py | 13 - .env/Lib/site-packages/pip/__main__.py | 24 - .env/Lib/site-packages/pip/__pip-runner__.py | 50 - .../pip/__pycache__/__init__.cpython-312.pyc | Bin 685 -> 0 bytes .../pip/__pycache__/__main__.cpython-312.pyc | Bin 841 -> 0 bytes .../__pip-runner__.cpython-312.pyc | Bin 2205 -> 0 bytes .../site-packages/pip/_internal/__init__.py | 18 - .../__pycache__/__init__.cpython-312.pyc | Bin 787 -> 0 bytes .../__pycache__/build_env.cpython-312.pyc | Bin 14472 -> 0 bytes .../__pycache__/cache.cpython-312.pyc | Bin 12667 -> 0 bytes .../__pycache__/configuration.cpython-312.pyc | Bin 17632 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 35553 -> 0 bytes .../__pycache__/main.cpython-312.pyc | Bin 670 -> 0 bytes .../__pycache__/pyproject.cpython-312.pyc | Bin 5118 -> 0 bytes .../self_outdated_check.cpython-312.pyc | Bin 10208 -> 0 bytes .../__pycache__/wheel_builder.cpython-312.pyc | Bin 13619 -> 0 bytes .../site-packages/pip/_internal/build_env.py | 315 - .env/Lib/site-packages/pip/_internal/cache.py | 290 - .../pip/_internal/cli/__init__.py | 4 - .../cli/__pycache__/__init__.cpython-312.pyc | Bin 278 -> 0 bytes .../autocompletion.cpython-312.pyc | Bin 8606 -> 0 bytes .../__pycache__/base_command.cpython-312.pyc | Bin 10196 -> 0 bytes .../__pycache__/cmdoptions.cpython-312.pyc | Bin 30395 -> 0 bytes .../command_context.cpython-312.pyc | Bin 1774 -> 0 bytes .../__pycache__/index_command.cpython-312.pyc | Bin 7124 -> 0 bytes .../cli/__pycache__/main.cpython-312.pyc | Bin 2300 -> 0 bytes .../__pycache__/main_parser.cpython-312.pyc | Bin 4906 -> 0 bytes .../cli/__pycache__/parser.cpython-312.pyc | Bin 15013 -> 0 bytes .../__pycache__/progress_bars.cpython-312.pyc | Bin 3834 -> 0 bytes .../__pycache__/req_command.cpython-312.pyc | Bin 12241 -> 0 bytes .../cli/__pycache__/spinners.cpython-312.pyc | Bin 7833 -> 0 bytes .../__pycache__/status_codes.cpython-312.pyc | Bin 375 -> 0 bytes .../pip/_internal/cli/autocompletion.py | 176 - .../pip/_internal/cli/base_command.py | 231 - .../pip/_internal/cli/cmdoptions.py | 1075 -- .../pip/_internal/cli/command_context.py | 27 - .../pip/_internal/cli/index_command.py | 170 - .../site-packages/pip/_internal/cli/main.py | 80 - .../pip/_internal/cli/main_parser.py | 134 - .../site-packages/pip/_internal/cli/parser.py | 294 - .../pip/_internal/cli/progress_bars.py | 94 - .../pip/_internal/cli/req_command.py | 329 - .../pip/_internal/cli/spinners.py | 159 - .../pip/_internal/cli/status_codes.py | 6 - .../pip/_internal/commands/__init__.py | 132 - .../__pycache__/__init__.cpython-312.pyc | Bin 4002 -> 0 bytes .../__pycache__/cache.cpython-312.pyc | Bin 9701 -> 0 bytes .../__pycache__/check.cpython-312.pyc | Bin 2592 -> 0 bytes .../__pycache__/completion.cpython-312.pyc | Bin 5193 -> 0 bytes .../__pycache__/configuration.cpython-312.pyc | Bin 13163 -> 0 bytes .../__pycache__/debug.cpython-312.pyc | Bin 10068 -> 0 bytes .../__pycache__/download.cpython-312.pyc | Bin 7503 -> 0 bytes .../__pycache__/freeze.cpython-312.pyc | Bin 4385 -> 0 bytes .../commands/__pycache__/hash.cpython-312.pyc | Bin 2968 -> 0 bytes .../commands/__pycache__/help.cpython-312.pyc | Bin 1673 -> 0 bytes .../__pycache__/index.cpython-312.pyc | Bin 6671 -> 0 bytes .../__pycache__/inspect.cpython-312.pyc | Bin 3980 -> 0 bytes .../__pycache__/install.cpython-312.pyc | Bin 29114 -> 0 bytes .../commands/__pycache__/list.cpython-312.pyc | Bin 15757 -> 0 bytes .../__pycache__/search.cpython-312.pyc | Bin 7518 -> 0 bytes .../commands/__pycache__/show.cpython-312.pyc | Bin 10477 -> 0 bytes .../__pycache__/uninstall.cpython-312.pyc | Bin 4709 -> 0 bytes .../__pycache__/wheel.cpython-312.pyc | Bin 8865 -> 0 bytes .../pip/_internal/commands/cache.py | 225 - .../pip/_internal/commands/check.py | 67 - .../pip/_internal/commands/completion.py | 130 - .../pip/_internal/commands/configuration.py | 280 - .../pip/_internal/commands/debug.py | 201 - .../pip/_internal/commands/download.py | 146 - .../pip/_internal/commands/freeze.py | 109 - .../pip/_internal/commands/hash.py | 59 - .../pip/_internal/commands/help.py | 41 - .../pip/_internal/commands/index.py | 139 - .../pip/_internal/commands/inspect.py | 92 - .../pip/_internal/commands/install.py | 783 -- .../pip/_internal/commands/list.py | 375 - .../pip/_internal/commands/search.py | 172 - .../pip/_internal/commands/show.py | 217 - .../pip/_internal/commands/uninstall.py | 114 - .../pip/_internal/commands/wheel.py | 182 - .../pip/_internal/configuration.py | 383 - .../pip/_internal/distributions/__init__.py | 21 - .../__pycache__/__init__.cpython-312.pyc | Bin 941 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 2893 -> 0 bytes .../__pycache__/installed.cpython-312.pyc | Bin 1700 -> 0 bytes .../__pycache__/sdist.cpython-312.pyc | Bin 8439 -> 0 bytes .../__pycache__/wheel.cpython-312.pyc | Bin 2281 -> 0 bytes .../pip/_internal/distributions/base.py | 53 - .../pip/_internal/distributions/installed.py | 29 - .../pip/_internal/distributions/sdist.py | 158 - .../pip/_internal/distributions/wheel.py | 42 - .../site-packages/pip/_internal/exceptions.py | 777 -- .../pip/_internal/index/__init__.py | 2 - .../__pycache__/__init__.cpython-312.pyc | Bin 232 -> 0 bytes .../__pycache__/collector.cpython-312.pyc | Bin 21617 -> 0 bytes .../package_finder.cpython-312.pyc | Bin 40648 -> 0 bytes .../index/__pycache__/sources.cpython-312.pyc | Bin 12588 -> 0 bytes .../pip/_internal/index/collector.py | 494 - .../pip/_internal/index/package_finder.py | 1020 -- .../pip/_internal/index/sources.py | 285 - .../pip/_internal/locations/__init__.py | 456 - .../__pycache__/__init__.cpython-312.pyc | Bin 16440 -> 0 bytes .../__pycache__/_distutils.cpython-312.pyc | Bin 6852 -> 0 bytes .../__pycache__/_sysconfig.cpython-312.pyc | Bin 8028 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 3781 -> 0 bytes .../pip/_internal/locations/_distutils.py | 172 - .../pip/_internal/locations/_sysconfig.py | 214 - .../pip/_internal/locations/base.py | 81 - .env/Lib/site-packages/pip/_internal/main.py | 12 - .../pip/_internal/metadata/__init__.py | 128 - .../__pycache__/__init__.cpython-312.pyc | Bin 5876 -> 0 bytes .../__pycache__/_json.cpython-312.pyc | Bin 2926 -> 0 bytes .../metadata/__pycache__/base.cpython-312.pyc | Bin 35198 -> 0 bytes .../__pycache__/pkg_resources.cpython-312.pyc | Bin 16084 -> 0 bytes .../pip/_internal/metadata/_json.py | 84 - .../pip/_internal/metadata/base.py | 688 -- .../_internal/metadata/importlib/__init__.py | 6 - .../__pycache__/__init__.cpython-312.pyc | Bin 358 -> 0 bytes .../__pycache__/_compat.cpython-312.pyc | Bin 4491 -> 0 bytes .../__pycache__/_dists.cpython-312.pyc | Bin 12565 -> 0 bytes .../__pycache__/_envs.cpython-312.pyc | Bin 11079 -> 0 bytes .../_internal/metadata/importlib/_compat.py | 85 - .../_internal/metadata/importlib/_dists.py | 221 - .../pip/_internal/metadata/importlib/_envs.py | 189 - .../pip/_internal/metadata/pkg_resources.py | 301 - .../pip/_internal/models/__init__.py | 2 - .../__pycache__/__init__.cpython-312.pyc | Bin 266 -> 0 bytes .../__pycache__/candidate.cpython-312.pyc | Bin 1604 -> 0 bytes .../__pycache__/direct_url.cpython-312.pyc | Bin 10844 -> 0 bytes .../format_control.cpython-312.pyc | Bin 4223 -> 0 bytes .../models/__pycache__/index.cpython-312.pyc | Bin 1694 -> 0 bytes .../installation_report.cpython-312.pyc | Bin 2277 -> 0 bytes .../models/__pycache__/link.cpython-312.pyc | Bin 26617 -> 0 bytes .../models/__pycache__/scheme.cpython-312.pyc | Bin 1023 -> 0 bytes .../__pycache__/search_scope.cpython-312.pyc | Bin 4987 -> 0 bytes .../selection_prefs.cpython-312.pyc | Bin 1851 -> 0 bytes .../__pycache__/target_python.cpython-312.pyc | Bin 4953 -> 0 bytes .../models/__pycache__/wheel.cpython-312.pyc | Bin 5760 -> 0 bytes .../pip/_internal/models/candidate.py | 25 - .../pip/_internal/models/direct_url.py | 224 - .../pip/_internal/models/format_control.py | 78 - .../pip/_internal/models/index.py | 28 - .../_internal/models/installation_report.py | 56 - .../pip/_internal/models/link.py | 590 -- .../pip/_internal/models/scheme.py | 25 - .../pip/_internal/models/search_scope.py | 127 - .../pip/_internal/models/selection_prefs.py | 53 - .../pip/_internal/models/target_python.py | 121 - .../pip/_internal/models/wheel.py | 93 - .../pip/_internal/network/__init__.py | 2 - .../__pycache__/__init__.cpython-312.pyc | Bin 254 -> 0 bytes .../network/__pycache__/auth.cpython-312.pyc | Bin 22099 -> 0 bytes .../network/__pycache__/cache.cpython-312.pyc | Bin 6460 -> 0 bytes .../__pycache__/download.cpython-312.pyc | Bin 8479 -> 0 bytes .../__pycache__/lazy_wheel.cpython-312.pyc | Bin 11617 -> 0 bytes .../__pycache__/session.cpython-312.pyc | Bin 18874 -> 0 bytes .../network/__pycache__/utils.cpython-312.pyc | Bin 2255 -> 0 bytes .../__pycache__/xmlrpc.cpython-312.pyc | Bin 2949 -> 0 bytes .../pip/_internal/network/auth.py | 566 -- .../pip/_internal/network/cache.py | 106 - .../pip/_internal/network/download.py | 187 - .../pip/_internal/network/lazy_wheel.py | 210 - .../pip/_internal/network/session.py | 522 - .../pip/_internal/network/utils.py | 98 - .../pip/_internal/network/xmlrpc.py | 62 - .../pip/_internal/operations/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 197 -> 0 bytes .../__pycache__/check.cpython-312.pyc | Bin 7104 -> 0 bytes .../__pycache__/freeze.cpython-312.pyc | Bin 10128 -> 0 bytes .../__pycache__/prepare.cpython-312.pyc | Bin 25791 -> 0 bytes .../_internal/operations/build/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 203 -> 0 bytes .../__pycache__/build_tracker.cpython-312.pyc | Bin 7667 -> 0 bytes .../__pycache__/metadata.cpython-312.pyc | Bin 1857 -> 0 bytes .../metadata_editable.cpython-312.pyc | Bin 1891 -> 0 bytes .../metadata_legacy.cpython-312.pyc | Bin 3011 -> 0 bytes .../build/__pycache__/wheel.cpython-312.pyc | Bin 1677 -> 0 bytes .../wheel_editable.cpython-312.pyc | Bin 2018 -> 0 bytes .../__pycache__/wheel_legacy.cpython-312.pyc | Bin 3850 -> 0 bytes .../operations/build/build_tracker.py | 138 - .../_internal/operations/build/metadata.py | 39 - .../operations/build/metadata_editable.py | 41 - .../operations/build/metadata_legacy.py | 74 - .../pip/_internal/operations/build/wheel.py | 37 - .../operations/build/wheel_editable.py | 46 - .../operations/build/wheel_legacy.py | 102 - .../pip/_internal/operations/check.py | 181 - .../pip/_internal/operations/freeze.py | 258 - .../_internal/operations/install/__init__.py | 2 - .../__pycache__/__init__.cpython-312.pyc | Bin 266 -> 0 bytes .../editable_legacy.cpython-312.pyc | Bin 1798 -> 0 bytes .../install/__pycache__/wheel.cpython-312.pyc | Bin 34105 -> 0 bytes .../operations/install/editable_legacy.py | 47 - .../pip/_internal/operations/install/wheel.py | 741 -- .../pip/_internal/operations/prepare.py | 732 -- .../site-packages/pip/_internal/pyproject.py | 185 - .../pip/_internal/req/__init__.py | 90 - .../req/__pycache__/__init__.cpython-312.pyc | Bin 3445 -> 0 bytes .../__pycache__/constructors.cpython-312.pyc | Bin 21220 -> 0 bytes .../req/__pycache__/req_file.cpython-312.pyc | Bin 21432 -> 0 bytes .../__pycache__/req_install.cpython-312.pyc | Bin 38484 -> 0 bytes .../req/__pycache__/req_set.cpython-312.pyc | Bin 5483 -> 0 bytes .../__pycache__/req_uninstall.cpython-312.pyc | Bin 32094 -> 0 bytes .../pip/_internal/req/constructors.py | 560 -- .../pip/_internal/req/req_file.py | 551 - .../pip/_internal/req/req_install.py | 934 -- .../pip/_internal/req/req_set.py | 82 - .../pip/_internal/req/req_uninstall.py | 633 -- .../pip/_internal/resolution/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 197 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 1185 -> 0 bytes .../pip/_internal/resolution/base.py | 20 - .../_internal/resolution/legacy/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 204 -> 0 bytes .../__pycache__/resolver.cpython-312.pyc | Bin 22579 -> 0 bytes .../_internal/resolution/legacy/resolver.py | 597 -- .../resolution/resolvelib/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 208 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 8149 -> 0 bytes .../__pycache__/candidates.cpython-312.pyc | Bin 29172 -> 0 bytes .../__pycache__/factory.cpython-312.pyc | Bin 32286 -> 0 bytes .../found_candidates.cpython-312.pyc | Bin 6793 -> 0 bytes .../__pycache__/provider.cpython-312.pyc | Bin 10523 -> 0 bytes .../__pycache__/reporter.cpython-312.pyc | Bin 5040 -> 0 bytes .../__pycache__/requirements.cpython-312.pyc | Bin 15356 -> 0 bytes .../__pycache__/resolver.cpython-312.pyc | Bin 12314 -> 0 bytes .../_internal/resolution/resolvelib/base.py | 139 - .../resolution/resolvelib/candidates.py | 569 -- .../resolution/resolvelib/factory.py | 817 -- .../resolution/resolvelib/found_candidates.py | 174 - .../resolution/resolvelib/provider.py | 258 - .../resolution/resolvelib/reporter.py | 81 - .../resolution/resolvelib/requirements.py | 245 - .../resolution/resolvelib/resolver.py | 317 - .../pip/_internal/self_outdated_check.py | 244 - .../pip/_internal/utils/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 192 -> 0 bytes .../__pycache__/_jaraco_text.cpython-312.pyc | Bin 4527 -> 0 bytes .../utils/__pycache__/_log.cpython-312.pyc | Bin 1863 -> 0 bytes .../utils/__pycache__/appdirs.cpython-312.pyc | Bin 2407 -> 0 bytes .../utils/__pycache__/compat.cpython-312.pyc | Bin 2904 -> 0 bytes .../compatibility_tags.cpython-312.pyc | Bin 5560 -> 0 bytes .../__pycache__/datetime.cpython-312.pyc | Bin 681 -> 0 bytes .../__pycache__/deprecation.cpython-312.pyc | Bin 4188 -> 0 bytes .../direct_url_helpers.cpython-312.pyc | Bin 3533 -> 0 bytes .../__pycache__/egg_link.cpython-312.pyc | Bin 3203 -> 0 bytes .../__pycache__/encoding.cpython-312.pyc | Bin 2145 -> 0 bytes .../__pycache__/entrypoints.cpython-312.pyc | Bin 3990 -> 0 bytes .../__pycache__/filesystem.cpython-312.pyc | Bin 7326 -> 0 bytes .../__pycache__/filetypes.cpython-312.pyc | Bin 1161 -> 0 bytes .../utils/__pycache__/glibc.cpython-312.pyc | Bin 2416 -> 0 bytes .../utils/__pycache__/hashes.cpython-312.pyc | Bin 7605 -> 0 bytes .../utils/__pycache__/logging.cpython-312.pyc | Bin 13555 -> 0 bytes .../utils/__pycache__/misc.cpython-312.pyc | Bin 33548 -> 0 bytes .../__pycache__/packaging.cpython-312.pyc | Bin 2580 -> 0 bytes .../utils/__pycache__/retry.cpython-312.pyc | Bin 2105 -> 0 bytes .../setuptools_build.cpython-312.pyc | Bin 4547 -> 0 bytes .../__pycache__/subprocess.cpython-312.pyc | Bin 8636 -> 0 bytes .../__pycache__/temp_dir.cpython-312.pyc | Bin 12021 -> 0 bytes .../__pycache__/unpacking.cpython-312.pyc | Bin 13495 -> 0 bytes .../utils/__pycache__/urls.cpython-312.pyc | Bin 2074 -> 0 bytes .../__pycache__/virtualenv.cpython-312.pyc | Bin 4468 -> 0 bytes .../utils/__pycache__/wheel.cpython-312.pyc | Bin 5900 -> 0 bytes .../pip/_internal/utils/_jaraco_text.py | 109 - .../site-packages/pip/_internal/utils/_log.py | 38 - .../pip/_internal/utils/appdirs.py | 52 - .../pip/_internal/utils/compat.py | 79 - .../pip/_internal/utils/compatibility_tags.py | 165 - .../pip/_internal/utils/datetime.py | 11 - .../pip/_internal/utils/deprecation.py | 124 - .../pip/_internal/utils/direct_url_helpers.py | 87 - .../pip/_internal/utils/egg_link.py | 80 - .../pip/_internal/utils/encoding.py | 36 - .../pip/_internal/utils/entrypoints.py | 84 - .../pip/_internal/utils/filesystem.py | 149 - .../pip/_internal/utils/filetypes.py | 27 - .../pip/_internal/utils/glibc.py | 101 - .../pip/_internal/utils/hashes.py | 147 - .../pip/_internal/utils/logging.py | 347 - .../site-packages/pip/_internal/utils/misc.py | 777 -- .../pip/_internal/utils/packaging.py | 57 - .../pip/_internal/utils/retry.py | 42 - .../pip/_internal/utils/setuptools_build.py | 146 - .../pip/_internal/utils/subprocess.py | 245 - .../pip/_internal/utils/temp_dir.py | 296 - .../pip/_internal/utils/unpacking.py | 337 - .../site-packages/pip/_internal/utils/urls.py | 55 - .../pip/_internal/utils/virtualenv.py | 104 - .../pip/_internal/utils/wheel.py | 134 - .../pip/_internal/vcs/__init__.py | 15 - .../vcs/__pycache__/__init__.cpython-312.pyc | Bin 531 -> 0 bytes .../vcs/__pycache__/bazaar.cpython-312.pyc | Bin 5052 -> 0 bytes .../vcs/__pycache__/git.cpython-312.pyc | Bin 19017 -> 0 bytes .../vcs/__pycache__/mercurial.cpython-312.pyc | Bin 7605 -> 0 bytes .../__pycache__/subversion.cpython-312.pyc | Bin 12524 -> 0 bytes .../versioncontrol.cpython-312.pyc | Bin 28997 -> 0 bytes .../site-packages/pip/_internal/vcs/bazaar.py | 112 - .../site-packages/pip/_internal/vcs/git.py | 527 - .../pip/_internal/vcs/mercurial.py | 163 - .../pip/_internal/vcs/subversion.py | 324 - .../pip/_internal/vcs/versioncontrol.py | 688 -- .../pip/_internal/wheel_builder.py | 354 - .../Lib/site-packages/pip/_vendor/__init__.py | 116 - .../__pycache__/__init__.cpython-312.pyc | Bin 4550 -> 0 bytes .../typing_extensions.cpython-312.pyc | Bin 139452 -> 0 bytes .../pip/_vendor/cachecontrol/__init__.py | 28 - .../__pycache__/__init__.cpython-312.pyc | Bin 903 -> 0 bytes .../__pycache__/_cmd.cpython-312.pyc | Bin 2647 -> 0 bytes .../__pycache__/adapter.cpython-312.pyc | Bin 6465 -> 0 bytes .../__pycache__/cache.cpython-312.pyc | Bin 3788 -> 0 bytes .../__pycache__/controller.cpython-312.pyc | Bin 16225 -> 0 bytes .../__pycache__/filewrapper.cpython-312.pyc | Bin 4348 -> 0 bytes .../__pycache__/heuristics.cpython-312.pyc | Bin 6695 -> 0 bytes .../__pycache__/serialize.cpython-312.pyc | Bin 5262 -> 0 bytes .../__pycache__/wrapper.cpython-312.pyc | Bin 1675 -> 0 bytes .../pip/_vendor/cachecontrol/_cmd.py | 70 - .../pip/_vendor/cachecontrol/adapter.py | 161 - .../pip/_vendor/cachecontrol/cache.py | 74 - .../_vendor/cachecontrol/caches/__init__.py | 8 - .../__pycache__/__init__.cpython-312.pyc | Bin 436 -> 0 bytes .../__pycache__/file_cache.cpython-312.pyc | Bin 7769 -> 0 bytes .../__pycache__/redis_cache.cpython-312.pyc | Bin 2734 -> 0 bytes .../_vendor/cachecontrol/caches/file_cache.py | 182 - .../cachecontrol/caches/redis_cache.py | 48 - .../pip/_vendor/cachecontrol/controller.py | 499 - .../pip/_vendor/cachecontrol/filewrapper.py | 119 - .../pip/_vendor/cachecontrol/heuristics.py | 154 - .../pip/_vendor/cachecontrol/py.typed | 0 .../pip/_vendor/cachecontrol/serialize.py | 146 - .../pip/_vendor/cachecontrol/wrapper.py | 43 - .../pip/_vendor/certifi/__init__.py | 4 - .../pip/_vendor/certifi/__main__.py | 12 - .../__pycache__/__init__.cpython-312.pyc | Bin 319 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 646 -> 0 bytes .../certifi/__pycache__/core.cpython-312.pyc | Bin 3217 -> 0 bytes .../pip/_vendor/certifi/cacert.pem | 4798 --------- .../site-packages/pip/_vendor/certifi/core.py | 114 - .../pip/_vendor/certifi/py.typed | 0 .../pip/_vendor/distlib/__init__.py | 33 - .../__pycache__/__init__.cpython-312.pyc | Bin 1270 -> 0 bytes .../__pycache__/compat.cpython-312.pyc | Bin 45532 -> 0 bytes .../__pycache__/database.cpython-312.pyc | Bin 65686 -> 0 bytes .../distlib/__pycache__/index.cpython-312.pyc | Bin 24317 -> 0 bytes .../__pycache__/locators.cpython-312.pyc | Bin 59931 -> 0 bytes .../__pycache__/manifest.cpython-312.pyc | Bin 15077 -> 0 bytes .../__pycache__/markers.cpython-312.pyc | Bin 7679 -> 0 bytes .../__pycache__/metadata.cpython-312.pyc | Bin 41667 -> 0 bytes .../__pycache__/resources.cpython-312.pyc | Bin 17318 -> 0 bytes .../__pycache__/scripts.cpython-312.pyc | Bin 19775 -> 0 bytes .../distlib/__pycache__/util.cpython-312.pyc | Bin 88018 -> 0 bytes .../__pycache__/version.cpython-312.pyc | Bin 30346 -> 0 bytes .../distlib/__pycache__/wheel.cpython-312.pyc | Bin 51455 -> 0 bytes .../pip/_vendor/distlib/compat.py | 1138 --- .../pip/_vendor/distlib/database.py | 1359 --- .../pip/_vendor/distlib/index.py | 508 - .../pip/_vendor/distlib/locators.py | 1303 --- .../pip/_vendor/distlib/manifest.py | 384 - .../pip/_vendor/distlib/markers.py | 167 - .../pip/_vendor/distlib/metadata.py | 1068 -- .../pip/_vendor/distlib/resources.py | 358 - .../pip/_vendor/distlib/scripts.py | 466 - .../site-packages/pip/_vendor/distlib/t32.exe | Bin 97792 -> 0 bytes .../pip/_vendor/distlib/t64-arm.exe | Bin 182784 -> 0 bytes .../site-packages/pip/_vendor/distlib/t64.exe | Bin 108032 -> 0 bytes .../site-packages/pip/_vendor/distlib/util.py | 2025 ---- .../pip/_vendor/distlib/version.py | 751 -- .../site-packages/pip/_vendor/distlib/w32.exe | Bin 91648 -> 0 bytes .../pip/_vendor/distlib/w64-arm.exe | Bin 168448 -> 0 bytes .../site-packages/pip/_vendor/distlib/w64.exe | Bin 101888 -> 0 bytes .../pip/_vendor/distlib/wheel.py | 1099 -- .../pip/_vendor/distro/__init__.py | 54 - .../pip/_vendor/distro/__main__.py | 4 - .../__pycache__/__init__.cpython-312.pyc | Bin 961 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 293 -> 0 bytes .../distro/__pycache__/distro.cpython-312.pyc | Bin 53809 -> 0 bytes .../pip/_vendor/distro/distro.py | 1403 --- .../site-packages/pip/_vendor/distro/py.typed | 0 .../pip/_vendor/idna/__init__.py | 44 - .../idna/__pycache__/__init__.cpython-312.pyc | Bin 882 -> 0 bytes .../idna/__pycache__/codec.cpython-312.pyc | Bin 4977 -> 0 bytes .../idna/__pycache__/compat.cpython-312.pyc | Bin 888 -> 0 bytes .../idna/__pycache__/core.cpython-312.pyc | Bin 15792 -> 0 bytes .../idna/__pycache__/idnadata.cpython-312.pyc | Bin 99477 -> 0 bytes .../__pycache__/intranges.cpython-312.pyc | Bin 2634 -> 0 bytes .../__pycache__/package_data.cpython-312.pyc | Bin 217 -> 0 bytes .../__pycache__/uts46data.cpython-312.pyc | Bin 158849 -> 0 bytes .../site-packages/pip/_vendor/idna/codec.py | 118 - .../site-packages/pip/_vendor/idna/compat.py | 13 - .../site-packages/pip/_vendor/idna/core.py | 395 - .../pip/_vendor/idna/idnadata.py | 4245 -------- .../pip/_vendor/idna/intranges.py | 54 - .../pip/_vendor/idna/package_data.py | 2 - .../site-packages/pip/_vendor/idna/py.typed | 0 .../pip/_vendor/idna/uts46data.py | 8598 ---------------- .../pip/_vendor/msgpack/__init__.py | 55 - .../__pycache__/__init__.cpython-312.pyc | Bin 1742 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 2026 -> 0 bytes .../msgpack/__pycache__/ext.cpython-312.pyc | Bin 8171 -> 0 bytes .../__pycache__/fallback.cpython-312.pyc | Bin 42044 -> 0 bytes .../pip/_vendor/msgpack/exceptions.py | 48 - .../site-packages/pip/_vendor/msgpack/ext.py | 168 - .../pip/_vendor/msgpack/fallback.py | 951 -- .../pip/_vendor/packaging/__init__.py | 15 - .../__pycache__/__init__.cpython-312.pyc | Bin 559 -> 0 bytes .../__pycache__/_elffile.cpython-312.pyc | Bin 4968 -> 0 bytes .../__pycache__/_manylinux.cpython-312.pyc | Bin 9696 -> 0 bytes .../__pycache__/_musllinux.cpython-312.pyc | Bin 4555 -> 0 bytes .../__pycache__/_parser.cpython-312.pyc | Bin 13986 -> 0 bytes .../__pycache__/_structures.cpython-312.pyc | Bin 3242 -> 0 bytes .../__pycache__/_tokenizer.cpython-312.pyc | Bin 7916 -> 0 bytes .../__pycache__/markers.cpython-312.pyc | Bin 11012 -> 0 bytes .../__pycache__/metadata.cpython-312.pyc | Bin 24954 -> 0 bytes .../__pycache__/requirements.cpython-312.pyc | Bin 4411 -> 0 bytes .../__pycache__/specifiers.cpython-312.pyc | Bin 38740 -> 0 bytes .../__pycache__/tags.cpython-312.pyc | Bin 21345 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 7342 -> 0 bytes .../__pycache__/version.cpython-312.pyc | Bin 19508 -> 0 bytes .../pip/_vendor/packaging/_elffile.py | 110 - .../pip/_vendor/packaging/_manylinux.py | 262 - .../pip/_vendor/packaging/_musllinux.py | 85 - .../pip/_vendor/packaging/_parser.py | 354 - .../pip/_vendor/packaging/_structures.py | 61 - .../pip/_vendor/packaging/_tokenizer.py | 194 - .../pip/_vendor/packaging/markers.py | 325 - .../pip/_vendor/packaging/metadata.py | 804 -- .../pip/_vendor/packaging/py.typed | 0 .../pip/_vendor/packaging/requirements.py | 91 - .../pip/_vendor/packaging/specifiers.py | 1009 -- .../pip/_vendor/packaging/tags.py | 568 -- .../pip/_vendor/packaging/utils.py | 174 - .../pip/_vendor/packaging/version.py | 563 -- .../pip/_vendor/pkg_resources/__init__.py | 3676 ------- .../__pycache__/__init__.cpython-312.pyc | Bin 161282 -> 0 bytes .../pip/_vendor/platformdirs/__init__.py | 627 -- .../pip/_vendor/platformdirs/__main__.py | 55 - .../__pycache__/__init__.cpython-312.pyc | Bin 19830 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 1949 -> 0 bytes .../__pycache__/android.cpython-312.pyc | Bin 10698 -> 0 bytes .../__pycache__/api.cpython-312.pyc | Bin 12912 -> 0 bytes .../__pycache__/macos.cpython-312.pyc | Bin 8008 -> 0 bytes .../__pycache__/unix.cpython-312.pyc | Bin 15038 -> 0 bytes .../__pycache__/version.cpython-312.pyc | Bin 598 -> 0 bytes .../__pycache__/windows.cpython-312.pyc | Bin 13675 -> 0 bytes .../pip/_vendor/platformdirs/android.py | 249 - .../pip/_vendor/platformdirs/api.py | 292 - .../pip/_vendor/platformdirs/macos.py | 130 - .../pip/_vendor/platformdirs/py.typed | 0 .../pip/_vendor/platformdirs/unix.py | 275 - .../pip/_vendor/platformdirs/version.py | 16 - .../pip/_vendor/platformdirs/windows.py | 272 - .../pip/_vendor/pygments/__init__.py | 82 - .../pip/_vendor/pygments/__main__.py | 17 - .../__pycache__/__init__.cpython-312.pyc | Bin 3490 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 736 -> 0 bytes .../__pycache__/cmdline.cpython-312.pyc | Bin 26586 -> 0 bytes .../__pycache__/console.cpython-312.pyc | Bin 2630 -> 0 bytes .../__pycache__/filter.cpython-312.pyc | Bin 3223 -> 0 bytes .../__pycache__/formatter.cpython-312.pyc | Bin 4722 -> 0 bytes .../__pycache__/lexer.cpython-312.pyc | Bin 38363 -> 0 bytes .../__pycache__/modeline.cpython-312.pyc | Bin 1561 -> 0 bytes .../__pycache__/plugin.cpython-312.pyc | Bin 2610 -> 0 bytes .../__pycache__/regexopt.cpython-312.pyc | Bin 4079 -> 0 bytes .../__pycache__/scanner.cpython-312.pyc | Bin 4758 -> 0 bytes .../__pycache__/sphinxext.cpython-312.pyc | Bin 12100 -> 0 bytes .../__pycache__/style.cpython-312.pyc | Bin 6695 -> 0 bytes .../__pycache__/token.cpython-312.pyc | Bin 8191 -> 0 bytes .../__pycache__/unistring.cpython-312.pyc | Bin 32974 -> 0 bytes .../pygments/__pycache__/util.cpython-312.pyc | Bin 14071 -> 0 bytes .../pip/_vendor/pygments/cmdline.py | 668 -- .../pip/_vendor/pygments/console.py | 70 - .../pip/_vendor/pygments/filter.py | 70 - .../pip/_vendor/pygments/filters/__init__.py | 940 -- .../__pycache__/__init__.cpython-312.pyc | Bin 37913 -> 0 bytes .../pip/_vendor/pygments/formatter.py | 129 - .../_vendor/pygments/formatters/__init__.py | 157 - .../__pycache__/__init__.cpython-312.pyc | Bin 6904 -> 0 bytes .../__pycache__/_mapping.cpython-312.pyc | Bin 4217 -> 0 bytes .../__pycache__/bbcode.cpython-312.pyc | Bin 4224 -> 0 bytes .../__pycache__/groff.cpython-312.pyc | Bin 7295 -> 0 bytes .../__pycache__/html.cpython-312.pyc | Bin 41028 -> 0 bytes .../__pycache__/img.cpython-312.pyc | Bin 28550 -> 0 bytes .../__pycache__/irc.cpython-312.pyc | Bin 6057 -> 0 bytes .../__pycache__/latex.cpython-312.pyc | Bin 20127 -> 0 bytes .../__pycache__/other.cpython-312.pyc | Bin 6879 -> 0 bytes .../__pycache__/pangomarkup.cpython-312.pyc | Bin 2960 -> 0 bytes .../__pycache__/rtf.cpython-312.pyc | Bin 13775 -> 0 bytes .../__pycache__/svg.cpython-312.pyc | Bin 9141 -> 0 bytes .../__pycache__/terminal.cpython-312.pyc | Bin 5821 -> 0 bytes .../__pycache__/terminal256.cpython-312.pyc | Bin 15120 -> 0 bytes .../_vendor/pygments/formatters/_mapping.py | 23 - .../pip/_vendor/pygments/formatters/bbcode.py | 108 - .../pip/_vendor/pygments/formatters/groff.py | 170 - .../pip/_vendor/pygments/formatters/html.py | 987 -- .../pip/_vendor/pygments/formatters/img.py | 685 -- .../pip/_vendor/pygments/formatters/irc.py | 154 - .../pip/_vendor/pygments/formatters/latex.py | 518 - .../pip/_vendor/pygments/formatters/other.py | 160 - .../pygments/formatters/pangomarkup.py | 83 - .../pip/_vendor/pygments/formatters/rtf.py | 349 - .../pip/_vendor/pygments/formatters/svg.py | 185 - .../_vendor/pygments/formatters/terminal.py | 127 - .../pygments/formatters/terminal256.py | 338 - .../pip/_vendor/pygments/lexer.py | 963 -- .../pip/_vendor/pygments/lexers/__init__.py | 362 - .../__pycache__/__init__.cpython-312.pyc | Bin 14623 -> 0 bytes .../__pycache__/_mapping.cpython-312.pyc | Bin 68265 -> 0 bytes .../lexers/__pycache__/python.cpython-312.pyc | Bin 42969 -> 0 bytes .../pip/_vendor/pygments/lexers/_mapping.py | 589 -- .../pip/_vendor/pygments/lexers/python.py | 1198 --- .../pip/_vendor/pygments/modeline.py | 43 - .../pip/_vendor/pygments/plugin.py | 72 - .../pip/_vendor/pygments/regexopt.py | 91 - .../pip/_vendor/pygments/scanner.py | 104 - .../pip/_vendor/pygments/sphinxext.py | 247 - .../pip/_vendor/pygments/style.py | 203 - .../pip/_vendor/pygments/styles/__init__.py | 61 - .../__pycache__/__init__.cpython-312.pyc | Bin 2666 -> 0 bytes .../__pycache__/_mapping.cpython-312.pyc | Bin 3650 -> 0 bytes .../pip/_vendor/pygments/styles/_mapping.py | 54 - .../pip/_vendor/pygments/token.py | 214 - .../pip/_vendor/pygments/unistring.py | 153 - .../pip/_vendor/pygments/util.py | 324 - .../pip/_vendor/pyproject_hooks/__init__.py | 23 - .../__pycache__/__init__.cpython-312.pyc | Bin 615 -> 0 bytes .../__pycache__/_compat.cpython-312.pyc | Bin 376 -> 0 bytes .../__pycache__/_impl.cpython-312.pyc | Bin 14702 -> 0 bytes .../pip/_vendor/pyproject_hooks/_compat.py | 8 - .../pip/_vendor/pyproject_hooks/_impl.py | 330 - .../pyproject_hooks/_in_process/__init__.py | 18 - .../__pycache__/__init__.cpython-312.pyc | Bin 1082 -> 0 bytes .../__pycache__/_in_process.cpython-312.pyc | Bin 14360 -> 0 bytes .../_in_process/_in_process.py | 353 - .../pip/_vendor/requests/__init__.py | 179 - .../__pycache__/__init__.cpython-312.pyc | Bin 5255 -> 0 bytes .../__pycache__/__version__.cpython-312.pyc | Bin 586 -> 0 bytes .../_internal_utils.cpython-312.pyc | Bin 2026 -> 0 bytes .../__pycache__/adapters.cpython-312.pyc | Bin 28433 -> 0 bytes .../requests/__pycache__/api.cpython-312.pyc | Bin 7200 -> 0 bytes .../requests/__pycache__/auth.cpython-312.pyc | Bin 13923 -> 0 bytes .../__pycache__/certs.cpython-312.pyc | Bin 924 -> 0 bytes .../__pycache__/compat.cpython-312.pyc | Bin 1679 -> 0 bytes .../__pycache__/cookies.cpython-312.pyc | Bin 25200 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 7600 -> 0 bytes .../requests/__pycache__/help.cpython-312.pyc | Bin 4230 -> 0 bytes .../__pycache__/hooks.cpython-312.pyc | Bin 1053 -> 0 bytes .../__pycache__/models.cpython-312.pyc | Bin 35430 -> 0 bytes .../__pycache__/packages.cpython-312.pyc | Bin 1268 -> 0 bytes .../__pycache__/sessions.cpython-312.pyc | Bin 27848 -> 0 bytes .../__pycache__/status_codes.cpython-312.pyc | Bin 6025 -> 0 bytes .../__pycache__/structures.cpython-312.pyc | Bin 5625 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 36368 -> 0 bytes .../pip/_vendor/requests/__version__.py | 14 - .../pip/_vendor/requests/_internal_utils.py | 50 - .../pip/_vendor/requests/adapters.py | 719 -- .../site-packages/pip/_vendor/requests/api.py | 157 - .../pip/_vendor/requests/auth.py | 314 - .../pip/_vendor/requests/certs.py | 24 - .../pip/_vendor/requests/compat.py | 78 - .../pip/_vendor/requests/cookies.py | 561 -- .../pip/_vendor/requests/exceptions.py | 151 - .../pip/_vendor/requests/help.py | 127 - .../pip/_vendor/requests/hooks.py | 33 - .../pip/_vendor/requests/models.py | 1037 -- .../pip/_vendor/requests/packages.py | 25 - .../pip/_vendor/requests/sessions.py | 831 -- .../pip/_vendor/requests/status_codes.py | 128 - .../pip/_vendor/requests/structures.py | 99 - .../pip/_vendor/requests/utils.py | 1096 -- .../pip/_vendor/resolvelib/__init__.py | 26 - .../__pycache__/__init__.cpython-312.pyc | Bin 636 -> 0 bytes .../__pycache__/providers.cpython-312.pyc | Bin 6853 -> 0 bytes .../__pycache__/reporters.cpython-312.pyc | Bin 2656 -> 0 bytes .../__pycache__/resolvers.cpython-312.pyc | Bin 25882 -> 0 bytes .../__pycache__/structs.cpython-312.pyc | Bin 10502 -> 0 bytes .../pip/_vendor/resolvelib/compat/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 202 -> 0 bytes .../collections_abc.cpython-312.pyc | Bin 422 -> 0 bytes .../resolvelib/compat/collections_abc.py | 6 - .../pip/_vendor/resolvelib/providers.py | 133 - .../pip/_vendor/resolvelib/py.typed | 0 .../pip/_vendor/resolvelib/reporters.py | 43 - .../pip/_vendor/resolvelib/resolvers.py | 547 - .../pip/_vendor/resolvelib/structs.py | 170 - .../pip/_vendor/rich/__init__.py | 177 - .../pip/_vendor/rich/__main__.py | 273 - .../rich/__pycache__/__init__.cpython-312.pyc | Bin 7017 -> 0 bytes .../rich/__pycache__/__main__.cpython-312.pyc | Bin 10294 -> 0 bytes .../__pycache__/_cell_widths.cpython-312.pyc | Bin 7874 -> 0 bytes .../__pycache__/_emoji_codes.cpython-312.pyc | Bin 205978 -> 0 bytes .../_emoji_replace.cpython-312.pyc | Bin 1731 -> 0 bytes .../_export_format.cpython-312.pyc | Bin 2351 -> 0 bytes .../__pycache__/_extension.cpython-312.pyc | Bin 539 -> 0 bytes .../rich/__pycache__/_fileno.cpython-312.pyc | Bin 857 -> 0 bytes .../rich/__pycache__/_inspect.cpython-312.pyc | Bin 12075 -> 0 bytes .../__pycache__/_log_render.cpython-312.pyc | Bin 4149 -> 0 bytes .../rich/__pycache__/_loop.cpython-312.pyc | Bin 1872 -> 0 bytes .../__pycache__/_null_file.cpython-312.pyc | Bin 3622 -> 0 bytes .../__pycache__/_palettes.cpython-312.pyc | Bin 5162 -> 0 bytes .../rich/__pycache__/_pick.cpython-312.pyc | Bin 723 -> 0 bytes .../rich/__pycache__/_ratio.cpython-312.pyc | Bin 6572 -> 0 bytes .../__pycache__/_spinners.cpython-312.pyc | Bin 13181 -> 0 bytes .../rich/__pycache__/_stack.cpython-312.pyc | Bin 967 -> 0 bytes .../rich/__pycache__/_timer.cpython-312.pyc | Bin 867 -> 0 bytes .../_win32_console.cpython-312.pyc | Bin 28978 -> 0 bytes .../rich/__pycache__/_windows.cpython-312.pyc | Bin 2492 -> 0 bytes .../_windows_renderer.cpython-312.pyc | Bin 3565 -> 0 bytes .../rich/__pycache__/_wrap.cpython-312.pyc | Bin 3328 -> 0 bytes .../rich/__pycache__/abc.cpython-312.pyc | Bin 1610 -> 0 bytes .../rich/__pycache__/align.cpython-312.pyc | Bin 12289 -> 0 bytes .../rich/__pycache__/ansi.cpython-312.pyc | Bin 9068 -> 0 bytes .../rich/__pycache__/bar.cpython-312.pyc | Bin 4274 -> 0 bytes .../rich/__pycache__/box.cpython-312.pyc | Bin 11840 -> 0 bytes .../rich/__pycache__/cells.cpython-312.pyc | Bin 5812 -> 0 bytes .../rich/__pycache__/color.cpython-312.pyc | Bin 26571 -> 0 bytes .../__pycache__/color_triplet.cpython-312.pyc | Bin 1703 -> 0 bytes .../rich/__pycache__/columns.cpython-312.pyc | Bin 8586 -> 0 bytes .../rich/__pycache__/console.cpython-312.pyc | Bin 113443 -> 0 bytes .../__pycache__/constrain.cpython-312.pyc | Bin 2260 -> 0 bytes .../__pycache__/containers.cpython-312.pyc | Bin 9212 -> 0 bytes .../rich/__pycache__/control.cpython-312.pyc | Bin 10943 -> 0 bytes .../default_styles.cpython-312.pyc | Bin 10368 -> 0 bytes .../rich/__pycache__/diagnose.cpython-312.pyc | Bin 1490 -> 0 bytes .../rich/__pycache__/emoji.cpython-312.pyc | Bin 4214 -> 0 bytes .../rich/__pycache__/errors.cpython-312.pyc | Bin 1847 -> 0 bytes .../__pycache__/file_proxy.cpython-312.pyc | Bin 3573 -> 0 bytes .../rich/__pycache__/filesize.cpython-312.pyc | Bin 3074 -> 0 bytes .../__pycache__/highlighter.cpython-312.pyc | Bin 9889 -> 0 bytes .../rich/__pycache__/json.cpython-312.pyc | Bin 6037 -> 0 bytes .../rich/__pycache__/jupyter.cpython-312.pyc | Bin 5210 -> 0 bytes .../rich/__pycache__/layout.cpython-312.pyc | Bin 20160 -> 0 bytes .../rich/__pycache__/live.cpython-312.pyc | Bin 19017 -> 0 bytes .../__pycache__/live_render.cpython-312.pyc | Bin 4891 -> 0 bytes .../rich/__pycache__/logging.cpython-312.pyc | Bin 13556 -> 0 bytes .../rich/__pycache__/markup.cpython-312.pyc | Bin 9569 -> 0 bytes .../rich/__pycache__/measure.cpython-312.pyc | Bin 6380 -> 0 bytes .../rich/__pycache__/padding.cpython-312.pyc | Bin 7126 -> 0 bytes .../rich/__pycache__/pager.cpython-312.pyc | Bin 1813 -> 0 bytes .../rich/__pycache__/palette.cpython-312.pyc | Bin 5299 -> 0 bytes .../rich/__pycache__/panel.cpython-312.pyc | Bin 12185 -> 0 bytes .../rich/__pycache__/pretty.cpython-312.pyc | Bin 40149 -> 0 bytes .../rich/__pycache__/progress.cpython-312.pyc | Bin 74975 -> 0 bytes .../__pycache__/progress_bar.cpython-312.pyc | Bin 10379 -> 0 bytes .../rich/__pycache__/prompt.cpython-312.pyc | Bin 14789 -> 0 bytes .../rich/__pycache__/protocol.cpython-312.pyc | Bin 1794 -> 0 bytes .../rich/__pycache__/region.cpython-312.pyc | Bin 569 -> 0 bytes .../rich/__pycache__/repr.cpython-312.pyc | Bin 6615 -> 0 bytes .../rich/__pycache__/rule.cpython-312.pyc | Bin 6570 -> 0 bytes .../rich/__pycache__/scope.cpython-312.pyc | Bin 3827 -> 0 bytes .../rich/__pycache__/screen.cpython-312.pyc | Bin 2481 -> 0 bytes .../rich/__pycache__/segment.cpython-312.pyc | Bin 28117 -> 0 bytes .../rich/__pycache__/spinner.cpython-312.pyc | Bin 6066 -> 0 bytes .../rich/__pycache__/status.cpython-312.pyc | Bin 6063 -> 0 bytes .../rich/__pycache__/style.cpython-312.pyc | Bin 33502 -> 0 bytes .../rich/__pycache__/styled.cpython-312.pyc | Bin 2141 -> 0 bytes .../rich/__pycache__/syntax.cpython-312.pyc | Bin 39950 -> 0 bytes .../rich/__pycache__/table.cpython-312.pyc | Bin 43541 -> 0 bytes .../terminal_theme.cpython-312.pyc | Bin 3350 -> 0 bytes .../rich/__pycache__/text.cpython-312.pyc | Bin 60851 -> 0 bytes .../rich/__pycache__/theme.cpython-312.pyc | Bin 6344 -> 0 bytes .../rich/__pycache__/themes.cpython-312.pyc | Bin 316 -> 0 bytes .../__pycache__/traceback.cpython-312.pyc | Bin 31513 -> 0 bytes .../rich/__pycache__/tree.cpython-312.pyc | Bin 11438 -> 0 bytes .../pip/_vendor/rich/_cell_widths.py | 454 - .../pip/_vendor/rich/_emoji_codes.py | 3610 ------- .../pip/_vendor/rich/_emoji_replace.py | 32 - .../pip/_vendor/rich/_export_format.py | 76 - .../pip/_vendor/rich/_extension.py | 10 - .../site-packages/pip/_vendor/rich/_fileno.py | 24 - .../pip/_vendor/rich/_inspect.py | 270 - .../pip/_vendor/rich/_log_render.py | 94 - .../site-packages/pip/_vendor/rich/_loop.py | 43 - .../pip/_vendor/rich/_null_file.py | 69 - .../pip/_vendor/rich/_palettes.py | 309 - .../site-packages/pip/_vendor/rich/_pick.py | 17 - .../site-packages/pip/_vendor/rich/_ratio.py | 159 - .../pip/_vendor/rich/_spinners.py | 482 - .../site-packages/pip/_vendor/rich/_stack.py | 16 - .../site-packages/pip/_vendor/rich/_timer.py | 19 - .../pip/_vendor/rich/_win32_console.py | 662 -- .../pip/_vendor/rich/_windows.py | 71 - .../pip/_vendor/rich/_windows_renderer.py | 56 - .../site-packages/pip/_vendor/rich/_wrap.py | 93 - .../Lib/site-packages/pip/_vendor/rich/abc.py | 33 - .../site-packages/pip/_vendor/rich/align.py | 311 - .../site-packages/pip/_vendor/rich/ansi.py | 240 - .../Lib/site-packages/pip/_vendor/rich/bar.py | 93 - .../Lib/site-packages/pip/_vendor/rich/box.py | 480 - .../site-packages/pip/_vendor/rich/cells.py | 167 - .../site-packages/pip/_vendor/rich/color.py | 621 -- .../pip/_vendor/rich/color_triplet.py | 38 - .../site-packages/pip/_vendor/rich/columns.py | 187 - .../site-packages/pip/_vendor/rich/console.py | 2633 ----- .../pip/_vendor/rich/constrain.py | 37 - .../pip/_vendor/rich/containers.py | 167 - .../site-packages/pip/_vendor/rich/control.py | 225 - .../pip/_vendor/rich/default_styles.py | 190 - .../pip/_vendor/rich/diagnose.py | 37 - .../site-packages/pip/_vendor/rich/emoji.py | 96 - .../site-packages/pip/_vendor/rich/errors.py | 34 - .../pip/_vendor/rich/file_proxy.py | 57 - .../pip/_vendor/rich/filesize.py | 89 - .../pip/_vendor/rich/highlighter.py | 232 - .../site-packages/pip/_vendor/rich/json.py | 139 - .../site-packages/pip/_vendor/rich/jupyter.py | 101 - .../site-packages/pip/_vendor/rich/layout.py | 442 - .../site-packages/pip/_vendor/rich/live.py | 375 - .../pip/_vendor/rich/live_render.py | 112 - .../site-packages/pip/_vendor/rich/logging.py | 289 - .../site-packages/pip/_vendor/rich/markup.py | 251 - .../site-packages/pip/_vendor/rich/measure.py | 151 - .../site-packages/pip/_vendor/rich/padding.py | 141 - .../site-packages/pip/_vendor/rich/pager.py | 34 - .../site-packages/pip/_vendor/rich/palette.py | 100 - .../site-packages/pip/_vendor/rich/panel.py | 312 - .../site-packages/pip/_vendor/rich/pretty.py | 995 -- .../pip/_vendor/rich/progress.py | 1699 ---- .../pip/_vendor/rich/progress_bar.py | 223 - .../site-packages/pip/_vendor/rich/prompt.py | 375 - .../pip/_vendor/rich/protocol.py | 42 - .../site-packages/pip/_vendor/rich/py.typed | 0 .../site-packages/pip/_vendor/rich/region.py | 10 - .../site-packages/pip/_vendor/rich/repr.py | 149 - .../site-packages/pip/_vendor/rich/rule.py | 130 - .../site-packages/pip/_vendor/rich/scope.py | 86 - .../site-packages/pip/_vendor/rich/screen.py | 54 - .../site-packages/pip/_vendor/rich/segment.py | 738 -- .../site-packages/pip/_vendor/rich/spinner.py | 137 - .../site-packages/pip/_vendor/rich/status.py | 131 - .../site-packages/pip/_vendor/rich/style.py | 796 -- .../site-packages/pip/_vendor/rich/styled.py | 42 - .../site-packages/pip/_vendor/rich/syntax.py | 958 -- .../site-packages/pip/_vendor/rich/table.py | 1000 -- .../pip/_vendor/rich/terminal_theme.py | 153 - .../site-packages/pip/_vendor/rich/text.py | 1357 --- .../site-packages/pip/_vendor/rich/theme.py | 115 - .../site-packages/pip/_vendor/rich/themes.py | 5 - .../pip/_vendor/rich/traceback.py | 753 -- .../site-packages/pip/_vendor/rich/tree.py | 249 - .../pip/_vendor/tomli/__init__.py | 11 - .../__pycache__/__init__.cpython-312.pyc | Bin 386 -> 0 bytes .../tomli/__pycache__/_parser.cpython-312.pyc | Bin 26901 -> 0 bytes .../tomli/__pycache__/_re.cpython-312.pyc | Bin 3910 -> 0 bytes .../tomli/__pycache__/_types.cpython-312.pyc | Bin 368 -> 0 bytes .../pip/_vendor/tomli/_parser.py | 691 -- .../site-packages/pip/_vendor/tomli/_re.py | 107 - .../site-packages/pip/_vendor/tomli/_types.py | 10 - .../site-packages/pip/_vendor/tomli/py.typed | 1 - .../pip/_vendor/truststore/__init__.py | 13 - .../__pycache__/__init__.cpython-312.pyc | Bin 620 -> 0 bytes .../__pycache__/_api.cpython-312.pyc | Bin 16542 -> 0 bytes .../__pycache__/_macos.cpython-312.pyc | Bin 16574 -> 0 bytes .../__pycache__/_openssl.cpython-312.pyc | Bin 2207 -> 0 bytes .../_ssl_constants.cpython-312.pyc | Bin 1101 -> 0 bytes .../__pycache__/_windows.cpython-312.pyc | Bin 15742 -> 0 bytes .../pip/_vendor/truststore/_api.py | 313 - .../pip/_vendor/truststore/_macos.py | 499 - .../pip/_vendor/truststore/_openssl.py | 66 - .../pip/_vendor/truststore/_ssl_constants.py | 31 - .../pip/_vendor/truststore/_windows.py | 564 -- .../pip/_vendor/truststore/py.typed | 0 .../pip/_vendor/typing_extensions.py | 3641 ------- .../pip/_vendor/urllib3/__init__.py | 102 - .../__pycache__/__init__.cpython-312.pyc | Bin 3407 -> 0 bytes .../__pycache__/_collections.cpython-312.pyc | Bin 16381 -> 0 bytes .../__pycache__/_version.cpython-312.pyc | Bin 220 -> 0 bytes .../__pycache__/connection.cpython-312.pyc | Bin 20405 -> 0 bytes .../connectionpool.cpython-312.pyc | Bin 36439 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 13495 -> 0 bytes .../__pycache__/fields.cpython-312.pyc | Bin 10404 -> 0 bytes .../__pycache__/filepost.cpython-312.pyc | Bin 4014 -> 0 bytes .../__pycache__/poolmanager.cpython-312.pyc | Bin 20434 -> 0 bytes .../__pycache__/request.cpython-312.pyc | Bin 7296 -> 0 bytes .../__pycache__/response.cpython-312.pyc | Bin 33949 -> 0 bytes .../pip/_vendor/urllib3/_collections.py | 355 - .../pip/_vendor/urllib3/_version.py | 2 - .../pip/_vendor/urllib3/connection.py | 572 -- .../pip/_vendor/urllib3/connectionpool.py | 1137 --- .../pip/_vendor/urllib3/contrib/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 200 -> 0 bytes .../_appengine_environ.cpython-312.pyc | Bin 1850 -> 0 bytes .../__pycache__/appengine.cpython-312.pyc | Bin 11566 -> 0 bytes .../__pycache__/ntlmpool.cpython-312.pyc | Bin 5716 -> 0 bytes .../__pycache__/pyopenssl.cpython-312.pyc | Bin 24450 -> 0 bytes .../securetransport.cpython-312.pyc | Bin 35507 -> 0 bytes .../contrib/__pycache__/socks.cpython-312.pyc | Bin 7513 -> 0 bytes .../urllib3/contrib/_appengine_environ.py | 36 - .../contrib/_securetransport/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 217 -> 0 bytes .../__pycache__/bindings.cpython-312.pyc | Bin 17429 -> 0 bytes .../__pycache__/low_level.cpython-312.pyc | Bin 14765 -> 0 bytes .../contrib/_securetransport/bindings.py | 519 - .../contrib/_securetransport/low_level.py | 397 - .../pip/_vendor/urllib3/contrib/appengine.py | 314 - .../pip/_vendor/urllib3/contrib/ntlmpool.py | 130 - .../pip/_vendor/urllib3/contrib/pyopenssl.py | 518 - .../urllib3/contrib/securetransport.py | 920 -- .../pip/_vendor/urllib3/contrib/socks.py | 216 - .../pip/_vendor/urllib3/exceptions.py | 323 - .../pip/_vendor/urllib3/fields.py | 274 - .../pip/_vendor/urllib3/filepost.py | 98 - .../pip/_vendor/urllib3/packages/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 201 -> 0 bytes .../packages/__pycache__/six.cpython-312.pyc | Bin 41257 -> 0 bytes .../urllib3/packages/backports/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 211 -> 0 bytes .../__pycache__/makefile.cpython-312.pyc | Bin 1827 -> 0 bytes .../weakref_finalize.cpython-312.pyc | Bin 7338 -> 0 bytes .../urllib3/packages/backports/makefile.py | 51 - .../packages/backports/weakref_finalize.py | 155 - .../pip/_vendor/urllib3/packages/six.py | 1076 -- .../pip/_vendor/urllib3/poolmanager.py | 540 - .../pip/_vendor/urllib3/request.py | 191 - .../pip/_vendor/urllib3/response.py | 879 -- .../pip/_vendor/urllib3/util/__init__.py | 49 - .../util/__pycache__/__init__.cpython-312.pyc | Bin 1148 -> 0 bytes .../__pycache__/connection.cpython-312.pyc | Bin 4749 -> 0 bytes .../util/__pycache__/proxy.cpython-312.pyc | Bin 1554 -> 0 bytes .../util/__pycache__/queue.cpython-312.pyc | Bin 1354 -> 0 bytes .../util/__pycache__/request.cpython-312.pyc | Bin 4185 -> 0 bytes .../util/__pycache__/response.cpython-312.pyc | Bin 2994 -> 0 bytes .../util/__pycache__/retry.cpython-312.pyc | Bin 21703 -> 0 bytes .../util/__pycache__/ssl_.cpython-312.pyc | Bin 15088 -> 0 bytes .../ssl_match_hostname.cpython-312.pyc | Bin 5053 -> 0 bytes .../__pycache__/ssltransport.cpython-312.pyc | Bin 10755 -> 0 bytes .../util/__pycache__/timeout.cpython-312.pyc | Bin 11141 -> 0 bytes .../util/__pycache__/url.cpython-312.pyc | Bin 15787 -> 0 bytes .../util/__pycache__/wait.cpython-312.pyc | Bin 4405 -> 0 bytes .../pip/_vendor/urllib3/util/connection.py | 149 - .../pip/_vendor/urllib3/util/proxy.py | 57 - .../pip/_vendor/urllib3/util/queue.py | 22 - .../pip/_vendor/urllib3/util/request.py | 137 - .../pip/_vendor/urllib3/util/response.py | 107 - .../pip/_vendor/urllib3/util/retry.py | 620 -- .../pip/_vendor/urllib3/util/ssl_.py | 495 - .../urllib3/util/ssl_match_hostname.py | 159 - .../pip/_vendor/urllib3/util/ssltransport.py | 221 - .../pip/_vendor/urllib3/util/timeout.py | 271 - .../pip/_vendor/urllib3/util/url.py | 435 - .../pip/_vendor/urllib3/util/wait.py | 152 - .env/Lib/site-packages/pip/_vendor/vendor.txt | 18 - .env/Lib/site-packages/pip/py.typed | 4 - .../pydantic-2.12.5.dist-info/INSTALLER | 1 - .../pydantic-2.12.5.dist-info/METADATA | 1029 -- .../pydantic-2.12.5.dist-info/RECORD | 217 - .../pydantic-2.12.5.dist-info/WHEEL | 4 - .../licenses/LICENSE | 21 - .env/Lib/site-packages/pydantic/__init__.py | 456 - .../__pycache__/__init__.cpython-312.pyc | Bin 14421 -> 0 bytes .../__pycache__/_migration.cpython-312.pyc | Bin 11028 -> 0 bytes .../alias_generators.cpython-312.pyc | Bin 3278 -> 0 bytes .../__pycache__/aliases.cpython-312.pyc | Bin 6558 -> 0 bytes .../annotated_handlers.cpython-312.pyc | Bin 5492 -> 0 bytes .../class_validators.cpython-312.pyc | Bin 366 -> 0 bytes .../__pycache__/color.cpython-312.pyc | Bin 30172 -> 0 bytes .../__pycache__/config.cpython-312.pyc | Bin 7498 -> 0 bytes .../__pycache__/dataclasses.cpython-312.pyc | Bin 17325 -> 0 bytes .../datetime_parse.cpython-312.pyc | Bin 366 -> 0 bytes .../__pycache__/decorator.cpython-312.pyc | Bin 356 -> 0 bytes .../__pycache__/env_settings.cpython-312.pyc | Bin 362 -> 0 bytes .../error_wrappers.cpython-312.pyc | Bin 366 -> 0 bytes .../__pycache__/errors.cpython-312.pyc | Bin 7627 -> 0 bytes .../__pycache__/fields.cpython-312.pyc | Bin 72923 -> 0 bytes .../functional_serializers.cpython-312.pyc | Bin 17929 -> 0 bytes .../functional_validators.cpython-312.pyc | Bin 34740 -> 0 bytes .../__pycache__/generics.cpython-312.pyc | Bin 354 -> 0 bytes .../pydantic/__pycache__/json.cpython-312.pyc | Bin 346 -> 0 bytes .../__pycache__/json_schema.cpython-312.pyc | Bin 120009 -> 0 bytes .../pydantic/__pycache__/main.cpython-312.pyc | Bin 77306 -> 0 bytes .../pydantic/__pycache__/mypy.cpython-312.pyc | Bin 61639 -> 0 bytes .../__pycache__/networks.cpython-312.pyc | Bin 50215 -> 0 bytes .../__pycache__/parse.cpython-312.pyc | Bin 348 -> 0 bytes .../__pycache__/root_model.cpython-312.pyc | Bin 7809 -> 0 bytes .../__pycache__/schema.cpython-312.pyc | Bin 350 -> 0 bytes .../__pycache__/tools.cpython-312.pyc | Bin 348 -> 0 bytes .../__pycache__/type_adapter.cpython-312.pyc | Bin 35545 -> 0 bytes .../__pycache__/types.cpython-312.pyc | Bin 96766 -> 0 bytes .../__pycache__/typing.cpython-312.pyc | Bin 346 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 348 -> 0 bytes .../validate_call_decorator.cpython-312.pyc | Bin 5444 -> 0 bytes .../__pycache__/validators.cpython-312.pyc | Bin 358 -> 0 bytes .../__pycache__/version.cpython-312.pyc | Bin 4794 -> 0 bytes .../__pycache__/warnings.cpython-312.pyc | Bin 7210 -> 0 bytes .../pydantic/_internal/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 191 -> 0 bytes .../__pycache__/_config.cpython-312.pyc | Bin 15670 -> 0 bytes .../_core_metadata.cpython-312.pyc | Bin 4611 -> 0 bytes .../__pycache__/_core_utils.cpython-312.pyc | Bin 7708 -> 0 bytes .../__pycache__/_dataclasses.cpython-312.pyc | Bin 13399 -> 0 bytes .../__pycache__/_decorators.cpython-312.pyc | Bin 36754 -> 0 bytes .../_decorators_v1.cpython-312.pyc | Bin 8611 -> 0 bytes .../_discriminated_union.cpython-312.pyc | Bin 20842 -> 0 bytes .../_docs_extraction.cpython-312.pyc | Bin 5304 -> 0 bytes .../__pycache__/_fields.cpython-312.pyc | Bin 24286 -> 0 bytes .../__pycache__/_forward_ref.cpython-312.pyc | Bin 1300 -> 0 bytes .../_generate_schema.cpython-312.pyc | Bin 130414 -> 0 bytes .../__pycache__/_generics.cpython-312.pyc | Bin 24327 -> 0 bytes .../__pycache__/_git.cpython-312.pyc | Bin 1506 -> 0 bytes .../__pycache__/_import_utils.cpython-312.pyc | Bin 819 -> 0 bytes .../_internal_dataclass.cpython-312.pyc | Bin 340 -> 0 bytes .../_known_annotated_metadata.cpython-312.pyc | Bin 14222 -> 0 bytes .../__pycache__/_mock_val_ser.cpython-312.pyc | Bin 11089 -> 0 bytes .../_model_construction.cpython-312.pyc | Bin 35101 -> 0 bytes .../_namespace_utils.cpython-312.pyc | Bin 12309 -> 0 bytes .../__pycache__/_repr.cpython-312.pyc | Bin 7849 -> 0 bytes .../_schema_gather.cpython-312.pyc | Bin 7636 -> 0 bytes .../_schema_generation_shared.cpython-312.pyc | Bin 6217 -> 0 bytes .../__pycache__/_serializers.cpython-312.pyc | Bin 2107 -> 0 bytes .../__pycache__/_signature.cpython-312.pyc | Bin 6757 -> 0 bytes .../__pycache__/_typing_extra.cpython-312.pyc | Bin 28176 -> 0 bytes .../__pycache__/_utils.cpython-312.pyc | Bin 19764 -> 0 bytes .../_validate_call.cpython-312.pyc | Bin 6978 -> 0 bytes .../__pycache__/_validators.cpython-312.pyc | Bin 22957 -> 0 bytes .../pydantic/_internal/_config.py | 383 - .../pydantic/_internal/_core_metadata.py | 97 - .../pydantic/_internal/_core_utils.py | 174 - .../pydantic/_internal/_dataclasses.py | 315 - .../pydantic/_internal/_decorators.py | 858 -- .../pydantic/_internal/_decorators_v1.py | 174 - .../_internal/_discriminated_union.py | 479 - .../pydantic/_internal/_docs_extraction.py | 113 - .../pydantic/_internal/_fields.py | 635 -- .../pydantic/_internal/_forward_ref.py | 23 - .../pydantic/_internal/_generate_schema.py | 2867 ------ .../pydantic/_internal/_generics.py | 543 - .../site-packages/pydantic/_internal/_git.py | 27 - .../pydantic/_internal/_import_utils.py | 20 - .../pydantic/_internal/_internal_dataclass.py | 7 - .../_internal/_known_annotated_metadata.py | 401 - .../pydantic/_internal/_mock_val_ser.py | 228 - .../pydantic/_internal/_model_construction.py | 848 -- .../pydantic/_internal/_namespace_utils.py | 293 - .../site-packages/pydantic/_internal/_repr.py | 124 - .../pydantic/_internal/_schema_gather.py | 209 - .../_internal/_schema_generation_shared.py | 125 - .../pydantic/_internal/_serializers.py | 53 - .../pydantic/_internal/_signature.py | 188 - .../pydantic/_internal/_typing_extra.py | 760 -- .../pydantic/_internal/_utils.py | 446 - .../pydantic/_internal/_validate_call.py | 140 - .../pydantic/_internal/_validators.py | 533 - .env/Lib/site-packages/pydantic/_migration.py | 316 - .../pydantic/alias_generators.py | 62 - .env/Lib/site-packages/pydantic/aliases.py | 135 - .../pydantic/annotated_handlers.py | 122 - .../pydantic/class_validators.py | 5 - .env/Lib/site-packages/pydantic/color.py | 604 -- .env/Lib/site-packages/pydantic/config.py | 1288 --- .../Lib/site-packages/pydantic/dataclasses.py | 413 - .../site-packages/pydantic/datetime_parse.py | 5 - .env/Lib/site-packages/pydantic/decorator.py | 5 - .../pydantic/deprecated/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 192 -> 0 bytes .../class_validators.cpython-312.pyc | Bin 11750 -> 0 bytes .../__pycache__/config.cpython-312.pyc | Bin 4069 -> 0 bytes .../copy_internals.cpython-312.pyc | Bin 8656 -> 0 bytes .../__pycache__/decorator.cpython-312.pyc | Bin 14020 -> 0 bytes .../__pycache__/json.cpython-312.pyc | Bin 6168 -> 0 bytes .../__pycache__/parse.cpython-312.pyc | Bin 3398 -> 0 bytes .../__pycache__/tools.cpython-312.pyc | Bin 3534 -> 0 bytes .../pydantic/deprecated/class_validators.py | 256 - .../pydantic/deprecated/config.py | 72 - .../pydantic/deprecated/copy_internals.py | 224 - .../pydantic/deprecated/decorator.py | 284 - .../site-packages/pydantic/deprecated/json.py | 141 - .../pydantic/deprecated/parse.py | 80 - .../pydantic/deprecated/tools.py | 103 - .../site-packages/pydantic/env_settings.py | 5 - .../site-packages/pydantic/error_wrappers.py | 5 - .env/Lib/site-packages/pydantic/errors.py | 189 - .../pydantic/experimental/__init__.py | 1 - .../__pycache__/__init__.cpython-312.pyc | Bin 306 -> 0 bytes .../arguments_schema.cpython-312.pyc | Bin 2398 -> 0 bytes .../missing_sentinel.cpython-312.pyc | Bin 347 -> 0 bytes .../__pycache__/pipeline.cpython-312.pyc | Bin 34455 -> 0 bytes .../pydantic/experimental/arguments_schema.py | 44 - .../pydantic/experimental/missing_sentinel.py | 5 - .../pydantic/experimental/pipeline.py | 654 -- .env/Lib/site-packages/pydantic/fields.py | 1834 ---- .../pydantic/functional_serializers.py | 451 - .../pydantic/functional_validators.py | 893 -- .env/Lib/site-packages/pydantic/generics.py | 5 - .env/Lib/site-packages/pydantic/json.py | 5 - .../Lib/site-packages/pydantic/json_schema.py | 2854 ------ .env/Lib/site-packages/pydantic/main.py | 1819 ---- .env/Lib/site-packages/pydantic/mypy.py | 1374 --- .env/Lib/site-packages/pydantic/networks.py | 1331 --- .env/Lib/site-packages/pydantic/parse.py | 5 - .../site-packages/pydantic/plugin/__init__.py | 193 - .../__pycache__/__init__.cpython-312.pyc | Bin 8934 -> 0 bytes .../__pycache__/_loader.cpython-312.pyc | Bin 2424 -> 0 bytes .../_schema_validator.cpython-312.pyc | Bin 6912 -> 0 bytes .../site-packages/pydantic/plugin/_loader.py | 58 - .../pydantic/plugin/_schema_validator.py | 140 - .env/Lib/site-packages/pydantic/py.typed | 0 .env/Lib/site-packages/pydantic/root_model.py | 155 - .env/Lib/site-packages/pydantic/schema.py | 5 - .env/Lib/site-packages/pydantic/tools.py | 5 - .../site-packages/pydantic/type_adapter.py | 795 -- .env/Lib/site-packages/pydantic/types.py | 3295 ------ .env/Lib/site-packages/pydantic/typing.py | 5 - .env/Lib/site-packages/pydantic/utils.py | 5 - .../Lib/site-packages/pydantic/v1/__init__.py | 142 - .../v1/__pycache__/__init__.cpython-312.pyc | Bin 2513 -> 0 bytes .../_hypothesis_plugin.cpython-312.pyc | Bin 20502 -> 0 bytes .../annotated_types.cpython-312.pyc | Bin 3879 -> 0 bytes .../class_validators.cpython-312.pyc | Bin 19639 -> 0 bytes .../v1/__pycache__/color.cpython-312.pyc | Bin 25845 -> 0 bytes .../v1/__pycache__/config.cpython-312.pyc | Bin 8394 -> 0 bytes .../__pycache__/dataclasses.cpython-312.pyc | Bin 22750 -> 0 bytes .../datetime_parse.cpython-312.pyc | Bin 10347 -> 0 bytes .../v1/__pycache__/decorator.cpython-312.pyc | Bin 13912 -> 0 bytes .../__pycache__/env_settings.cpython-312.pyc | Bin 17714 -> 0 bytes .../error_wrappers.cpython-312.pyc | Bin 8922 -> 0 bytes .../v1/__pycache__/errors.cpython-312.pyc | Bin 29597 -> 0 bytes .../v1/__pycache__/fields.cpython-312.pyc | Bin 57437 -> 0 bytes .../v1/__pycache__/generics.cpython-312.pyc | Bin 16987 -> 0 bytes .../v1/__pycache__/json.cpython-312.pyc | Bin 5205 -> 0 bytes .../v1/__pycache__/main.cpython-312.pyc | Bin 48211 -> 0 bytes .../v1/__pycache__/mypy.cpython-312.pyc | Bin 46355 -> 0 bytes .../v1/__pycache__/networks.cpython-312.pyc | Bin 29544 -> 0 bytes .../v1/__pycache__/parse.cpython-312.pyc | Bin 2742 -> 0 bytes .../v1/__pycache__/schema.cpython-312.pyc | Bin 48470 -> 0 bytes .../v1/__pycache__/tools.cpython-312.pyc | Bin 3875 -> 0 bytes .../v1/__pycache__/types.cpython-312.pyc | Bin 48482 -> 0 bytes .../v1/__pycache__/typing.cpython-312.pyc | Bin 22625 -> 0 bytes .../v1/__pycache__/utils.cpython-312.pyc | Bin 35232 -> 0 bytes .../v1/__pycache__/validators.cpython-312.pyc | Bin 30890 -> 0 bytes .../v1/__pycache__/version.cpython-312.pyc | Bin 1953 -> 0 bytes .../pydantic/v1/_hypothesis_plugin.py | 391 - .../pydantic/v1/annotated_types.py | 72 - .../pydantic/v1/class_validators.py | 361 - .env/Lib/site-packages/pydantic/v1/color.py | 494 - .env/Lib/site-packages/pydantic/v1/config.py | 191 - .../site-packages/pydantic/v1/dataclasses.py | 500 - .../pydantic/v1/datetime_parse.py | 248 - .../site-packages/pydantic/v1/decorator.py | 264 - .../site-packages/pydantic/v1/env_settings.py | 350 - .../pydantic/v1/error_wrappers.py | 161 - .env/Lib/site-packages/pydantic/v1/errors.py | 646 -- .env/Lib/site-packages/pydantic/v1/fields.py | 1253 --- .../Lib/site-packages/pydantic/v1/generics.py | 400 - .env/Lib/site-packages/pydantic/v1/json.py | 112 - .env/Lib/site-packages/pydantic/v1/main.py | 1113 --- .env/Lib/site-packages/pydantic/v1/mypy.py | 949 -- .../Lib/site-packages/pydantic/v1/networks.py | 747 -- .env/Lib/site-packages/pydantic/v1/parse.py | 66 - .env/Lib/site-packages/pydantic/v1/py.typed | 0 .env/Lib/site-packages/pydantic/v1/schema.py | 1163 --- .env/Lib/site-packages/pydantic/v1/tools.py | 92 - .env/Lib/site-packages/pydantic/v1/types.py | 1205 --- .env/Lib/site-packages/pydantic/v1/typing.py | 614 -- .env/Lib/site-packages/pydantic/v1/utils.py | 806 -- .../site-packages/pydantic/v1/validators.py | 768 -- .env/Lib/site-packages/pydantic/v1/version.py | 38 - .../pydantic/validate_call_decorator.py | 116 - .env/Lib/site-packages/pydantic/validators.py | 5 - .env/Lib/site-packages/pydantic/version.py | 113 - .env/Lib/site-packages/pydantic/warnings.py | 122 - .../pydantic_core-2.41.5.dist-info/INSTALLER | 1 - .../pydantic_core-2.41.5.dist-info/METADATA | 180 - .../pydantic_core-2.41.5.dist-info/RECORD | 12 - .../pydantic_core-2.41.5.dist-info/WHEEL | 4 - .../licenses/LICENSE | 21 - .../site-packages/pydantic_core/__init__.py | 171 - .../__pycache__/__init__.cpython-312.pyc | Bin 3147 -> 0 bytes .../__pycache__/core_schema.cpython-312.pyc | Bin 155153 -> 0 bytes .../_pydantic_core.cp312-win_amd64.pyd | Bin 5436928 -> 0 bytes .../pydantic_core/_pydantic_core.pyi | 1046 -- .../pydantic_core/core_schema.py | 4435 --------- .env/Lib/site-packages/pydantic_core/py.typed | 0 .../sqlalchemy-2.0.46.dist-info/INSTALLER | 1 - .../sqlalchemy-2.0.46.dist-info/METADATA | 243 - .../sqlalchemy-2.0.46.dist-info/RECORD | 532 - .../sqlalchemy-2.0.46.dist-info/REQUESTED | 0 .../sqlalchemy-2.0.46.dist-info/WHEEL | 5 - .../licenses/LICENSE | 19 - .../sqlalchemy-2.0.46.dist-info/top_level.txt | 1 - .env/Lib/site-packages/sqlalchemy/__init__.py | 283 - .../__pycache__/__init__.cpython-312.pyc | Bin 9419 -> 0 bytes .../__pycache__/events.cpython-312.pyc | Bin 556 -> 0 bytes .../__pycache__/exc.cpython-312.pyc | Bin 31227 -> 0 bytes .../__pycache__/inspection.cpython-312.pyc | Bin 6655 -> 0 bytes .../__pycache__/log.cpython-312.pyc | Bin 11613 -> 0 bytes .../__pycache__/schema.cpython-312.pyc | Bin 2371 -> 0 bytes .../__pycache__/types.cpython-312.pyc | Bin 2281 -> 0 bytes .../sqlalchemy/connectors/__init__.py | 18 - .../__pycache__/__init__.cpython-312.pyc | Bin 595 -> 0 bytes .../__pycache__/aioodbc.cpython-312.pyc | Bin 7837 -> 0 bytes .../__pycache__/asyncio.cpython-312.pyc | Bin 21707 -> 0 bytes .../__pycache__/pyodbc.cpython-312.pyc | Bin 9630 -> 0 bytes .../sqlalchemy/connectors/aioodbc.py | 184 - .../sqlalchemy/connectors/asyncio.py | 429 - .../sqlalchemy/connectors/pyodbc.py | 250 - .../sqlalchemy/cyextension/__init__.py | 6 - .../__pycache__/__init__.cpython-312.pyc | Bin 195 -> 0 bytes .../collections.cp312-win_amd64.pyd | Bin 163840 -> 0 bytes .../sqlalchemy/cyextension/collections.pyx | 409 - .../immutabledict.cp312-win_amd64.pyd | Bin 68608 -> 0 bytes .../sqlalchemy/cyextension/immutabledict.pxd | 8 - .../sqlalchemy/cyextension/immutabledict.pyx | 133 - .../processors.cp312-win_amd64.pyd | Bin 58880 -> 0 bytes .../sqlalchemy/cyextension/processors.pyx | 68 - .../resultproxy.cp312-win_amd64.pyd | Bin 59392 -> 0 bytes .../sqlalchemy/cyextension/resultproxy.pyx | 102 - .../cyextension/util.cp312-win_amd64.pyd | Bin 70656 -> 0 bytes .../sqlalchemy/cyextension/util.pyx | 90 - .../sqlalchemy/dialects/__init__.py | 62 - .../__pycache__/__init__.cpython-312.pyc | Bin 1909 -> 0 bytes .../__pycache__/_typing.cpython-312.pyc | Bin 1085 -> 0 bytes .../sqlalchemy/dialects/_typing.py | 30 - .../sqlalchemy/dialects/mssql/__init__.py | 88 - .../__pycache__/__init__.cpython-312.pyc | Bin 1663 -> 0 bytes .../mssql/__pycache__/aioodbc.cpython-312.pyc | Bin 2404 -> 0 bytes .../mssql/__pycache__/base.cpython-312.pyc | Bin 153644 -> 0 bytes .../information_schema.cpython-312.pyc | Bin 8939 -> 0 bytes .../mssql/__pycache__/json.cpython-312.pyc | Bin 5225 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 8190 -> 0 bytes .../mssql/__pycache__/pymssql.cpython-312.pyc | Bin 5955 -> 0 bytes .../mssql/__pycache__/pyodbc.cpython-312.pyc | Bin 30633 -> 0 bytes .../sqlalchemy/dialects/mssql/aioodbc.py | 63 - .../sqlalchemy/dialects/mssql/base.py | 4093 -------- .../dialects/mssql/information_schema.py | 285 - .../sqlalchemy/dialects/mssql/json.py | 129 - .../sqlalchemy/dialects/mssql/provision.py | 185 - .../sqlalchemy/dialects/mssql/pymssql.py | 126 - .../sqlalchemy/dialects/mssql/pyodbc.py | 760 -- .../sqlalchemy/dialects/mysql/__init__.py | 104 - .../__pycache__/__init__.cpython-312.pyc | Bin 1992 -> 0 bytes .../__pycache__/aiomysql.cpython-312.pyc | Bin 11735 -> 0 bytes .../mysql/__pycache__/asyncmy.cpython-312.pyc | Bin 11198 -> 0 bytes .../mysql/__pycache__/base.cpython-312.pyc | Bin 155113 -> 0 bytes .../mysql/__pycache__/cymysql.cpython-312.pyc | Bin 4094 -> 0 bytes .../mysql/__pycache__/dml.cpython-312.pyc | Bin 8354 -> 0 bytes .../__pycache__/enumerated.cpython-312.pyc | Bin 11537 -> 0 bytes .../__pycache__/expression.cpython-312.pyc | Bin 5226 -> 0 bytes .../mysql/__pycache__/json.cpython-312.pyc | Bin 3945 -> 0 bytes .../mysql/__pycache__/mariadb.cpython-312.pyc | Bin 2660 -> 0 bytes .../mariadbconnector.cpython-312.pyc | Bin 13788 -> 0 bytes .../mysqlconnector.cpython-312.pyc | Bin 14207 -> 0 bytes .../mysql/__pycache__/mysqldb.cpython-312.pyc | Bin 12375 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 6374 -> 0 bytes .../mysql/__pycache__/pymysql.cpython-312.pyc | Bin 6216 -> 0 bytes .../mysql/__pycache__/pyodbc.cpython-312.pyc | Bin 6102 -> 0 bytes .../__pycache__/reflection.cpython-312.pyc | Bin 25859 -> 0 bytes .../reserved_words.cpython-312.pyc | Bin 4392 -> 0 bytes .../mysql/__pycache__/types.cpython-312.pyc | Bin 33056 -> 0 bytes .../sqlalchemy/dialects/mysql/aiomysql.py | 250 - .../sqlalchemy/dialects/mysql/asyncmy.py | 231 - .../sqlalchemy/dialects/mysql/base.py | 3943 -------- .../sqlalchemy/dialects/mysql/cymysql.py | 106 - .../sqlalchemy/dialects/mysql/dml.py | 225 - .../sqlalchemy/dialects/mysql/enumerated.py | 282 - .../sqlalchemy/dialects/mysql/expression.py | 146 - .../sqlalchemy/dialects/mysql/json.py | 91 - .../sqlalchemy/dialects/mysql/mariadb.py | 72 - .../dialects/mysql/mariadbconnector.py | 322 - .../dialects/mysql/mysqlconnector.py | 302 - .../sqlalchemy/dialects/mysql/mysqldb.py | 314 - .../sqlalchemy/dialects/mysql/provision.py | 147 - .../sqlalchemy/dialects/mysql/pymysql.py | 158 - .../sqlalchemy/dialects/mysql/pyodbc.py | 157 - .../sqlalchemy/dialects/mysql/reflection.py | 727 -- .../dialects/mysql/reserved_words.py | 570 -- .../sqlalchemy/dialects/mysql/types.py | 835 -- .../sqlalchemy/dialects/oracle/__init__.py | 81 - .../__pycache__/__init__.cpython-312.pyc | Bin 1664 -> 0 bytes .../oracle/__pycache__/base.cpython-312.pyc | Bin 153938 -> 0 bytes .../__pycache__/cx_oracle.cpython-312.pyc | Bin 60982 -> 0 bytes .../__pycache__/dictionary.cpython-312.pyc | Bin 24599 -> 0 bytes .../__pycache__/oracledb.cpython-312.pyc | Bin 41079 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 13198 -> 0 bytes .../oracle/__pycache__/types.cpython-312.pyc | Bin 13213 -> 0 bytes .../oracle/__pycache__/vector.cpython-312.pyc | Bin 12222 -> 0 bytes .../sqlalchemy/dialects/oracle/base.py | 3802 ------- .../sqlalchemy/dialects/oracle/cx_oracle.py | 1555 --- .../sqlalchemy/dialects/oracle/dictionary.py | 507 - .../sqlalchemy/dialects/oracle/oracledb.py | 941 -- .../sqlalchemy/dialects/oracle/provision.py | 297 - .../sqlalchemy/dialects/oracle/types.py | 316 - .../sqlalchemy/dialects/oracle/vector.py | 365 - .../dialects/postgresql/__init__.py | 167 - .../__pycache__/__init__.cpython-312.pyc | Bin 3433 -> 0 bytes .../_psycopg_common.cpython-312.pyc | Bin 7926 -> 0 bytes .../__pycache__/array.cpython-312.pyc | Bin 19568 -> 0 bytes .../__pycache__/asyncpg.cpython-312.pyc | Bin 58825 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 216419 -> 0 bytes .../__pycache__/dml.cpython-312.pyc | Bin 12597 -> 0 bytes .../__pycache__/ext.cpython-312.pyc | Bin 20629 -> 0 bytes .../__pycache__/hstore.cpython-312.pyc | Bin 15368 -> 0 bytes .../__pycache__/json.cpython-312.pyc | Bin 17139 -> 0 bytes .../__pycache__/named_types.cpython-312.pyc | Bin 23034 -> 0 bytes .../__pycache__/operators.cpython-312.pyc | Bin 2156 -> 0 bytes .../__pycache__/pg8000.cpython-312.pyc | Bin 30523 -> 0 bytes .../__pycache__/pg_catalog.cpython-312.pyc | Bin 11875 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 7712 -> 0 bytes .../__pycache__/psycopg.cpython-312.pyc | Bin 40797 -> 0 bytes .../__pycache__/psycopg2.cpython-312.pyc | Bin 35972 -> 0 bytes .../__pycache__/psycopg2cffi.cpython-312.pyc | Bin 2163 -> 0 bytes .../__pycache__/ranges.cpython-312.pyc | Bin 34768 -> 0 bytes .../__pycache__/types.cpython-312.pyc | Bin 11516 -> 0 bytes .../dialects/postgresql/_psycopg_common.py | 189 - .../sqlalchemy/dialects/postgresql/array.py | 519 - .../sqlalchemy/dialects/postgresql/asyncpg.py | 1284 --- .../sqlalchemy/dialects/postgresql/base.py | 5333 ---------- .../sqlalchemy/dialects/postgresql/dml.py | 339 - .../sqlalchemy/dialects/postgresql/ext.py | 540 - .../sqlalchemy/dialects/postgresql/hstore.py | 406 - .../sqlalchemy/dialects/postgresql/json.py | 404 - .../dialects/postgresql/named_types.py | 524 - .../dialects/postgresql/operators.py | 129 - .../sqlalchemy/dialects/postgresql/pg8000.py | 669 -- .../dialects/postgresql/pg_catalog.py | 326 - .../dialects/postgresql/provision.py | 175 - .../sqlalchemy/dialects/postgresql/psycopg.py | 862 -- .../dialects/postgresql/psycopg2.py | 892 -- .../dialects/postgresql/psycopg2cffi.py | 61 - .../sqlalchemy/dialects/postgresql/ranges.py | 1031 -- .../sqlalchemy/dialects/postgresql/types.py | 313 - .../sqlalchemy/dialects/sqlite/__init__.py | 57 - .../__pycache__/__init__.cpython-312.pyc | Bin 1059 -> 0 bytes .../__pycache__/aiosqlite.cpython-312.pyc | Bin 21993 -> 0 bytes .../sqlite/__pycache__/base.cpython-312.pyc | Bin 109076 -> 0 bytes .../sqlite/__pycache__/dml.cpython-312.pyc | Bin 10052 -> 0 bytes .../sqlite/__pycache__/json.cpython-312.pyc | Bin 3776 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 8731 -> 0 bytes .../__pycache__/pysqlcipher.cpython-312.pyc | Bin 6211 -> 0 bytes .../__pycache__/pysqlite.cpython-312.pyc | Bin 31212 -> 0 bytes .../sqlalchemy/dialects/sqlite/aiosqlite.py | 482 - .../sqlalchemy/dialects/sqlite/base.py | 3045 ------ .../sqlalchemy/dialects/sqlite/dml.py | 263 - .../sqlalchemy/dialects/sqlite/json.py | 92 - .../sqlalchemy/dialects/sqlite/provision.py | 223 - .../sqlalchemy/dialects/sqlite/pysqlcipher.py | 157 - .../sqlalchemy/dialects/sqlite/pysqlite.py | 756 -- .../dialects/type_migration_guidelines.txt | 145 - .../sqlalchemy/engine/__init__.py | 62 - .../__pycache__/__init__.cpython-312.pyc | Bin 2274 -> 0 bytes .../_py_processors.cpython-312.pyc | Bin 4497 -> 0 bytes .../__pycache__/_py_row.cpython-312.pyc | Bin 5766 -> 0 bytes .../__pycache__/_py_util.cpython-312.pyc | Bin 2205 -> 0 bytes .../engine/__pycache__/base.cpython-312.pyc | Bin 130233 -> 0 bytes .../characteristics.cpython-312.pyc | Bin 6841 -> 0 bytes .../engine/__pycache__/create.cpython-312.pyc | Bin 34955 -> 0 bytes .../engine/__pycache__/cursor.cpython-312.pyc | Bin 83073 -> 0 bytes .../__pycache__/default.cpython-312.pyc | Bin 89201 -> 0 bytes .../engine/__pycache__/events.cpython-312.pyc | Bin 39920 -> 0 bytes .../__pycache__/interfaces.cpython-312.pyc | Bin 102601 -> 0 bytes .../engine/__pycache__/mock.cpython-312.pyc | Bin 5671 -> 0 bytes .../__pycache__/processors.cpython-312.pyc | Bin 1288 -> 0 bytes .../__pycache__/reflection.cpython-312.pyc | Bin 80610 -> 0 bytes .../engine/__pycache__/result.cpython-312.pyc | Bin 91799 -> 0 bytes .../engine/__pycache__/row.cpython-312.pyc | Bin 17436 -> 0 bytes .../__pycache__/strategies.cpython-312.pyc | Bin 552 -> 0 bytes .../engine/__pycache__/url.cpython-312.pyc | Bin 34402 -> 0 bytes .../engine/__pycache__/util.cpython-312.pyc | Bin 6633 -> 0 bytes .../sqlalchemy/engine/_py_processors.py | 136 - .../sqlalchemy/engine/_py_row.py | 128 - .../sqlalchemy/engine/_py_util.py | 74 - .../site-packages/sqlalchemy/engine/base.py | 3382 ------- .../sqlalchemy/engine/characteristics.py | 155 - .../site-packages/sqlalchemy/engine/create.py | 893 -- .../site-packages/sqlalchemy/engine/cursor.py | 2298 ----- .../sqlalchemy/engine/default.py | 2394 ----- .../site-packages/sqlalchemy/engine/events.py | 965 -- .../sqlalchemy/engine/interfaces.py | 3471 ------- .../site-packages/sqlalchemy/engine/mock.py | 134 - .../sqlalchemy/engine/processors.py | 61 - .../sqlalchemy/engine/reflection.py | 2102 ---- .../site-packages/sqlalchemy/engine/result.py | 2399 ----- .../site-packages/sqlalchemy/engine/row.py | 400 - .../sqlalchemy/engine/strategies.py | 16 - .../site-packages/sqlalchemy/engine/url.py | 924 -- .../site-packages/sqlalchemy/engine/util.py | 167 - .../sqlalchemy/event/__init__.py | 26 - .../__pycache__/__init__.cpython-312.pyc | Bin 883 -> 0 bytes .../event/__pycache__/api.cpython-312.pyc | Bin 9070 -> 0 bytes .../event/__pycache__/attr.cpython-312.pyc | Bin 31041 -> 0 bytes .../event/__pycache__/base.cpython-312.pyc | Bin 19959 -> 0 bytes .../event/__pycache__/legacy.cpython-312.pyc | Bin 9697 -> 0 bytes .../__pycache__/registry.cpython-312.pyc | Bin 12595 -> 0 bytes .../Lib/site-packages/sqlalchemy/event/api.py | 220 - .../site-packages/sqlalchemy/event/attr.py | 676 -- .../site-packages/sqlalchemy/event/base.py | 472 - .../site-packages/sqlalchemy/event/legacy.py | 258 - .../sqlalchemy/event/registry.py | 390 - .env/Lib/site-packages/sqlalchemy/events.py | 17 - .env/Lib/site-packages/sqlalchemy/exc.py | 832 -- .../site-packages/sqlalchemy/ext/__init__.py | 11 - .../ext/__pycache__/__init__.cpython-312.pyc | Bin 350 -> 0 bytes .../associationproxy.cpython-312.pyc | Bin 87182 -> 0 bytes .../ext/__pycache__/automap.cpython-312.pyc | Bin 57398 -> 0 bytes .../ext/__pycache__/baked.cpython-312.pyc | Bin 23385 -> 0 bytes .../ext/__pycache__/compiler.cpython-312.pyc | Bin 21234 -> 0 bytes .../horizontal_shard.cpython-312.pyc | Bin 17646 -> 0 bytes .../ext/__pycache__/hybrid.cpython-312.pyc | Bin 59991 -> 0 bytes .../ext/__pycache__/indexable.cpython-312.pyc | Bin 13084 -> 0 bytes .../instrumentation.cpython-312.pyc | Bin 19866 -> 0 bytes .../ext/__pycache__/mutable.cpython-312.pyc | Bin 45832 -> 0 bytes .../__pycache__/orderinglist.cpython-312.pyc | Bin 18077 -> 0 bytes .../__pycache__/serializer.cpython-312.pyc | Bin 8045 -> 0 bytes .../sqlalchemy/ext/associationproxy.py | 2027 ---- .../sqlalchemy/ext/asyncio/__init__.py | 25 - .../__pycache__/__init__.cpython-312.pyc | Bin 973 -> 0 bytes .../asyncio/__pycache__/base.cpython-312.pyc | Bin 11351 -> 0 bytes .../__pycache__/engine.cpython-312.pyc | Bin 57508 -> 0 bytes .../asyncio/__pycache__/exc.cpython-312.pyc | Bin 1033 -> 0 bytes .../__pycache__/result.cpython-312.pyc | Bin 37676 -> 0 bytes .../__pycache__/scoping.cpython-312.pyc | Bin 56493 -> 0 bytes .../__pycache__/session.cpython-312.pyc | Bin 71602 -> 0 bytes .../sqlalchemy/ext/asyncio/base.py | 281 - .../sqlalchemy/ext/asyncio/engine.py | 1471 --- .../sqlalchemy/ext/asyncio/exc.py | 21 - .../sqlalchemy/ext/asyncio/result.py | 965 -- .../sqlalchemy/ext/asyncio/scoping.py | 1599 --- .../sqlalchemy/ext/asyncio/session.py | 1947 ---- .../site-packages/sqlalchemy/ext/automap.py | 1701 ---- .../Lib/site-packages/sqlalchemy/ext/baked.py | 570 -- .../site-packages/sqlalchemy/ext/compiler.py | 600 -- .../sqlalchemy/ext/declarative/__init__.py | 65 - .../__pycache__/__init__.cpython-312.pyc | Bin 2009 -> 0 bytes .../__pycache__/extensions.cpython-312.pyc | Bin 21232 -> 0 bytes .../sqlalchemy/ext/declarative/extensions.py | 564 -- .../sqlalchemy/ext/horizontal_shard.py | 478 - .../site-packages/sqlalchemy/ext/hybrid.py | 1535 --- .../site-packages/sqlalchemy/ext/indexable.py | 364 - .../sqlalchemy/ext/instrumentation.py | 450 - .../site-packages/sqlalchemy/ext/mutable.py | 1085 -- .../sqlalchemy/ext/mypy/__init__.py | 6 - .../mypy/__pycache__/__init__.cpython-312.pyc | Bin 192 -> 0 bytes .../mypy/__pycache__/apply.cpython-312.pyc | Bin 10510 -> 0 bytes .../__pycache__/decl_class.cpython-312.pyc | Bin 15816 -> 0 bytes .../mypy/__pycache__/infer.cpython-312.pyc | Bin 15622 -> 0 bytes .../mypy/__pycache__/names.cpython-312.pyc | Bin 11035 -> 0 bytes .../mypy/__pycache__/plugin.cpython-312.pyc | Bin 12511 -> 0 bytes .../ext/mypy/__pycache__/util.cpython-312.pyc | Bin 14006 -> 0 bytes .../sqlalchemy/ext/mypy/apply.py | 324 - .../sqlalchemy/ext/mypy/decl_class.py | 515 - .../sqlalchemy/ext/mypy/infer.py | 590 -- .../sqlalchemy/ext/mypy/names.py | 335 - .../sqlalchemy/ext/mypy/plugin.py | 303 - .../site-packages/sqlalchemy/ext/mypy/util.py | 357 - .../sqlalchemy/ext/orderinglist.py | 439 - .../sqlalchemy/ext/serializer.py | 185 - .../sqlalchemy/future/__init__.py | 16 - .../__pycache__/__init__.cpython-312.pyc | Bin 460 -> 0 bytes .../future/__pycache__/engine.cpython-312.pyc | Bin 393 -> 0 bytes .../site-packages/sqlalchemy/future/engine.py | 15 - .../site-packages/sqlalchemy/inspection.py | 174 - .env/Lib/site-packages/sqlalchemy/log.py | 288 - .../site-packages/sqlalchemy/orm/__init__.py | 171 - .../orm/__pycache__/__init__.cpython-312.pyc | Bin 6384 -> 0 bytes .../_orm_constructors.cpython-312.pyc | Bin 107037 -> 0 bytes .../orm/__pycache__/_typing.cpython-312.pyc | Bin 6810 -> 0 bytes .../__pycache__/attributes.cpython-312.pyc | Bin 100599 -> 0 bytes .../orm/__pycache__/base.cpython-312.pyc | Bin 30343 -> 0 bytes .../bulk_persistence.cpython-312.pyc | Bin 64779 -> 0 bytes .../__pycache__/clsregistry.cpython-312.pyc | Bin 23750 -> 0 bytes .../__pycache__/collections.cpython-312.pyc | Bin 61762 -> 0 bytes .../orm/__pycache__/context.cpython-312.pyc | Bin 103144 -> 0 bytes .../orm/__pycache__/decl_api.cpython-312.pyc | Bin 69715 -> 0 bytes .../orm/__pycache__/decl_base.cpython-312.pyc | Bin 68900 -> 0 bytes .../__pycache__/dependency.cpython-312.pyc | Bin 43113 -> 0 bytes .../descriptor_props.cpython-312.pyc | Bin 49633 -> 0 bytes .../orm/__pycache__/dynamic.cpython-312.pyc | Bin 12939 -> 0 bytes .../orm/__pycache__/evaluator.cpython-312.pyc | Bin 16745 -> 0 bytes .../orm/__pycache__/events.cpython-312.pyc | Bin 136484 -> 0 bytes .../orm/__pycache__/exc.cpython-312.pyc | Bin 10215 -> 0 bytes .../orm/__pycache__/identity.cpython-312.pyc | Bin 12613 -> 0 bytes .../instrumentation.cpython-312.pyc | Bin 31196 -> 0 bytes .../__pycache__/interfaces.cpython-312.pyc | Bin 54576 -> 0 bytes .../orm/__pycache__/loading.cpython-312.pyc | Bin 47062 -> 0 bytes .../mapped_collection.cpython-312.pyc | Bin 21884 -> 0 bytes .../orm/__pycache__/mapper.cpython-312.pyc | Bin 168916 -> 0 bytes .../__pycache__/path_registry.cpython-312.pyc | Bin 31492 -> 0 bytes .../__pycache__/persistence.cpython-312.pyc | Bin 48292 -> 0 bytes .../__pycache__/properties.cpython-312.pyc | Bin 34650 -> 0 bytes .../orm/__pycache__/query.cpython-312.pyc | Bin 128767 -> 0 bytes .../__pycache__/relationships.cpython-312.pyc | Bin 130391 -> 0 bytes .../orm/__pycache__/scoping.cpython-312.pyc | Bin 83106 -> 0 bytes .../orm/__pycache__/session.cpython-312.pyc | Bin 202714 -> 0 bytes .../orm/__pycache__/state.cpython-312.pyc | Bin 45866 -> 0 bytes .../__pycache__/state_changes.cpython-312.pyc | Bin 7003 -> 0 bytes .../__pycache__/strategies.cpython-312.pyc | Bin 105069 -> 0 bytes .../strategy_options.cpython-312.pyc | Bin 87512 -> 0 bytes .../orm/__pycache__/sync.cpython-312.pyc | Bin 6569 -> 0 bytes .../__pycache__/unitofwork.cpython-312.pyc | Bin 34030 -> 0 bytes .../orm/__pycache__/util.cpython-312.pyc | Bin 85026 -> 0 bytes .../orm/__pycache__/writeonly.cpython-312.pyc | Bin 28776 -> 0 bytes .../sqlalchemy/orm/_orm_constructors.py | 2661 ----- .../site-packages/sqlalchemy/orm/_typing.py | 179 - .../sqlalchemy/orm/attributes.py | 2845 ------ .env/Lib/site-packages/sqlalchemy/orm/base.py | 971 -- .../sqlalchemy/orm/bulk_persistence.py | 2135 ---- .../sqlalchemy/orm/clsregistry.py | 571 -- .../sqlalchemy/orm/collections.py | 1627 --- .../site-packages/sqlalchemy/orm/context.py | 3334 ------- .../site-packages/sqlalchemy/orm/decl_api.py | 2004 ---- .../site-packages/sqlalchemy/orm/decl_base.py | 2192 ---- .../sqlalchemy/orm/dependency.py | 1302 --- .../sqlalchemy/orm/descriptor_props.py | 1092 -- .../site-packages/sqlalchemy/orm/dynamic.py | 300 - .../site-packages/sqlalchemy/orm/evaluator.py | 379 - .../site-packages/sqlalchemy/orm/events.py | 3252 ------ .env/Lib/site-packages/sqlalchemy/orm/exc.py | 237 - .../site-packages/sqlalchemy/orm/identity.py | 302 - .../sqlalchemy/orm/instrumentation.py | 754 -- .../sqlalchemy/orm/interfaces.py | 1496 --- .../site-packages/sqlalchemy/orm/loading.py | 1686 ---- .../sqlalchemy/orm/mapped_collection.py | 557 -- .../site-packages/sqlalchemy/orm/mapper.py | 4444 --------- .../sqlalchemy/orm/path_registry.py | 809 -- .../sqlalchemy/orm/persistence.py | 1788 ---- .../sqlalchemy/orm/properties.py | 935 -- .../Lib/site-packages/sqlalchemy/orm/query.py | 3459 ------- .../sqlalchemy/orm/relationships.py | 3508 ------- .../site-packages/sqlalchemy/orm/scoping.py | 2148 ---- .../site-packages/sqlalchemy/orm/session.py | 5280 ---------- .../Lib/site-packages/sqlalchemy/orm/state.py | 1168 --- .../sqlalchemy/orm/state_changes.py | 196 - .../sqlalchemy/orm/strategies.py | 3470 ------- .../sqlalchemy/orm/strategy_options.py | 2568 ----- .env/Lib/site-packages/sqlalchemy/orm/sync.py | 164 - .../sqlalchemy/orm/unitofwork.py | 796 -- .env/Lib/site-packages/sqlalchemy/orm/util.py | 2403 ----- .../site-packages/sqlalchemy/orm/writeonly.py | 674 -- .../site-packages/sqlalchemy/pool/__init__.py | 44 - .../pool/__pycache__/__init__.cpython-312.pyc | Bin 1511 -> 0 bytes .../pool/__pycache__/base.cpython-312.pyc | Bin 56399 -> 0 bytes .../pool/__pycache__/events.cpython-312.pyc | Bin 14322 -> 0 bytes .../pool/__pycache__/impl.cpython-312.pyc | Bin 26297 -> 0 bytes .../Lib/site-packages/sqlalchemy/pool/base.py | 1516 --- .../site-packages/sqlalchemy/pool/events.py | 372 - .../Lib/site-packages/sqlalchemy/pool/impl.py | 588 -- .env/Lib/site-packages/sqlalchemy/py.typed | 0 .env/Lib/site-packages/sqlalchemy/schema.py | 69 - .../site-packages/sqlalchemy/sql/__init__.py | 145 - .../sql/__pycache__/__init__.cpython-312.pyc | Bin 4684 -> 0 bytes .../_dml_constructors.cpython-312.pyc | Bin 3992 -> 0 bytes .../_elements_constructors.cpython-312.pyc | Bin 66012 -> 0 bytes .../__pycache__/_orm_types.cpython-312.pyc | Bin 624 -> 0 bytes .../sql/__pycache__/_py_util.cpython-312.pyc | Bin 2963 -> 0 bytes .../_selectable_constructors.cpython-312.pyc | Bin 25147 -> 0 bytes .../sql/__pycache__/_typing.cpython-312.pyc | Bin 15222 -> 0 bytes .../__pycache__/annotation.cpython-312.pyc | Bin 21392 -> 0 bytes .../sql/__pycache__/base.cpython-312.pyc | Bin 100136 -> 0 bytes .../sql/__pycache__/cache_key.cpython-312.pyc | Bin 35639 -> 0 bytes .../sql/__pycache__/coercions.cpython-312.pyc | Bin 49057 -> 0 bytes .../sql/__pycache__/compiler.cpython-312.pyc | Bin 280291 -> 0 bytes .../sql/__pycache__/crud.cpython-312.pyc | Bin 48062 -> 0 bytes .../sql/__pycache__/ddl.cpython-312.pyc | Bin 58531 -> 0 bytes .../default_comparator.cpython-312.pyc | Bin 19477 -> 0 bytes .../sql/__pycache__/dml.cpython-312.pyc | Bin 74314 -> 0 bytes .../sql/__pycache__/elements.cpython-312.pyc | Bin 214337 -> 0 bytes .../sql/__pycache__/events.cpython-312.pyc | Bin 19219 -> 0 bytes .../__pycache__/expression.cpython-312.pyc | Bin 5144 -> 0 bytes .../sql/__pycache__/functions.cpython-312.pyc | Bin 79671 -> 0 bytes .../sql/__pycache__/lambdas.cpython-312.pyc | Bin 54787 -> 0 bytes .../sql/__pycache__/naming.cpython-312.pyc | Bin 8456 -> 0 bytes .../sql/__pycache__/operators.cpython-312.pyc | Bin 89889 -> 0 bytes .../sql/__pycache__/roles.cpython-312.pyc | Bin 12243 -> 0 bytes .../sql/__pycache__/schema.cpython-312.pyc | Bin 245961 -> 0 bytes .../__pycache__/selectable.cpython-312.pyc | Bin 264585 -> 0 bytes .../sql/__pycache__/sqltypes.cpython-312.pyc | Bin 155301 -> 0 bytes .../__pycache__/traversals.cpython-312.pyc | Bin 42346 -> 0 bytes .../sql/__pycache__/type_api.cpython-312.pyc | Bin 87815 -> 0 bytes .../sql/__pycache__/util.cpython-312.pyc | Bin 54386 -> 0 bytes .../sql/__pycache__/visitors.cpython-312.pyc | Bin 36065 -> 0 bytes .../sqlalchemy/sql/_dml_constructors.py | 132 - .../sqlalchemy/sql/_elements_constructors.py | 1872 ---- .../sqlalchemy/sql/_orm_types.py | 20 - .../site-packages/sqlalchemy/sql/_py_util.py | 75 - .../sql/_selectable_constructors.py | 763 -- .../site-packages/sqlalchemy/sql/_typing.py | 482 - .../sqlalchemy/sql/annotation.py | 587 -- .env/Lib/site-packages/sqlalchemy/sql/base.py | 2219 ----- .../site-packages/sqlalchemy/sql/cache_key.py | 1057 -- .../site-packages/sqlalchemy/sql/coercions.py | 1404 --- .../site-packages/sqlalchemy/sql/compiler.py | 8035 --------------- .env/Lib/site-packages/sqlalchemy/sql/crud.py | 1744 ---- .env/Lib/site-packages/sqlalchemy/sql/ddl.py | 1444 --- .../sqlalchemy/sql/default_comparator.py | 551 - .env/Lib/site-packages/sqlalchemy/sql/dml.py | 1850 ---- .../site-packages/sqlalchemy/sql/elements.py | 5589 ----------- .../site-packages/sqlalchemy/sql/events.py | 458 - .../sqlalchemy/sql/expression.py | 159 - .../site-packages/sqlalchemy/sql/functions.py | 2158 ---- .../site-packages/sqlalchemy/sql/lambdas.py | 1442 --- .../site-packages/sqlalchemy/sql/naming.py | 209 - .../site-packages/sqlalchemy/sql/operators.py | 2623 ----- .../Lib/site-packages/sqlalchemy/sql/roles.py | 323 - .../site-packages/sqlalchemy/sql/schema.py | 6222 ------------ .../sqlalchemy/sql/selectable.py | 7265 -------------- .../site-packages/sqlalchemy/sql/sqltypes.py | 3930 -------- .../sqlalchemy/sql/traversals.py | 1024 -- .../site-packages/sqlalchemy/sql/type_api.py | 2368 ----- .env/Lib/site-packages/sqlalchemy/sql/util.py | 1485 --- .../site-packages/sqlalchemy/sql/visitors.py | 1164 --- .../sqlalchemy/testing/__init__.py | 96 - .../__pycache__/__init__.cpython-312.pyc | Bin 3328 -> 0 bytes .../__pycache__/assertions.cpython-312.pyc | Bin 42055 -> 0 bytes .../__pycache__/assertsql.cpython-312.pyc | Bin 20273 -> 0 bytes .../__pycache__/asyncio.cpython-312.pyc | Bin 4148 -> 0 bytes .../__pycache__/config.cpython-312.pyc | Bin 18193 -> 0 bytes .../__pycache__/engines.cpython-312.pyc | Bin 21183 -> 0 bytes .../__pycache__/entities.cpython-312.pyc | Bin 4982 -> 0 bytes .../__pycache__/exclusions.cpython-312.pyc | Bin 23442 -> 0 bytes .../__pycache__/pickleable.cpython-312.pyc | Bin 6687 -> 0 bytes .../__pycache__/profiling.cpython-312.pyc | Bin 13197 -> 0 bytes .../__pycache__/provision.cpython-312.pyc | Bin 24983 -> 0 bytes .../__pycache__/requirements.cpython-312.pyc | Bin 92296 -> 0 bytes .../__pycache__/schema.cpython-312.pyc | Bin 8369 -> 0 bytes .../testing/__pycache__/util.cpython-312.pyc | Bin 21772 -> 0 bytes .../__pycache__/warnings.cpython-312.pyc | Bin 2007 -> 0 bytes .../sqlalchemy/testing/assertions.py | 994 -- .../sqlalchemy/testing/assertsql.py | 520 - .../sqlalchemy/testing/asyncio.py | 135 - .../sqlalchemy/testing/config.py | 434 - .../sqlalchemy/testing/engines.py | 478 - .../sqlalchemy/testing/entities.py | 117 - .../sqlalchemy/testing/exclusions.py | 476 - .../sqlalchemy/testing/fixtures/__init__.py | 28 - .../__pycache__/__init__.cpython-312.pyc | Bin 909 -> 0 bytes .../fixtures/__pycache__/base.cpython-312.pyc | Bin 13464 -> 0 bytes .../fixtures/__pycache__/mypy.cpython-312.pyc | Bin 13670 -> 0 bytes .../fixtures/__pycache__/orm.cpython-312.pyc | Bin 11448 -> 0 bytes .../fixtures/__pycache__/sql.cpython-312.pyc | Bin 21401 -> 0 bytes .../sqlalchemy/testing/fixtures/base.py | 366 - .../sqlalchemy/testing/fixtures/mypy.py | 332 - .../sqlalchemy/testing/fixtures/orm.py | 227 - .../sqlalchemy/testing/fixtures/sql.py | 482 - .../sqlalchemy/testing/pickleable.py | 155 - .../sqlalchemy/testing/plugin/__init__.py | 6 - .../__pycache__/__init__.cpython-312.pyc | Bin 198 -> 0 bytes .../__pycache__/bootstrap.cpython-312.pyc | Bin 2149 -> 0 bytes .../__pycache__/plugin_base.cpython-312.pyc | Bin 29229 -> 0 bytes .../__pycache__/pytestplugin.cpython-312.pyc | Bin 33612 -> 0 bytes .../sqlalchemy/testing/plugin/bootstrap.py | 51 - .../sqlalchemy/testing/plugin/plugin_base.py | 828 -- .../sqlalchemy/testing/plugin/pytestplugin.py | 892 -- .../sqlalchemy/testing/profiling.py | 329 - .../sqlalchemy/testing/provision.py | 597 -- .../sqlalchemy/testing/requirements.py | 1940 ---- .../sqlalchemy/testing/schema.py | 198 - .../sqlalchemy/testing/suite/__init__.py | 19 - .../__pycache__/__init__.cpython-312.pyc | Bin 565 -> 0 bytes .../__pycache__/test_cte.cpython-312.pyc | Bin 11080 -> 0 bytes .../__pycache__/test_ddl.cpython-312.pyc | Bin 18764 -> 0 bytes .../test_deprecations.cpython-312.pyc | Bin 9006 -> 0 bytes .../__pycache__/test_dialect.cpython-312.pyc | Bin 36190 -> 0 bytes .../__pycache__/test_insert.cpython-312.pyc | Bin 25274 -> 0 bytes .../test_reflection.cpython-312.pyc | Bin 155409 -> 0 bytes .../__pycache__/test_results.cpython-312.pyc | Bin 25494 -> 0 bytes .../__pycache__/test_rowcount.cpython-312.pyc | Bin 10349 -> 0 bytes .../__pycache__/test_select.cpython-312.pyc | Bin 115091 -> 0 bytes .../__pycache__/test_sequence.cpython-312.pyc | Bin 14999 -> 0 bytes .../__pycache__/test_types.cpython-312.pyc | Bin 99068 -> 0 bytes .../test_unicode_ddl.cpython-312.pyc | Bin 7619 -> 0 bytes .../test_update_delete.cpython-312.pyc | Bin 7398 -> 0 bytes .../sqlalchemy/testing/suite/test_cte.py | 237 - .../sqlalchemy/testing/suite/test_ddl.py | 389 - .../testing/suite/test_deprecations.py | 153 - .../sqlalchemy/testing/suite/test_dialect.py | 776 -- .../sqlalchemy/testing/suite/test_insert.py | 630 -- .../testing/suite/test_reflection.py | 3557 ------- .../sqlalchemy/testing/suite/test_results.py | 504 - .../sqlalchemy/testing/suite/test_rowcount.py | 258 - .../sqlalchemy/testing/suite/test_select.py | 2010 ---- .../sqlalchemy/testing/suite/test_sequence.py | 317 - .../sqlalchemy/testing/suite/test_types.py | 2147 ---- .../testing/suite/test_unicode_ddl.py | 189 - .../testing/suite/test_update_delete.py | 139 - .../site-packages/sqlalchemy/testing/util.py | 535 - .../sqlalchemy/testing/warnings.py | 52 - .env/Lib/site-packages/sqlalchemy/types.py | 74 - .../site-packages/sqlalchemy/util/__init__.py | 162 - .../util/__pycache__/__init__.cpython-312.pyc | Bin 5737 -> 0 bytes .../__pycache__/_collections.cpython-312.pyc | Bin 31785 -> 0 bytes .../_concurrency_py3k.cpython-312.pyc | Bin 10844 -> 0 bytes .../util/__pycache__/_has_cy.cpython-312.pyc | Bin 1089 -> 0 bytes .../_py_collections.cpython-312.pyc | Bin 29232 -> 0 bytes .../util/__pycache__/compat.cpython-312.pyc | Bin 12937 -> 0 bytes .../__pycache__/concurrency.cpython-312.pyc | Bin 4093 -> 0 bytes .../__pycache__/deprecations.cpython-312.pyc | Bin 13642 -> 0 bytes .../__pycache__/langhelpers.cpython-312.pyc | Bin 86730 -> 0 bytes .../__pycache__/preloaded.cpython-312.pyc | Bin 5881 -> 0 bytes .../util/__pycache__/queue.cpython-312.pyc | Bin 14589 -> 0 bytes .../__pycache__/tool_support.cpython-312.pyc | Bin 8694 -> 0 bytes .../__pycache__/topological.cpython-312.pyc | Bin 3905 -> 0 bytes .../util/__pycache__/typing.cpython-312.pyc | Bin 26158 -> 0 bytes .../sqlalchemy/util/_collections.py | 717 -- .../sqlalchemy/util/_concurrency_py3k.py | 288 - .../site-packages/sqlalchemy/util/_has_cy.py | 40 - .../sqlalchemy/util/_py_collections.py | 541 - .../site-packages/sqlalchemy/util/compat.py | 316 - .../sqlalchemy/util/concurrency.py | 110 - .../sqlalchemy/util/deprecations.py | 401 - .../sqlalchemy/util/langhelpers.py | 2306 ----- .../sqlalchemy/util/preloaded.py | 150 - .../site-packages/sqlalchemy/util/queue.py | 322 - .../sqlalchemy/util/tool_support.py | 201 - .../sqlalchemy/util/topological.py | 120 - .../site-packages/sqlalchemy/util/typing.py | 734 -- .../starlette-0.52.1.dist-info/INSTALLER | 1 - .../starlette-0.52.1.dist-info/METADATA | 177 - .../starlette-0.52.1.dist-info/RECORD | 74 - .../starlette-0.52.1.dist-info/WHEEL | 4 - .../licenses/LICENSE.md | 27 - .env/Lib/site-packages/starlette/__init__.py | 1 - .../__pycache__/__init__.cpython-312.pyc | Bin 209 -> 0 bytes .../_exception_handler.cpython-312.pyc | Bin 3068 -> 0 bytes .../__pycache__/_utils.cpython-312.pyc | Bin 5117 -> 0 bytes .../__pycache__/applications.cpython-312.pyc | Bin 12879 -> 0 bytes .../authentication.cpython-312.pyc | Bin 7710 -> 0 bytes .../__pycache__/background.cpython-312.pyc | Bin 2530 -> 0 bytes .../__pycache__/concurrency.cpython-312.pyc | Bin 3170 -> 0 bytes .../__pycache__/config.cpython-312.pyc | Bin 7508 -> 0 bytes .../__pycache__/convertors.cpython-312.pyc | Bin 4761 -> 0 bytes .../datastructures.cpython-312.pyc | Bin 40316 -> 0 bytes .../__pycache__/endpoints.cpython-312.pyc | Bin 7950 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 2393 -> 0 bytes .../__pycache__/formparsers.cpython-312.pyc | Bin 14265 -> 0 bytes .../__pycache__/requests.cpython-312.pyc | Bin 16676 -> 0 bytes .../__pycache__/responses.cpython-312.pyc | Bin 30226 -> 0 bytes .../__pycache__/routing.cpython-312.pyc | Bin 43920 -> 0 bytes .../__pycache__/schemas.cpython-312.pyc | Bin 7060 -> 0 bytes .../__pycache__/staticfiles.cpython-312.pyc | Bin 11536 -> 0 bytes .../__pycache__/status.cpython-312.pyc | Bin 5179 -> 0 bytes .../__pycache__/templating.cpython-312.pyc | Bin 9981 -> 0 bytes .../__pycache__/testclient.cpython-312.pyc | Bin 33135 -> 0 bytes .../__pycache__/types.cpython-312.pyc | Bin 1314 -> 0 bytes .../__pycache__/websockets.cpython-312.pyc | Bin 11818 -> 0 bytes .../starlette/_exception_handler.py | 65 - .env/Lib/site-packages/starlette/_utils.py | 105 - .../site-packages/starlette/applications.py | 244 - .../site-packages/starlette/authentication.py | 142 - .../Lib/site-packages/starlette/background.py | 36 - .../site-packages/starlette/concurrency.py | 57 - .env/Lib/site-packages/starlette/config.py | 140 - .../Lib/site-packages/starlette/convertors.py | 89 - .../site-packages/starlette/datastructures.py | 706 -- .env/Lib/site-packages/starlette/endpoints.py | 123 - .../Lib/site-packages/starlette/exceptions.py | 33 - .../site-packages/starlette/formparsers.py | 276 - .../starlette/middleware/__init__.py | 37 - .../__pycache__/__init__.cpython-312.pyc | Bin 2533 -> 0 bytes .../authentication.cpython-312.pyc | Bin 2893 -> 0 bytes .../__pycache__/base.cpython-312.pyc | Bin 11796 -> 0 bytes .../__pycache__/cors.cpython-312.pyc | Bin 7835 -> 0 bytes .../__pycache__/errors.cpython-312.pyc | Bin 9933 -> 0 bytes .../__pycache__/exceptions.cpython-312.pyc | Bin 4099 -> 0 bytes .../__pycache__/gzip.cpython-312.pyc | Bin 8788 -> 0 bytes .../__pycache__/httpsredirect.cpython-312.pyc | Bin 1768 -> 0 bytes .../__pycache__/sessions.cpython-312.pyc | Bin 4661 -> 0 bytes .../__pycache__/trustedhost.cpython-312.pyc | Bin 3134 -> 0 bytes .../__pycache__/wsgi.cpython-312.pyc | Bin 8558 -> 0 bytes .../starlette/middleware/authentication.py | 52 - .../starlette/middleware/base.py | 244 - .../starlette/middleware/cors.py | 181 - .../starlette/middleware/errors.py | 259 - .../starlette/middleware/exceptions.py | 73 - .../starlette/middleware/gzip.py | 145 - .../starlette/middleware/httpsredirect.py | 19 - .../starlette/middleware/sessions.py | 85 - .../starlette/middleware/trustedhost.py | 60 - .../starlette/middleware/wsgi.py | 154 - .env/Lib/site-packages/starlette/py.typed | 0 .env/Lib/site-packages/starlette/requests.py | 332 - .env/Lib/site-packages/starlette/responses.py | 566 -- .env/Lib/site-packages/starlette/routing.py | 876 -- .env/Lib/site-packages/starlette/schemas.py | 148 - .../site-packages/starlette/staticfiles.py | 217 - .env/Lib/site-packages/starlette/status.py | 209 - .../Lib/site-packages/starlette/templating.py | 217 - .../Lib/site-packages/starlette/testclient.py | 739 -- .env/Lib/site-packages/starlette/types.py | 26 - .../Lib/site-packages/starlette/websockets.py | 196 - .../INSTALLER | 1 - .../METADATA | 72 - .../typing_extensions-4.15.0.dist-info/RECORD | 7 - .../typing_extensions-4.15.0.dist-info/WHEEL | 4 - .../licenses/LICENSE | 279 - .env/Lib/site-packages/typing_extensions.py | 4317 -------- .../INSTALLER | 1 - .../METADATA | 49 - .../typing_inspection-0.4.2.dist-info/RECORD | 13 - .../typing_inspection-0.4.2.dist-info/WHEEL | 4 - .../licenses/LICENSE | 21 - .../typing_inspection/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 190 -> 0 bytes .../__pycache__/introspection.cpython-312.pyc | Bin 17787 -> 0 bytes .../typing_objects.cpython-312.pyc | Bin 17366 -> 0 bytes .../typing_inspection/introspection.py | 587 -- .../site-packages/typing_inspection/py.typed | 0 .../typing_inspection/typing_objects.py | 607 -- .../typing_inspection/typing_objects.pyi | 417 - .../uvicorn-0.40.0.dist-info/INSTALLER | 1 - .../uvicorn-0.40.0.dist-info/METADATA | 191 - .../uvicorn-0.40.0.dist-info/RECORD | 89 - .../uvicorn-0.40.0.dist-info/REQUESTED | 0 .../uvicorn-0.40.0.dist-info/WHEEL | 4 - .../uvicorn-0.40.0.dist-info/entry_points.txt | 2 - .../licenses/LICENSE.md | 27 - .env/Lib/site-packages/uvicorn/__init__.py | 5 - .env/Lib/site-packages/uvicorn/__main__.py | 4 - .../__pycache__/__init__.cpython-312.pyc | Bin 378 -> 0 bytes .../__pycache__/__main__.cpython-312.pyc | Bin 298 -> 0 bytes .../__pycache__/_compat.cpython-312.pyc | Bin 4000 -> 0 bytes .../__pycache__/_subprocess.cpython-312.pyc | Bin 2941 -> 0 bytes .../__pycache__/_types.cpython-312.pyc | Bin 11409 -> 0 bytes .../__pycache__/config.cpython-312.pyc | Bin 25898 -> 0 bytes .../__pycache__/importer.cpython-312.pyc | Bin 1771 -> 0 bytes .../__pycache__/logging.cpython-312.pyc | Bin 7780 -> 0 bytes .../uvicorn/__pycache__/main.cpython-312.pyc | Bin 20991 -> 0 bytes .../__pycache__/server.cpython-312.pyc | Bin 16702 -> 0 bytes .../__pycache__/workers.cpython-312.pyc | Bin 6559 -> 0 bytes .env/Lib/site-packages/uvicorn/_compat.py | 91 - .env/Lib/site-packages/uvicorn/_subprocess.py | 84 - .env/Lib/site-packages/uvicorn/_types.py | 274 - .env/Lib/site-packages/uvicorn/config.py | 549 - .env/Lib/site-packages/uvicorn/importer.py | 34 - .../uvicorn/lifespan/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 189 -> 0 bytes .../lifespan/__pycache__/off.cpython-312.pyc | Bin 978 -> 0 bytes .../lifespan/__pycache__/on.cpython-312.pyc | Bin 7937 -> 0 bytes .../Lib/site-packages/uvicorn/lifespan/off.py | 17 - .env/Lib/site-packages/uvicorn/lifespan/on.py | 137 - .env/Lib/site-packages/uvicorn/logging.py | 117 - .../site-packages/uvicorn/loops/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 186 -> 0 bytes .../loops/__pycache__/asyncio.cpython-312.pyc | Bin 718 -> 0 bytes .../loops/__pycache__/auto.cpython-312.pyc | Bin 818 -> 0 bytes .../loops/__pycache__/uvloop.cpython-312.pyc | Bin 575 -> 0 bytes .../site-packages/uvicorn/loops/asyncio.py | 11 - .env/Lib/site-packages/uvicorn/loops/auto.py | 17 - .../Lib/site-packages/uvicorn/loops/uvloop.py | 10 - .env/Lib/site-packages/uvicorn/main.py | 618 -- .../uvicorn/middleware/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 191 -> 0 bytes .../__pycache__/asgi2.cpython-312.pyc | Bin 1001 -> 0 bytes .../message_logger.cpython-312.pyc | Bin 4392 -> 0 bytes .../__pycache__/proxy_headers.cpython-312.pyc | Bin 5811 -> 0 bytes .../__pycache__/wsgi.cpython-312.pyc | Bin 9932 -> 0 bytes .../site-packages/uvicorn/middleware/asgi2.py | 15 - .../uvicorn/middleware/message_logger.py | 87 - .../uvicorn/middleware/proxy_headers.py | 142 - .../site-packages/uvicorn/middleware/wsgi.py | 199 - .../uvicorn/protocols/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 190 -> 0 bytes .../__pycache__/utils.cpython-312.pyc | Bin 2961 -> 0 bytes .../uvicorn/protocols/http/__init__.py | 0 .../http/__pycache__/__init__.cpython-312.pyc | Bin 195 -> 0 bytes .../http/__pycache__/auto.cpython-312.pyc | Bin 606 -> 0 bytes .../__pycache__/flow_control.cpython-312.pyc | Bin 3036 -> 0 bytes .../http/__pycache__/h11_impl.cpython-312.pyc | Bin 27177 -> 0 bytes .../httptools_impl.cpython-312.pyc | Bin 29180 -> 0 bytes .../uvicorn/protocols/http/auto.py | 15 - .../uvicorn/protocols/http/flow_control.py | 54 - .../uvicorn/protocols/http/h11_impl.py | 550 - .../uvicorn/protocols/http/httptools_impl.py | 577 -- .../site-packages/uvicorn/protocols/utils.py | 56 - .../uvicorn/protocols/websockets/__init__.py | 0 .../__pycache__/__init__.cpython-312.pyc | Bin 201 -> 0 bytes .../__pycache__/auto.cpython-312.pyc | Bin 810 -> 0 bytes .../websockets_impl.cpython-312.pyc | Bin 20300 -> 0 bytes .../websockets_sansio_impl.cpython-312.pyc | Bin 24379 -> 0 bytes .../__pycache__/wsproto_impl.cpython-312.pyc | Bin 20872 -> 0 bytes .../uvicorn/protocols/websockets/auto.py | 21 - .../protocols/websockets/websockets_impl.py | 384 - .../websockets/websockets_sansio_impl.py | 415 - .../protocols/websockets/wsproto_impl.py | 375 - .env/Lib/site-packages/uvicorn/py.typed | 1 - .env/Lib/site-packages/uvicorn/server.py | 338 - .../uvicorn/supervisors/__init__.py | 16 - .../__pycache__/__init__.cpython-312.pyc | Bin 755 -> 0 bytes .../__pycache__/basereload.cpython-312.pyc | Bin 7027 -> 0 bytes .../__pycache__/multiprocess.cpython-312.pyc | Bin 13446 -> 0 bytes .../__pycache__/statreload.cpython-312.pyc | Bin 2821 -> 0 bytes .../watchfilesreload.cpython-312.pyc | Bin 4241 -> 0 bytes .../uvicorn/supervisors/basereload.py | 125 - .../uvicorn/supervisors/multiprocess.py | 223 - .../uvicorn/supervisors/statreload.py | 52 - .../uvicorn/supervisors/watchfilesreload.py | 85 - .env/Lib/site-packages/uvicorn/workers.py | 111 - .env/Scripts/Activate.ps1 | 502 - .env/Scripts/activate | 70 - .env/Scripts/activate.bat | 34 - .env/Scripts/deactivate.bat | 22 - .env/Scripts/fastapi.exe | Bin 108414 -> 0 bytes .env/Scripts/pip.exe | Bin 108425 -> 0 bytes .env/Scripts/pip3.12.exe | Bin 108425 -> 0 bytes .env/Scripts/pip3.exe | Bin 108425 -> 0 bytes .env/Scripts/python.exe | Bin 270104 -> 0 bytes .env/Scripts/pythonw.exe | Bin 258840 -> 0 bytes .env/Scripts/uvicorn.exe | Bin 108415 -> 0 bytes .env/pyvenv.cfg | 8 +- README.txt | 2 +- 2264 files changed, 5 insertions(+), 531366 deletions(-) delete mode 100644 .env/Include/site/python3.12/greenlet/greenlet.h delete mode 100644 .env/Lib/site-packages/__pycache__/typing_extensions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/entry_points.txt delete mode 100644 .env/Lib/site-packages/annotated_doc-0.0.4.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/annotated_doc/__init__.py delete mode 100644 .env/Lib/site-packages/annotated_doc/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/annotated_doc/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/annotated_doc/main.py delete mode 100644 .env/Lib/site-packages/annotated_doc/py.typed delete mode 100644 .env/Lib/site-packages/annotated_types-0.7.0.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/annotated_types-0.7.0.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/annotated_types-0.7.0.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/annotated_types-0.7.0.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/annotated_types-0.7.0.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/annotated_types/__init__.py delete mode 100644 .env/Lib/site-packages/annotated_types/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/annotated_types/__pycache__/test_cases.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/annotated_types/py.typed delete mode 100644 .env/Lib/site-packages/annotated_types/test_cases.py delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/entry_points.txt delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/anyio-4.12.1.dist-info/top_level.txt delete mode 100644 .env/Lib/site-packages/anyio/__init__.py delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/from_thread.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/functools.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/lowlevel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/pytest_plugin.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/to_interpreter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/to_process.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/__pycache__/to_thread.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_backends/__init__.py delete mode 100644 .env/Lib/site-packages/anyio/_backends/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_backends/__pycache__/_asyncio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_backends/__pycache__/_trio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_backends/_asyncio.py delete mode 100644 .env/Lib/site-packages/anyio/_backends/_trio.py delete mode 100644 .env/Lib/site-packages/anyio/_core/__init__.py delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_asyncio_selector_thread.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_contextmanagers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_eventloop.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_fileio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_resources.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_signals.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_sockets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_streams.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_subprocesses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_synchronization.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_tasks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_tempfile.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_testing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/__pycache__/_typedattr.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/_core/_asyncio_selector_thread.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_contextmanagers.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_eventloop.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_exceptions.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_fileio.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_resources.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_signals.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_sockets.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_streams.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_subprocesses.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_synchronization.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_tasks.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_tempfile.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_testing.py delete mode 100644 .env/Lib/site-packages/anyio/_core/_typedattr.py delete mode 100644 .env/Lib/site-packages/anyio/abc/__init__.py delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_eventloop.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_resources.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_sockets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_streams.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_subprocesses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_tasks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/__pycache__/_testing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/abc/_eventloop.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_resources.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_sockets.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_streams.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_subprocesses.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_tasks.py delete mode 100644 .env/Lib/site-packages/anyio/abc/_testing.py delete mode 100644 .env/Lib/site-packages/anyio/from_thread.py delete mode 100644 .env/Lib/site-packages/anyio/functools.py delete mode 100644 .env/Lib/site-packages/anyio/lowlevel.py delete mode 100644 .env/Lib/site-packages/anyio/py.typed delete mode 100644 .env/Lib/site-packages/anyio/pytest_plugin.py delete mode 100644 .env/Lib/site-packages/anyio/streams/__init__.py delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/buffered.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/file.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/memory.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/stapled.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/text.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/__pycache__/tls.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/anyio/streams/buffered.py delete mode 100644 .env/Lib/site-packages/anyio/streams/file.py delete mode 100644 .env/Lib/site-packages/anyio/streams/memory.py delete mode 100644 .env/Lib/site-packages/anyio/streams/stapled.py delete mode 100644 .env/Lib/site-packages/anyio/streams/text.py delete mode 100644 .env/Lib/site-packages/anyio/streams/tls.py delete mode 100644 .env/Lib/site-packages/anyio/to_interpreter.py delete mode 100644 .env/Lib/site-packages/anyio/to_process.py delete mode 100644 .env/Lib/site-packages/anyio/to_thread.py delete mode 100644 .env/Lib/site-packages/click-8.3.1.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/click-8.3.1.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/click-8.3.1.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/click-8.3.1.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/click-8.3.1.dist-info/licenses/LICENSE.txt delete mode 100644 .env/Lib/site-packages/click/__init__.py delete mode 100644 .env/Lib/site-packages/click/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/_compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/_termui_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/_textwrap.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/_winconsole.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/core.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/decorators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/formatting.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/globals.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/parser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/shell_completion.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/termui.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/testing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/click/_compat.py delete mode 100644 .env/Lib/site-packages/click/_termui_impl.py delete mode 100644 .env/Lib/site-packages/click/_textwrap.py delete mode 100644 .env/Lib/site-packages/click/_utils.py delete mode 100644 .env/Lib/site-packages/click/_winconsole.py delete mode 100644 .env/Lib/site-packages/click/core.py delete mode 100644 .env/Lib/site-packages/click/decorators.py delete mode 100644 .env/Lib/site-packages/click/exceptions.py delete mode 100644 .env/Lib/site-packages/click/formatting.py delete mode 100644 .env/Lib/site-packages/click/globals.py delete mode 100644 .env/Lib/site-packages/click/parser.py delete mode 100644 .env/Lib/site-packages/click/py.typed delete mode 100644 .env/Lib/site-packages/click/shell_completion.py delete mode 100644 .env/Lib/site-packages/click/termui.py delete mode 100644 .env/Lib/site-packages/click/testing.py delete mode 100644 .env/Lib/site-packages/click/types.py delete mode 100644 .env/Lib/site-packages/click/utils.py delete mode 100644 .env/Lib/site-packages/colorama-0.4.6.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/colorama-0.4.6.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/colorama-0.4.6.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/colorama-0.4.6.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/colorama-0.4.6.dist-info/licenses/LICENSE.txt delete mode 100644 .env/Lib/site-packages/colorama/__init__.py delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/ansi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/ansitowin32.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/initialise.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/win32.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/__pycache__/winterm.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/ansi.py delete mode 100644 .env/Lib/site-packages/colorama/ansitowin32.py delete mode 100644 .env/Lib/site-packages/colorama/initialise.py delete mode 100644 .env/Lib/site-packages/colorama/tests/__init__.py delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/ansi_test.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/ansitowin32_test.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/initialise_test.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/isatty_test.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/__pycache__/winterm_test.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/colorama/tests/ansi_test.py delete mode 100644 .env/Lib/site-packages/colorama/tests/ansitowin32_test.py delete mode 100644 .env/Lib/site-packages/colorama/tests/initialise_test.py delete mode 100644 .env/Lib/site-packages/colorama/tests/isatty_test.py delete mode 100644 .env/Lib/site-packages/colorama/tests/utils.py delete mode 100644 .env/Lib/site-packages/colorama/tests/winterm_test.py delete mode 100644 .env/Lib/site-packages/colorama/win32.py delete mode 100644 .env/Lib/site-packages/colorama/winterm.py delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/REQUESTED delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/entry_points.txt delete mode 100644 .env/Lib/site-packages/fastapi-0.128.3.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/fastapi/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/__main__.py delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/applications.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/background.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/cli.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/concurrency.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/datastructures.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/encoders.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/exception_handlers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/logger.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/param_functions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/params.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/requests.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/responses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/routing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/staticfiles.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/templating.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/testclient.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/__pycache__/websockets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/_compat/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/_compat/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/_compat/__pycache__/shared.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/_compat/__pycache__/v2.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/_compat/shared.py delete mode 100644 .env/Lib/site-packages/fastapi/_compat/v2.py delete mode 100644 .env/Lib/site-packages/fastapi/applications.py delete mode 100644 .env/Lib/site-packages/fastapi/background.py delete mode 100644 .env/Lib/site-packages/fastapi/cli.py delete mode 100644 .env/Lib/site-packages/fastapi/concurrency.py delete mode 100644 .env/Lib/site-packages/fastapi/datastructures.py delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/__pycache__/models.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/models.py delete mode 100644 .env/Lib/site-packages/fastapi/dependencies/utils.py delete mode 100644 .env/Lib/site-packages/fastapi/encoders.py delete mode 100644 .env/Lib/site-packages/fastapi/exception_handlers.py delete mode 100644 .env/Lib/site-packages/fastapi/exceptions.py delete mode 100644 .env/Lib/site-packages/fastapi/logger.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/asyncexitstack.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/cors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/gzip.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/httpsredirect.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/trustedhost.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/__pycache__/wsgi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/middleware/asyncexitstack.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/cors.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/gzip.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/httpsredirect.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/trustedhost.py delete mode 100644 .env/Lib/site-packages/fastapi/middleware/wsgi.py delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__pycache__/constants.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__pycache__/docs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__pycache__/models.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/openapi/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/openapi/constants.py delete mode 100644 .env/Lib/site-packages/fastapi/openapi/docs.py delete mode 100644 .env/Lib/site-packages/fastapi/openapi/models.py delete mode 100644 .env/Lib/site-packages/fastapi/openapi/utils.py delete mode 100644 .env/Lib/site-packages/fastapi/param_functions.py delete mode 100644 .env/Lib/site-packages/fastapi/params.py delete mode 100644 .env/Lib/site-packages/fastapi/py.typed delete mode 100644 .env/Lib/site-packages/fastapi/requests.py delete mode 100644 .env/Lib/site-packages/fastapi/responses.py delete mode 100644 .env/Lib/site-packages/fastapi/routing.py delete mode 100644 .env/Lib/site-packages/fastapi/security/__init__.py delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/api_key.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/http.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/oauth2.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/open_id_connect_url.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/fastapi/security/api_key.py delete mode 100644 .env/Lib/site-packages/fastapi/security/base.py delete mode 100644 .env/Lib/site-packages/fastapi/security/http.py delete mode 100644 .env/Lib/site-packages/fastapi/security/oauth2.py delete mode 100644 .env/Lib/site-packages/fastapi/security/open_id_connect_url.py delete mode 100644 .env/Lib/site-packages/fastapi/security/utils.py delete mode 100644 .env/Lib/site-packages/fastapi/staticfiles.py delete mode 100644 .env/Lib/site-packages/fastapi/templating.py delete mode 100644 .env/Lib/site-packages/fastapi/testclient.py delete mode 100644 .env/Lib/site-packages/fastapi/types.py delete mode 100644 .env/Lib/site-packages/fastapi/utils.py delete mode 100644 .env/Lib/site-packages/fastapi/websockets.py delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE.PSF delete mode 100644 .env/Lib/site-packages/greenlet-3.3.1.dist-info/top_level.txt delete mode 100644 .env/Lib/site-packages/greenlet/CObjects.cpp delete mode 100644 .env/Lib/site-packages/greenlet/PyGreenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/PyGreenlet.hpp delete mode 100644 .env/Lib/site-packages/greenlet/PyGreenletUnswitchable.cpp delete mode 100644 .env/Lib/site-packages/greenlet/PyModule.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TBrokenGreenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TExceptionState.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TGreenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TGreenlet.hpp delete mode 100644 .env/Lib/site-packages/greenlet/TGreenletGlobals.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TMainGreenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TPythonState.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TStackState.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TThreadState.hpp delete mode 100644 .env/Lib/site-packages/greenlet/TThreadStateCreator.hpp delete mode 100644 .env/Lib/site-packages/greenlet/TThreadStateDestroy.cpp delete mode 100644 .env/Lib/site-packages/greenlet/TUserGreenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/__init__.py delete mode 100644 .env/Lib/site-packages/greenlet/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/_greenlet.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/greenlet/greenlet.cpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet.h delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_allocator.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_compiler_compat.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_cpython_compat.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_exceptions.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_internal.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_msvc_compat.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_refs.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_slp_switch.hpp delete mode 100644 .env/Lib/site-packages/greenlet/greenlet_thread_support.hpp delete mode 100644 .env/Lib/site-packages/greenlet/platform/__init__.py delete mode 100644 .env/Lib/site-packages/greenlet/platform/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/platform/setup_switch_x64_masm.cmd delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_aarch64_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_alpha_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_amd64_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_arm32_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_arm32_ios.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_arm64_masm.asm delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_arm64_masm.obj delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_arm64_msvc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_csky_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_loongarch64_linux.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_m68k_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_mips_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc64_aix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc64_linux.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc_aix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc_linux.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc_macosx.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_ppc_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_riscv_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_s390_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_sh_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_sparc_sun_gcc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x32_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x64_masm.asm delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x64_masm.obj delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x64_msvc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x86_msvc.h delete mode 100644 .env/Lib/site-packages/greenlet/platform/switch_x86_unix.h delete mode 100644 .env/Lib/site-packages/greenlet/slp_platformselect.h delete mode 100644 .env/Lib/site-packages/greenlet/tests/__init__.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_clearing_run_switches.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_cpp_exception.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_initialstub_already_started.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_slp_switch.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets2.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_two_greenlets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/leakcheck.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_contextvars.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_cpp.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_extension_interface.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_gc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_generator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_generator_nested.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_greenlet.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_greenlet_trash.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_leaks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_stack_saved.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_throw.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_tracing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/__pycache__/test_weakref.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/greenlet/tests/_test_extension.c delete mode 100644 .env/Lib/site-packages/greenlet/tests/_test_extension.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/greenlet/tests/_test_extension_cpp.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/greenlet/tests/_test_extension_cpp.cpp delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_clearing_run_switches.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_cpp_exception.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_initialstub_already_started.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_slp_switch.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets2.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/fail_switch_two_greenlets.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/leakcheck.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_contextvars.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_cpp.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_extension_interface.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_gc.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_generator.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_generator_nested.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_greenlet.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_greenlet_trash.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_leaks.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_stack_saved.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_throw.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_tracing.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_version.py delete mode 100644 .env/Lib/site-packages/greenlet/tests/test_weakref.py delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/licenses/LICENSE.txt delete mode 100644 .env/Lib/site-packages/h11-0.16.0.dist-info/top_level.txt delete mode 100644 .env/Lib/site-packages/h11/__init__.py delete mode 100644 .env/Lib/site-packages/h11/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_abnf.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_connection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_headers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_readers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_receivebuffer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_state.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/__pycache__/_writers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/h11/_abnf.py delete mode 100644 .env/Lib/site-packages/h11/_connection.py delete mode 100644 .env/Lib/site-packages/h11/_events.py delete mode 100644 .env/Lib/site-packages/h11/_headers.py delete mode 100644 .env/Lib/site-packages/h11/_readers.py delete mode 100644 .env/Lib/site-packages/h11/_receivebuffer.py delete mode 100644 .env/Lib/site-packages/h11/_state.py delete mode 100644 .env/Lib/site-packages/h11/_util.py delete mode 100644 .env/Lib/site-packages/h11/_version.py delete mode 100644 .env/Lib/site-packages/h11/_writers.py delete mode 100644 .env/Lib/site-packages/h11/py.typed delete mode 100644 .env/Lib/site-packages/idna-3.11.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/idna-3.11.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/idna-3.11.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/idna-3.11.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/idna-3.11.dist-info/licenses/LICENSE.md delete mode 100644 .env/Lib/site-packages/idna/__init__.py delete mode 100644 .env/Lib/site-packages/idna/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/codec.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/core.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/idnadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/intranges.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/package_data.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/__pycache__/uts46data.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/idna/codec.py delete mode 100644 .env/Lib/site-packages/idna/compat.py delete mode 100644 .env/Lib/site-packages/idna/core.py delete mode 100644 .env/Lib/site-packages/idna/idnadata.py delete mode 100644 .env/Lib/site-packages/idna/intranges.py delete mode 100644 .env/Lib/site-packages/idna/package_data.py delete mode 100644 .env/Lib/site-packages/idna/py.typed delete mode 100644 .env/Lib/site-packages/idna/uts46data.py delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/AUTHORS.txt delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/LICENSE.txt delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/REQUESTED delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/entry_points.txt delete mode 100644 .env/Lib/site-packages/pip-24.2.dist-info/top_level.txt delete mode 100644 .env/Lib/site-packages/pip/__init__.py delete mode 100644 .env/Lib/site-packages/pip/__main__.py delete mode 100644 .env/Lib/site-packages/pip/__pip-runner__.py delete mode 100644 .env/Lib/site-packages/pip/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/__pycache__/__pip-runner__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/build_env.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/configuration.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/pyproject.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/self_outdated_check.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/__pycache__/wheel_builder.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/build_env.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cache.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/autocompletion.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/base_command.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/cmdoptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/command_context.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/index_command.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/main_parser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/parser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/progress_bars.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/req_command.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/spinners.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/__pycache__/status_codes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/autocompletion.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/base_command.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/cmdoptions.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/command_context.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/index_command.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/main.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/main_parser.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/parser.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/progress_bars.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/req_command.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/spinners.py delete mode 100644 .env/Lib/site-packages/pip/_internal/cli/status_codes.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/check.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/completion.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/configuration.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/debug.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/download.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/freeze.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/hash.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/help.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/index.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/inspect.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/install.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/list.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/search.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/show.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/uninstall.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/cache.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/check.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/completion.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/configuration.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/debug.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/download.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/freeze.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/hash.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/help.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/index.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/inspect.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/install.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/list.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/search.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/show.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/uninstall.py delete mode 100644 .env/Lib/site-packages/pip/_internal/commands/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/configuration.py delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__pycache__/installed.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__pycache__/sdist.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/base.py delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/installed.py delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/sdist.py delete mode 100644 .env/Lib/site-packages/pip/_internal/distributions/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/exceptions.py delete mode 100644 .env/Lib/site-packages/pip/_internal/index/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/index/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/index/__pycache__/collector.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/index/__pycache__/package_finder.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/index/__pycache__/sources.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/index/collector.py delete mode 100644 .env/Lib/site-packages/pip/_internal/index/package_finder.py delete mode 100644 .env/Lib/site-packages/pip/_internal/index/sources.py delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/__pycache__/_distutils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/__pycache__/_sysconfig.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/_distutils.py delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/_sysconfig.py delete mode 100644 .env/Lib/site-packages/pip/_internal/locations/base.py delete mode 100644 .env/Lib/site-packages/pip/_internal/main.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/__pycache__/_json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/__pycache__/pkg_resources.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/_json.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/base.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/__pycache__/_compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/__pycache__/_dists.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/__pycache__/_envs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/_compat.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/_dists.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/importlib/_envs.py delete mode 100644 .env/Lib/site-packages/pip/_internal/metadata/pkg_resources.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/candidate.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/direct_url.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/format_control.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/index.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/installation_report.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/link.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/scheme.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/search_scope.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/selection_prefs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/target_python.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/models/candidate.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/direct_url.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/format_control.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/index.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/installation_report.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/link.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/scheme.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/search_scope.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/selection_prefs.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/target_python.py delete mode 100644 .env/Lib/site-packages/pip/_internal/models/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/auth.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/download.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/lazy_wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/session.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/__pycache__/xmlrpc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/network/auth.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/cache.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/download.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/lazy_wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/session.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/utils.py delete mode 100644 .env/Lib/site-packages/pip/_internal/network/xmlrpc.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/__pycache__/check.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/__pycache__/freeze.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/__pycache__/prepare.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/build_tracker.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/metadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/metadata_editable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/metadata_legacy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/wheel_editable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/__pycache__/wheel_legacy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/build_tracker.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/metadata.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/metadata_editable.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/metadata_legacy.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/wheel_editable.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/build/wheel_legacy.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/check.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/freeze.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/__pycache__/editable_legacy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/editable_legacy.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/install/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/operations/prepare.py delete mode 100644 .env/Lib/site-packages/pip/_internal/pyproject.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/constructors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/req_file.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/req_install.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/req_set.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/__pycache__/req_uninstall.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/req/constructors.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/req_file.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/req_install.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/req_set.py delete mode 100644 .env/Lib/site-packages/pip/_internal/req/req_uninstall.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/base.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/legacy/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/legacy/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/legacy/__pycache__/resolver.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/legacy/resolver.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/candidates.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/factory.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/found_candidates.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/provider.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/reporter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/requirements.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/__pycache__/resolver.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/base.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/candidates.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/factory.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/found_candidates.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/provider.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/reporter.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/requirements.py delete mode 100644 .env/Lib/site-packages/pip/_internal/resolution/resolvelib/resolver.py delete mode 100644 .env/Lib/site-packages/pip/_internal/self_outdated_check.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/_jaraco_text.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/_log.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/appdirs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/compatibility_tags.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/datetime.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/deprecation.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/direct_url_helpers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/egg_link.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/encoding.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/entrypoints.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/filesystem.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/filetypes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/glibc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/hashes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/logging.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/misc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/packaging.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/retry.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/setuptools_build.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/subprocess.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/temp_dir.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/unpacking.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/urls.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/virtualenv.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/_jaraco_text.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/_log.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/appdirs.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/compat.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/compatibility_tags.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/datetime.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/deprecation.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/direct_url_helpers.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/egg_link.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/encoding.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/entrypoints.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/filesystem.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/filetypes.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/glibc.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/hashes.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/logging.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/misc.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/packaging.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/retry.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/setuptools_build.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/subprocess.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/temp_dir.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/unpacking.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/urls.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/virtualenv.py delete mode 100644 .env/Lib/site-packages/pip/_internal/utils/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/bazaar.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/git.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/mercurial.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/subversion.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/__pycache__/versioncontrol.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/bazaar.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/git.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/mercurial.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/subversion.py delete mode 100644 .env/Lib/site-packages/pip/_internal/vcs/versioncontrol.py delete mode 100644 .env/Lib/site-packages/pip/_internal/wheel_builder.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/__pycache__/typing_extensions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/_cmd.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/adapter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/controller.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/filewrapper.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/heuristics.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/serialize.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/__pycache__/wrapper.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/_cmd.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/adapter.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/cache.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/__pycache__/file_cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/__pycache__/redis_cache.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/file_cache.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/caches/redis_cache.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/controller.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/filewrapper.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/heuristics.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/serialize.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/cachecontrol/wrapper.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/__main__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/__pycache__/core.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/cacert.pem delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/core.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/certifi/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/database.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/index.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/locators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/manifest.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/markers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/metadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/resources.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/scripts.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/__pycache__/wheel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/compat.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/database.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/index.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/locators.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/manifest.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/markers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/metadata.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/resources.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/scripts.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/t32.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/t64-arm.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/t64.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/util.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/version.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/w32.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/w64-arm.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/w64.exe delete mode 100644 .env/Lib/site-packages/pip/_vendor/distlib/wheel.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/__main__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/__pycache__/distro.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/distro.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/distro/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/codec.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/core.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/idnadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/intranges.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/package_data.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/__pycache__/uts46data.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/codec.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/compat.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/core.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/idnadata.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/intranges.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/package_data.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/idna/uts46data.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/__pycache__/ext.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/__pycache__/fallback.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/exceptions.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/ext.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/msgpack/fallback.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_elffile.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_manylinux.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_musllinux.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_parser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_structures.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/_tokenizer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/markers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/metadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/requirements.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/specifiers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/tags.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/__pycache__/version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_elffile.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_manylinux.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_musllinux.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_parser.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_structures.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/_tokenizer.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/markers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/metadata.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/requirements.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/specifiers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/tags.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/utils.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/packaging/version.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pkg_resources/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pkg_resources/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__main__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/android.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/macos.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/unix.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/__pycache__/windows.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/android.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/api.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/macos.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/unix.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/version.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/platformdirs/windows.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__main__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/cmdline.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/console.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/filter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/formatter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/lexer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/modeline.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/plugin.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/regexopt.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/scanner.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/sphinxext.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/style.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/token.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/unistring.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/cmdline.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/console.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/filter.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/filters/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/filters/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatter.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/_mapping.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/bbcode.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/groff.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/html.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/img.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/irc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/latex.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/other.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/pangomarkup.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/rtf.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/svg.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/terminal.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/__pycache__/terminal256.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/_mapping.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/bbcode.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/groff.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/html.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/img.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/irc.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/latex.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/other.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/pangomarkup.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/rtf.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/svg.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/terminal.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/formatters/terminal256.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexer.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/__pycache__/_mapping.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/__pycache__/python.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/_mapping.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/lexers/python.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/modeline.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/plugin.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/regexopt.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/scanner.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/sphinxext.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/style.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/styles/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/styles/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/styles/__pycache__/_mapping.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/styles/_mapping.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/token.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/unistring.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pygments/util.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/__pycache__/_compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/__pycache__/_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_compat.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_impl.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_in_process/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_in_process/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_in_process/__pycache__/_in_process.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/pyproject_hooks/_in_process/_in_process.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/__version__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/_internal_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/adapters.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/auth.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/certs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/cookies.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/help.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/hooks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/models.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/packages.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/sessions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/status_codes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/structures.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/__version__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/_internal_utils.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/adapters.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/api.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/auth.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/certs.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/compat.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/cookies.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/exceptions.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/help.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/hooks.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/models.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/packages.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/sessions.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/status_codes.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/structures.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/requests/utils.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__pycache__/providers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__pycache__/reporters.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__pycache__/resolvers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/__pycache__/structs.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/compat/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/compat/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/compat/__pycache__/collections_abc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/compat/collections_abc.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/providers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/reporters.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/resolvers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/resolvelib/structs.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__main__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_cell_widths.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_emoji_codes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_emoji_replace.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_export_format.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_extension.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_fileno.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_inspect.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_log_render.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_loop.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_null_file.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_palettes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_pick.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_ratio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_spinners.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_stack.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_timer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_win32_console.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_windows.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_windows_renderer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/_wrap.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/abc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/align.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/ansi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/bar.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/box.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/cells.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/color.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/color_triplet.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/columns.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/console.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/constrain.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/containers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/control.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/default_styles.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/diagnose.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/emoji.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/errors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/file_proxy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/filesize.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/highlighter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/jupyter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/layout.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/live.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/live_render.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/logging.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/markup.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/measure.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/padding.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/pager.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/palette.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/panel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/pretty.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/progress.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/progress_bar.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/prompt.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/protocol.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/region.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/repr.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/rule.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/scope.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/screen.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/segment.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/spinner.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/status.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/style.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/styled.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/syntax.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/table.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/terminal_theme.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/text.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/theme.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/themes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/traceback.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/__pycache__/tree.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_cell_widths.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_emoji_codes.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_emoji_replace.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_export_format.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_extension.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_fileno.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_inspect.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_log_render.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_loop.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_null_file.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_palettes.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_pick.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_ratio.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_spinners.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_stack.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_timer.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_win32_console.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_windows.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_windows_renderer.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/_wrap.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/abc.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/align.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/ansi.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/bar.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/box.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/cells.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/color.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/color_triplet.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/columns.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/console.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/constrain.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/containers.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/control.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/default_styles.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/diagnose.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/emoji.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/errors.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/file_proxy.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/filesize.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/highlighter.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/json.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/jupyter.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/layout.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/live.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/live_render.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/logging.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/markup.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/measure.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/padding.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/pager.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/palette.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/panel.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/pretty.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/progress.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/progress_bar.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/prompt.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/protocol.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/region.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/repr.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/rule.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/scope.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/screen.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/segment.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/spinner.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/status.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/style.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/styled.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/syntax.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/table.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/terminal_theme.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/text.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/theme.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/themes.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/traceback.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/rich/tree.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/__pycache__/_parser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/__pycache__/_re.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/__pycache__/_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/_parser.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/_re.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/_types.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/tomli/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/_api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/_macos.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/_openssl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/_ssl_constants.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/__pycache__/_windows.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/_api.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/_macos.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/_openssl.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/_ssl_constants.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/_windows.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/truststore/py.typed delete mode 100644 .env/Lib/site-packages/pip/_vendor/typing_extensions.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/_collections.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/_version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/connection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/connectionpool.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/fields.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/filepost.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/poolmanager.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/request.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/__pycache__/response.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/_collections.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/_version.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/connection.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/connectionpool.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/_appengine_environ.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/appengine.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/ntlmpool.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/pyopenssl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/securetransport.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/__pycache__/socks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_appengine_environ.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/__pycache__/bindings.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/__pycache__/low_level.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/bindings.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/_securetransport/low_level.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/appengine.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/ntlmpool.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/pyopenssl.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/securetransport.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/contrib/socks.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/exceptions.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/fields.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/filepost.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/__pycache__/six.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/__pycache__/makefile.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/__pycache__/weakref_finalize.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/makefile.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/backports/weakref_finalize.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/packages/six.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/poolmanager.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/request.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/response.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__init__.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/connection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/proxy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/queue.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/request.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/response.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/retry.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/ssl_.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/ssl_match_hostname.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/ssltransport.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/timeout.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/url.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/__pycache__/wait.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/connection.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/proxy.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/queue.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/request.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/response.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/retry.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/ssl_.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/ssl_match_hostname.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/ssltransport.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/timeout.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/url.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/urllib3/util/wait.py delete mode 100644 .env/Lib/site-packages/pip/_vendor/vendor.txt delete mode 100644 .env/Lib/site-packages/pip/py.typed delete mode 100644 .env/Lib/site-packages/pydantic-2.12.5.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/pydantic-2.12.5.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/pydantic-2.12.5.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/pydantic-2.12.5.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/pydantic-2.12.5.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/pydantic/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/_migration.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/alias_generators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/aliases.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/annotated_handlers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/class_validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/color.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/dataclasses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/datetime_parse.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/decorator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/env_settings.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/error_wrappers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/errors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/fields.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/functional_serializers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/functional_validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/generics.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/json_schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/mypy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/networks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/parse.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/root_model.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/tools.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/type_adapter.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/validate_call_decorator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/__pycache__/warnings.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_core_metadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_core_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_dataclasses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_decorators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_decorators_v1.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_discriminated_union.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_docs_extraction.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_fields.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_forward_ref.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_generate_schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_generics.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_git.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_import_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_internal_dataclass.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_known_annotated_metadata.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_mock_val_ser.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_model_construction.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_namespace_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_repr.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_schema_gather.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_schema_generation_shared.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_serializers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_signature.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_typing_extra.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_validate_call.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/__pycache__/_validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_config.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_core_metadata.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_core_utils.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_dataclasses.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_decorators.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_decorators_v1.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_discriminated_union.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_docs_extraction.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_fields.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_forward_ref.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_generate_schema.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_generics.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_git.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_import_utils.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_internal_dataclass.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_known_annotated_metadata.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_mock_val_ser.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_model_construction.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_namespace_utils.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_repr.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_schema_gather.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_schema_generation_shared.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_serializers.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_signature.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_typing_extra.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_utils.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_validate_call.py delete mode 100644 .env/Lib/site-packages/pydantic/_internal/_validators.py delete mode 100644 .env/Lib/site-packages/pydantic/_migration.py delete mode 100644 .env/Lib/site-packages/pydantic/alias_generators.py delete mode 100644 .env/Lib/site-packages/pydantic/aliases.py delete mode 100644 .env/Lib/site-packages/pydantic/annotated_handlers.py delete mode 100644 .env/Lib/site-packages/pydantic/class_validators.py delete mode 100644 .env/Lib/site-packages/pydantic/color.py delete mode 100644 .env/Lib/site-packages/pydantic/config.py delete mode 100644 .env/Lib/site-packages/pydantic/dataclasses.py delete mode 100644 .env/Lib/site-packages/pydantic/datetime_parse.py delete mode 100644 .env/Lib/site-packages/pydantic/decorator.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/class_validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/copy_internals.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/decorator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/parse.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/__pycache__/tools.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/class_validators.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/config.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/copy_internals.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/decorator.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/json.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/parse.py delete mode 100644 .env/Lib/site-packages/pydantic/deprecated/tools.py delete mode 100644 .env/Lib/site-packages/pydantic/env_settings.py delete mode 100644 .env/Lib/site-packages/pydantic/error_wrappers.py delete mode 100644 .env/Lib/site-packages/pydantic/errors.py delete mode 100644 .env/Lib/site-packages/pydantic/experimental/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/experimental/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/experimental/__pycache__/arguments_schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/experimental/__pycache__/missing_sentinel.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/experimental/__pycache__/pipeline.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/experimental/arguments_schema.py delete mode 100644 .env/Lib/site-packages/pydantic/experimental/missing_sentinel.py delete mode 100644 .env/Lib/site-packages/pydantic/experimental/pipeline.py delete mode 100644 .env/Lib/site-packages/pydantic/fields.py delete mode 100644 .env/Lib/site-packages/pydantic/functional_serializers.py delete mode 100644 .env/Lib/site-packages/pydantic/functional_validators.py delete mode 100644 .env/Lib/site-packages/pydantic/generics.py delete mode 100644 .env/Lib/site-packages/pydantic/json.py delete mode 100644 .env/Lib/site-packages/pydantic/json_schema.py delete mode 100644 .env/Lib/site-packages/pydantic/main.py delete mode 100644 .env/Lib/site-packages/pydantic/mypy.py delete mode 100644 .env/Lib/site-packages/pydantic/networks.py delete mode 100644 .env/Lib/site-packages/pydantic/parse.py delete mode 100644 .env/Lib/site-packages/pydantic/plugin/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/plugin/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/plugin/__pycache__/_loader.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/plugin/__pycache__/_schema_validator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/plugin/_loader.py delete mode 100644 .env/Lib/site-packages/pydantic/plugin/_schema_validator.py delete mode 100644 .env/Lib/site-packages/pydantic/py.typed delete mode 100644 .env/Lib/site-packages/pydantic/root_model.py delete mode 100644 .env/Lib/site-packages/pydantic/schema.py delete mode 100644 .env/Lib/site-packages/pydantic/tools.py delete mode 100644 .env/Lib/site-packages/pydantic/type_adapter.py delete mode 100644 .env/Lib/site-packages/pydantic/types.py delete mode 100644 .env/Lib/site-packages/pydantic/typing.py delete mode 100644 .env/Lib/site-packages/pydantic/utils.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/_hypothesis_plugin.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/annotated_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/class_validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/color.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/dataclasses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/datetime_parse.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/decorator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/env_settings.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/error_wrappers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/errors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/fields.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/generics.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/mypy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/networks.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/parse.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/tools.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/validators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/__pycache__/version.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic/v1/_hypothesis_plugin.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/annotated_types.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/class_validators.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/color.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/config.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/dataclasses.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/datetime_parse.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/decorator.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/env_settings.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/error_wrappers.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/errors.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/fields.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/generics.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/json.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/main.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/mypy.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/networks.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/parse.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/py.typed delete mode 100644 .env/Lib/site-packages/pydantic/v1/schema.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/tools.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/types.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/typing.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/utils.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/validators.py delete mode 100644 .env/Lib/site-packages/pydantic/v1/version.py delete mode 100644 .env/Lib/site-packages/pydantic/validate_call_decorator.py delete mode 100644 .env/Lib/site-packages/pydantic/validators.py delete mode 100644 .env/Lib/site-packages/pydantic/version.py delete mode 100644 .env/Lib/site-packages/pydantic/warnings.py delete mode 100644 .env/Lib/site-packages/pydantic_core-2.41.5.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/pydantic_core-2.41.5.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/pydantic_core-2.41.5.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/pydantic_core-2.41.5.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/pydantic_core-2.41.5.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/pydantic_core/__init__.py delete mode 100644 .env/Lib/site-packages/pydantic_core/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic_core/__pycache__/core_schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/pydantic_core/_pydantic_core.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/pydantic_core/_pydantic_core.pyi delete mode 100644 .env/Lib/site-packages/pydantic_core/core_schema.py delete mode 100644 .env/Lib/site-packages/pydantic_core/py.typed delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/REQUESTED delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/sqlalchemy-2.0.46.dist-info/top_level.txt delete mode 100644 .env/Lib/site-packages/sqlalchemy/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/exc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/inspection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/log.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/__pycache__/aioodbc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/__pycache__/asyncio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/__pycache__/pyodbc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/aioodbc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/asyncio.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/connectors/pyodbc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/collections.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/collections.pyx delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/immutabledict.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/immutabledict.pxd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/immutabledict.pyx delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/processors.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/processors.pyx delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/resultproxy.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/resultproxy.pyx delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/util.cp312-win_amd64.pyd delete mode 100644 .env/Lib/site-packages/sqlalchemy/cyextension/util.pyx delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/__pycache__/_typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/_typing.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/aioodbc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/information_schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/pymssql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/__pycache__/pyodbc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/aioodbc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/information_schema.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/json.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/pymssql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mssql/pyodbc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/aiomysql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/asyncmy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/cymysql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/dml.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/enumerated.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/expression.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/mariadb.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/mariadbconnector.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/mysqlconnector.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/mysqldb.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/pymysql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/pyodbc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/reflection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/reserved_words.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/aiomysql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/asyncmy.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/cymysql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/dml.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/enumerated.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/expression.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/json.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/mariadb.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/mariadbconnector.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/mysqlconnector.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/mysqldb.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/pymysql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/pyodbc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/reflection.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/reserved_words.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/mysql/types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/cx_oracle.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/dictionary.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/oracledb.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/__pycache__/vector.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/cx_oracle.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/dictionary.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/oracledb.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/oracle/vector.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/_psycopg_common.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/array.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/asyncpg.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/dml.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/ext.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/hstore.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/named_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/operators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/pg8000.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/pg_catalog.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/psycopg.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/psycopg2.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/psycopg2cffi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/ranges.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/_psycopg_common.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/array.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/asyncpg.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/dml.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/ext.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/hstore.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/json.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/named_types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/operators.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/pg8000.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/pg_catalog.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/psycopg.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/psycopg2.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/psycopg2cffi.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/ranges.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/postgresql/types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/aiosqlite.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/dml.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/json.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/pysqlcipher.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/__pycache__/pysqlite.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/aiosqlite.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/dml.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/json.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/pysqlcipher.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/sqlite/pysqlite.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/dialects/type_migration_guidelines.txt delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/_py_processors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/_py_row.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/_py_util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/characteristics.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/create.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/cursor.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/default.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/interfaces.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/mock.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/processors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/reflection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/result.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/row.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/strategies.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/url.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/_py_processors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/_py_row.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/_py_util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/characteristics.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/create.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/cursor.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/default.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/events.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/interfaces.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/mock.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/processors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/reflection.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/result.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/row.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/strategies.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/url.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/engine/util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/attr.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/legacy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/__pycache__/registry.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/api.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/attr.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/legacy.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/event/registry.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/events.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/exc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/associationproxy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/automap.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/baked.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/compiler.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/horizontal_shard.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/hybrid.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/indexable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/instrumentation.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/mutable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/orderinglist.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/__pycache__/serializer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/associationproxy.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/engine.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/exc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/result.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/scoping.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/__pycache__/session.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/engine.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/exc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/result.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/scoping.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/asyncio/session.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/automap.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/baked.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/compiler.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/declarative/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/declarative/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/declarative/__pycache__/extensions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/declarative/extensions.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/horizontal_shard.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/hybrid.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/indexable.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/instrumentation.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mutable.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/apply.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/decl_class.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/infer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/names.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/plugin.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/apply.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/decl_class.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/infer.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/names.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/plugin.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/mypy/util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/orderinglist.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/ext/serializer.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/future/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/future/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/future/__pycache__/engine.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/future/engine.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/inspection.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/log.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/_orm_constructors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/_typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/attributes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/bulk_persistence.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/clsregistry.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/collections.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/context.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/decl_api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/decl_base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/dependency.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/descriptor_props.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/dynamic.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/evaluator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/exc.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/identity.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/instrumentation.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/interfaces.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/loading.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/mapped_collection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/mapper.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/path_registry.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/persistence.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/properties.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/query.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/relationships.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/scoping.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/session.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/state.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/state_changes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/strategies.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/strategy_options.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/sync.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/unitofwork.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/__pycache__/writeonly.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/_orm_constructors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/_typing.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/attributes.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/bulk_persistence.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/clsregistry.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/collections.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/context.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/decl_api.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/decl_base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/dependency.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/descriptor_props.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/dynamic.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/evaluator.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/events.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/exc.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/identity.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/instrumentation.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/interfaces.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/loading.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/mapped_collection.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/mapper.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/path_registry.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/persistence.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/properties.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/query.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/relationships.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/scoping.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/session.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/state.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/state_changes.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/strategies.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/strategy_options.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/sync.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/unitofwork.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/orm/writeonly.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/__pycache__/events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/__pycache__/impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/events.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/pool/impl.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/py.typed delete mode 100644 .env/Lib/site-packages/sqlalchemy/schema.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_dml_constructors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_elements_constructors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_orm_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_py_util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_selectable_constructors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/_typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/annotation.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/cache_key.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/coercions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/compiler.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/crud.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/ddl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/default_comparator.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/dml.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/elements.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/events.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/expression.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/functions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/lambdas.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/naming.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/operators.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/roles.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/selectable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/sqltypes.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/traversals.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/type_api.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/__pycache__/visitors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_dml_constructors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_elements_constructors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_orm_types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_py_util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_selectable_constructors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/_typing.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/annotation.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/cache_key.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/coercions.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/compiler.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/crud.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/ddl.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/default_comparator.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/dml.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/elements.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/events.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/expression.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/functions.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/lambdas.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/naming.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/operators.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/roles.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/schema.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/selectable.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/sqltypes.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/traversals.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/type_api.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/sql/visitors.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/assertions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/assertsql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/asyncio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/engines.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/entities.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/exclusions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/pickleable.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/profiling.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/provision.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/requirements.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/schema.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/util.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/__pycache__/warnings.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/assertions.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/assertsql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/asyncio.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/config.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/engines.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/entities.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/exclusions.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__pycache__/mypy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__pycache__/orm.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/__pycache__/sql.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/mypy.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/orm.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/fixtures/sql.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/pickleable.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/__pycache__/bootstrap.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/__pycache__/plugin_base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/__pycache__/pytestplugin.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/bootstrap.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/plugin_base.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/plugin/pytestplugin.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/profiling.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/provision.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/requirements.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/schema.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_cte.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_ddl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_deprecations.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_dialect.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_insert.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_reflection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_results.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_rowcount.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_select.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_sequence.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_unicode_ddl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/__pycache__/test_update_delete.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_cte.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_ddl.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_deprecations.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_dialect.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_insert.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_reflection.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_results.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_rowcount.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_select.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_sequence.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_unicode_ddl.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/suite/test_update_delete.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/util.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/testing/warnings.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/types.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__init__.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/_collections.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/_concurrency_py3k.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/_has_cy.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/_py_collections.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/concurrency.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/deprecations.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/langhelpers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/preloaded.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/queue.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/tool_support.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/topological.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/__pycache__/typing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/_collections.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/_concurrency_py3k.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/_has_cy.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/_py_collections.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/compat.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/concurrency.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/deprecations.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/langhelpers.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/preloaded.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/queue.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/tool_support.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/topological.py delete mode 100644 .env/Lib/site-packages/sqlalchemy/util/typing.py delete mode 100644 .env/Lib/site-packages/starlette-0.52.1.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/starlette-0.52.1.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/starlette-0.52.1.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/starlette-0.52.1.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/starlette-0.52.1.dist-info/licenses/LICENSE.md delete mode 100644 .env/Lib/site-packages/starlette/__init__.py delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/_exception_handler.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/_utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/applications.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/authentication.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/background.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/concurrency.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/convertors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/datastructures.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/endpoints.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/formparsers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/requests.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/responses.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/routing.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/schemas.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/staticfiles.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/status.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/templating.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/testclient.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/__pycache__/websockets.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/_exception_handler.py delete mode 100644 .env/Lib/site-packages/starlette/_utils.py delete mode 100644 .env/Lib/site-packages/starlette/applications.py delete mode 100644 .env/Lib/site-packages/starlette/authentication.py delete mode 100644 .env/Lib/site-packages/starlette/background.py delete mode 100644 .env/Lib/site-packages/starlette/concurrency.py delete mode 100644 .env/Lib/site-packages/starlette/config.py delete mode 100644 .env/Lib/site-packages/starlette/convertors.py delete mode 100644 .env/Lib/site-packages/starlette/datastructures.py delete mode 100644 .env/Lib/site-packages/starlette/endpoints.py delete mode 100644 .env/Lib/site-packages/starlette/exceptions.py delete mode 100644 .env/Lib/site-packages/starlette/formparsers.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/__init__.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/authentication.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/base.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/cors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/errors.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/exceptions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/gzip.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/httpsredirect.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/sessions.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/trustedhost.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/__pycache__/wsgi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/starlette/middleware/authentication.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/base.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/cors.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/errors.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/exceptions.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/gzip.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/httpsredirect.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/sessions.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/trustedhost.py delete mode 100644 .env/Lib/site-packages/starlette/middleware/wsgi.py delete mode 100644 .env/Lib/site-packages/starlette/py.typed delete mode 100644 .env/Lib/site-packages/starlette/requests.py delete mode 100644 .env/Lib/site-packages/starlette/responses.py delete mode 100644 .env/Lib/site-packages/starlette/routing.py delete mode 100644 .env/Lib/site-packages/starlette/schemas.py delete mode 100644 .env/Lib/site-packages/starlette/staticfiles.py delete mode 100644 .env/Lib/site-packages/starlette/status.py delete mode 100644 .env/Lib/site-packages/starlette/templating.py delete mode 100644 .env/Lib/site-packages/starlette/testclient.py delete mode 100644 .env/Lib/site-packages/starlette/types.py delete mode 100644 .env/Lib/site-packages/starlette/websockets.py delete mode 100644 .env/Lib/site-packages/typing_extensions-4.15.0.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/typing_extensions-4.15.0.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/typing_extensions-4.15.0.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/typing_extensions-4.15.0.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/typing_extensions-4.15.0.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/typing_extensions.py delete mode 100644 .env/Lib/site-packages/typing_inspection-0.4.2.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/typing_inspection-0.4.2.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/typing_inspection-0.4.2.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/typing_inspection-0.4.2.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/typing_inspection-0.4.2.dist-info/licenses/LICENSE delete mode 100644 .env/Lib/site-packages/typing_inspection/__init__.py delete mode 100644 .env/Lib/site-packages/typing_inspection/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/typing_inspection/__pycache__/introspection.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/typing_inspection/__pycache__/typing_objects.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/typing_inspection/introspection.py delete mode 100644 .env/Lib/site-packages/typing_inspection/py.typed delete mode 100644 .env/Lib/site-packages/typing_inspection/typing_objects.py delete mode 100644 .env/Lib/site-packages/typing_inspection/typing_objects.pyi delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/INSTALLER delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/METADATA delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/RECORD delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/REQUESTED delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/WHEEL delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/entry_points.txt delete mode 100644 .env/Lib/site-packages/uvicorn-0.40.0.dist-info/licenses/LICENSE.md delete mode 100644 .env/Lib/site-packages/uvicorn/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/__main__.py delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/__main__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/_compat.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/_subprocess.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/_types.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/config.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/importer.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/logging.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/main.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/server.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/__pycache__/workers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/_compat.py delete mode 100644 .env/Lib/site-packages/uvicorn/_subprocess.py delete mode 100644 .env/Lib/site-packages/uvicorn/_types.py delete mode 100644 .env/Lib/site-packages/uvicorn/config.py delete mode 100644 .env/Lib/site-packages/uvicorn/importer.py delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/__pycache__/off.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/__pycache__/on.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/off.py delete mode 100644 .env/Lib/site-packages/uvicorn/lifespan/on.py delete mode 100644 .env/Lib/site-packages/uvicorn/logging.py delete mode 100644 .env/Lib/site-packages/uvicorn/loops/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/loops/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/loops/__pycache__/asyncio.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/loops/__pycache__/auto.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/loops/__pycache__/uvloop.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/loops/asyncio.py delete mode 100644 .env/Lib/site-packages/uvicorn/loops/auto.py delete mode 100644 .env/Lib/site-packages/uvicorn/loops/uvloop.py delete mode 100644 .env/Lib/site-packages/uvicorn/main.py delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__pycache__/asgi2.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__pycache__/message_logger.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__pycache__/proxy_headers.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/__pycache__/wsgi.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/asgi2.py delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/message_logger.py delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/proxy_headers.py delete mode 100644 .env/Lib/site-packages/uvicorn/middleware/wsgi.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/__pycache__/utils.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__pycache__/auto.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__pycache__/flow_control.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__pycache__/h11_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/__pycache__/httptools_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/auto.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/flow_control.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/h11_impl.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/http/httptools_impl.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/utils.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__pycache__/auto.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__pycache__/websockets_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__pycache__/websockets_sansio_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/__pycache__/wsproto_impl.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/auto.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/websockets_impl.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/websockets_sansio_impl.py delete mode 100644 .env/Lib/site-packages/uvicorn/protocols/websockets/wsproto_impl.py delete mode 100644 .env/Lib/site-packages/uvicorn/py.typed delete mode 100644 .env/Lib/site-packages/uvicorn/server.py delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__init__.py delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__pycache__/__init__.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__pycache__/basereload.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__pycache__/multiprocess.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__pycache__/statreload.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/__pycache__/watchfilesreload.cpython-312.pyc delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/basereload.py delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/multiprocess.py delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/statreload.py delete mode 100644 .env/Lib/site-packages/uvicorn/supervisors/watchfilesreload.py delete mode 100644 .env/Lib/site-packages/uvicorn/workers.py delete mode 100644 .env/Scripts/Activate.ps1 delete mode 100644 .env/Scripts/activate delete mode 100644 .env/Scripts/activate.bat delete mode 100644 .env/Scripts/deactivate.bat delete mode 100644 .env/Scripts/fastapi.exe delete mode 100644 .env/Scripts/pip.exe delete mode 100644 .env/Scripts/pip3.12.exe delete mode 100644 .env/Scripts/pip3.exe delete mode 100644 .env/Scripts/python.exe delete mode 100644 .env/Scripts/pythonw.exe delete mode 100644 .env/Scripts/uvicorn.exe diff --git a/.env/Include/site/python3.12/greenlet/greenlet.h b/.env/Include/site/python3.12/greenlet/greenlet.h deleted file mode 100644 index d02a16e..0000000 --- a/.env/Include/site/python3.12/greenlet/greenlet.h +++ /dev/null @@ -1,164 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ - -/* Greenlet object interface */ - -#ifndef Py_GREENLETOBJECT_H -#define Py_GREENLETOBJECT_H - - -#include - -#ifdef __cplusplus -extern "C" { -#endif - -/* This is deprecated and undocumented. It does not change. */ -#define GREENLET_VERSION "1.0.0" - -#ifndef GREENLET_MODULE -#define implementation_ptr_t void* -#endif - -typedef struct _greenlet { - PyObject_HEAD - PyObject* weakreflist; - PyObject* dict; - implementation_ptr_t pimpl; -} PyGreenlet; - -#define PyGreenlet_Check(op) (op && PyObject_TypeCheck(op, &PyGreenlet_Type)) - - -/* C API functions */ - -/* Total number of symbols that are exported */ -#define PyGreenlet_API_pointers 12 - -#define PyGreenlet_Type_NUM 0 -#define PyExc_GreenletError_NUM 1 -#define PyExc_GreenletExit_NUM 2 - -#define PyGreenlet_New_NUM 3 -#define PyGreenlet_GetCurrent_NUM 4 -#define PyGreenlet_Throw_NUM 5 -#define PyGreenlet_Switch_NUM 6 -#define PyGreenlet_SetParent_NUM 7 - -#define PyGreenlet_MAIN_NUM 8 -#define PyGreenlet_STARTED_NUM 9 -#define PyGreenlet_ACTIVE_NUM 10 -#define PyGreenlet_GET_PARENT_NUM 11 - -#ifndef GREENLET_MODULE -/* This section is used by modules that uses the greenlet C API */ -static void** _PyGreenlet_API = NULL; - -# define PyGreenlet_Type \ - (*(PyTypeObject*)_PyGreenlet_API[PyGreenlet_Type_NUM]) - -# define PyExc_GreenletError \ - ((PyObject*)_PyGreenlet_API[PyExc_GreenletError_NUM]) - -# define PyExc_GreenletExit \ - ((PyObject*)_PyGreenlet_API[PyExc_GreenletExit_NUM]) - -/* - * PyGreenlet_New(PyObject *args) - * - * greenlet.greenlet(run, parent=None) - */ -# define PyGreenlet_New \ - (*(PyGreenlet * (*)(PyObject * run, PyGreenlet * parent)) \ - _PyGreenlet_API[PyGreenlet_New_NUM]) - -/* - * PyGreenlet_GetCurrent(void) - * - * greenlet.getcurrent() - */ -# define PyGreenlet_GetCurrent \ - (*(PyGreenlet * (*)(void)) _PyGreenlet_API[PyGreenlet_GetCurrent_NUM]) - -/* - * PyGreenlet_Throw( - * PyGreenlet *greenlet, - * PyObject *typ, - * PyObject *val, - * PyObject *tb) - * - * g.throw(...) - */ -# define PyGreenlet_Throw \ - (*(PyObject * (*)(PyGreenlet * self, \ - PyObject * typ, \ - PyObject * val, \ - PyObject * tb)) \ - _PyGreenlet_API[PyGreenlet_Throw_NUM]) - -/* - * PyGreenlet_Switch(PyGreenlet *greenlet, PyObject *args) - * - * g.switch(*args, **kwargs) - */ -# define PyGreenlet_Switch \ - (*(PyObject * \ - (*)(PyGreenlet * greenlet, PyObject * args, PyObject * kwargs)) \ - _PyGreenlet_API[PyGreenlet_Switch_NUM]) - -/* - * PyGreenlet_SetParent(PyObject *greenlet, PyObject *new_parent) - * - * g.parent = new_parent - */ -# define PyGreenlet_SetParent \ - (*(int (*)(PyGreenlet * greenlet, PyGreenlet * nparent)) \ - _PyGreenlet_API[PyGreenlet_SetParent_NUM]) - -/* - * PyGreenlet_GetParent(PyObject* greenlet) - * - * return greenlet.parent; - * - * This could return NULL even if there is no exception active. - * If it does not return NULL, you are responsible for decrementing the - * reference count. - */ -# define PyGreenlet_GetParent \ - (*(PyGreenlet* (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_GET_PARENT_NUM]) - -/* - * deprecated, undocumented alias. - */ -# define PyGreenlet_GET_PARENT PyGreenlet_GetParent - -# define PyGreenlet_MAIN \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_MAIN_NUM]) - -# define PyGreenlet_STARTED \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_STARTED_NUM]) - -# define PyGreenlet_ACTIVE \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_ACTIVE_NUM]) - - - - -/* Macro that imports greenlet and initializes C API */ -/* NOTE: This has actually moved to ``greenlet._greenlet._C_API``, but we - keep the older definition to be sure older code that might have a copy of - the header still works. */ -# define PyGreenlet_Import() \ - { \ - _PyGreenlet_API = (void**)PyCapsule_Import("greenlet._C_API", 0); \ - } - -#endif /* GREENLET_MODULE */ - -#ifdef __cplusplus -} -#endif -#endif /* !Py_GREENLETOBJECT_H */ diff --git a/.env/Lib/site-packages/__pycache__/typing_extensions.cpython-312.pyc b/.env/Lib/site-packages/__pycache__/typing_extensions.cpython-312.pyc deleted file mode 100644 index 8a66703a2d679599ba945c5a33cef41952ce6a87..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 163639 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 -License-Expression: MIT -License-File: LICENSE -Classifier: Intended Audience :: Information Technology -Classifier: Intended Audience :: System Administrators -Classifier: Operating System :: OS Independent -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python -Classifier: Topic :: Internet -Classifier: Topic :: Software Development :: Libraries :: Application Frameworks -Classifier: Topic :: Software Development :: Libraries :: Python Modules -Classifier: Topic :: Software Development :: Libraries -Classifier: Topic :: Software Development -Classifier: Typing :: Typed -Classifier: Development Status :: 4 - Beta -Classifier: Intended Audience :: Developers -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: 3.13 -Classifier: Programming Language :: Python :: 3.14 -Project-URL: Homepage, https://github.com/fastapi/annotated-doc -Project-URL: Documentation, https://github.com/fastapi/annotated-doc -Project-URL: Repository, https://github.com/fastapi/annotated-doc -Project-URL: Issues, https://github.com/fastapi/annotated-doc/issues -Project-URL: Changelog, https://github.com/fastapi/annotated-doc/release-notes.md -Requires-Python: >=3.8 -Description-Content-Type: text/markdown - -# Annotated Doc - -Document parameters, class attributes, return types, and variables inline, with `Annotated`. - - - Test - - - Coverage - - - Package version - - - Supported Python versions - - -## Installation - -```bash -pip install annotated-doc -``` - -Or with `uv`: - -```Python -uv add annotated-doc -``` - -## Usage - -Import `Doc` and pass a single literal string with the documentation for the specific parameter, class attribute, return type, or variable. - -For example, to document a parameter `name` in a function `hi` you could do: - -```Python -from typing import Annotated - -from annotated_doc import Doc - -def hi(name: Annotated[str, Doc("Who to say hi to")]) -> None: - print(f"Hi, {name}!") -``` - -You can also use it to document class attributes: - -```Python -from typing import Annotated - -from annotated_doc import Doc - -class User: - name: Annotated[str, Doc("The user's name")] - age: Annotated[int, Doc("The user's age")] -``` - -The same way, you could document return types and variables, or anything that could have a type annotation with `Annotated`. - -## Who Uses This - -`annotated-doc` was made for: - -* [FastAPI](https://fastapi.tiangolo.com/) -* [Typer](https://typer.tiangolo.com/) -* [SQLModel](https://sqlmodel.tiangolo.com/) -* [Asyncer](https://asyncer.tiangolo.com/) - -`annotated-doc` is supported by [griffe-typingdoc](https://github.com/mkdocstrings/griffe-typingdoc), which powers reference documentation like the one in the [FastAPI Reference](https://fastapi.tiangolo.com/reference/). - -## Reasons not to use `annotated-doc` - -You are already comfortable with one of the existing docstring formats, like: - -* Sphinx -* numpydoc -* Google -* Keras - -Your team is already comfortable using them. - -You prefer having the documentation about parameters all together in a docstring, separated from the code defining them. - -You care about a specific set of users, using one specific editor, and that editor already has support for the specific docstring format you use. - -## Reasons to use `annotated-doc` - -* No micro-syntax to learn for newcomers, it’s **just Python** syntax. -* **Editing** would be already fully supported by default by any editor (current or future) supporting Python syntax, including syntax errors, syntax highlighting, etc. -* **Rendering** would be relatively straightforward to implement by static tools (tools that don't need runtime execution), as the information can be extracted from the AST they normally already create. -* **Deduplication of information**: the name of a parameter would be defined in a single place, not duplicated inside of a docstring. -* **Elimination** of the possibility of having **inconsistencies** when removing a parameter or class variable and **forgetting to remove** its documentation. -* **Minimization** of the probability of adding a new parameter or class variable and **forgetting to add its documentation**. -* **Elimination** of the possibility of having **inconsistencies** between the **name** of a parameter in the **signature** and the name in the docstring when it is renamed. -* **Access** to the documentation string for each symbol at **runtime**, including existing (older) Python versions. -* A more formalized way to document other symbols, like type aliases, that could use Annotated. -* **Support** for apps using FastAPI, Typer and others. -* **AI Accessibility**: AI tools will have an easier way understanding each parameter as the distance from documentation to parameter is much closer. - -## History - -I ([@tiangolo](https://github.com/tiangolo)) originally wanted for this to be part of the Python standard library (in [PEP 727](https://peps.python.org/pep-0727/)), but the proposal was withdrawn as there was a fair amount of negative feedback and opposition. - -The conclusion was that this was better done as an external effort, in a third-party library. - -So, here it is, with a simpler approach, as a third-party library, in a way that can be used by others, starting with FastAPI and friends. - -## License - -This project is licensed under the terms of the MIT license. diff --git a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/RECORD b/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/RECORD deleted file mode 100644 index 06bbc8d..0000000 --- a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/RECORD +++ /dev/null @@ -1,11 +0,0 @@ -annotated_doc-0.0.4.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -annotated_doc-0.0.4.dist-info/METADATA,sha256=Irm5KJua33dY2qKKAjJ-OhKaVBVIfwFGej_dSe3Z1TU,6566 -annotated_doc-0.0.4.dist-info/RECORD,, -annotated_doc-0.0.4.dist-info/WHEEL,sha256=9P2ygRxDrTJz3gsagc0Z96ukrxjr-LFBGOgv3AuKlCA,90 -annotated_doc-0.0.4.dist-info/entry_points.txt,sha256=6OYgBcLyFCUgeqLgnvMyOJxPCWzgy7se4rLPKtNonMs,34 -annotated_doc-0.0.4.dist-info/licenses/LICENSE,sha256=__Fwd5pqy_ZavbQFwIfxzuF4ZpHkqWpANFF-SlBKDN8,1086 -annotated_doc/__init__.py,sha256=VuyxxUe80kfEyWnOrCx_Bk8hybo3aKo6RYBlkBBYW8k,52 -annotated_doc/__pycache__/__init__.cpython-312.pyc,, -annotated_doc/__pycache__/main.cpython-312.pyc,, -annotated_doc/main.py,sha256=5Zfvxv80SwwLqpRW73AZyZyiM4bWma9QWRbp_cgD20s,1075 -annotated_doc/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 diff --git a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/WHEEL b/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/WHEEL deleted file mode 100644 index 045c8ac..0000000 --- a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/WHEEL +++ /dev/null @@ -1,4 +0,0 @@ -Wheel-Version: 1.0 -Generator: pdm-backend (2.4.5) -Root-Is-Purelib: true -Tag: py3-none-any diff --git a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/entry_points.txt b/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/entry_points.txt deleted file mode 100644 index c3ad472..0000000 --- a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/entry_points.txt +++ /dev/null @@ -1,4 +0,0 @@ -[console_scripts] - -[gui_scripts] - diff --git a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/licenses/LICENSE b/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/licenses/LICENSE deleted file mode 100644 index 7a25446..0000000 --- a/.env/Lib/site-packages/annotated_doc-0.0.4.dist-info/licenses/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2025 Sebastián Ramírez - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. diff --git a/.env/Lib/site-packages/annotated_doc/__init__.py b/.env/Lib/site-packages/annotated_doc/__init__.py deleted file mode 100644 index a0152a7..0000000 --- a/.env/Lib/site-packages/annotated_doc/__init__.py +++ /dev/null @@ -1,3 +0,0 @@ -from .main import Doc as Doc - -__version__ = "0.0.4" diff --git a/.env/Lib/site-packages/annotated_doc/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/annotated_doc/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 9d34a5144ab1cfe7995919d7b67c5ee5773cc776..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 250 zcmX@j%ge<81bYs&Wtsr##~=<2FhLogMSzUy3@HpLj5!Rsj8Tk?3@J=eOqI-> zQksmnm|gOdt5^;64D?L=G?{O)bQ=D0ns#}nl zoSm2s6i&>`%P&bRNll4Q$xqggkI&4@EQycTE2#X%VUwGmQks)$SHual3FNwBULf&- UnURt4E`!ViKG8<*B6gq%0G)zG761SM diff --git a/.env/Lib/site-packages/annotated_doc/__pycache__/main.cpython-312.pyc b/.env/Lib/site-packages/annotated_doc/__pycache__/main.cpython-312.pyc deleted file mode 100644 index e91e2635b3c465d928174a69fc4a6a9eca477b27..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1898 zcmZux&2Jk;6rc639mh>DkhJ1Mps}m8Hg${;mo!3^5GfRiTBspH3P$Vk&N!ZCz2nZV z11>@8AwnE`OC*Gp0|%mB_yag_=i&%Zs#S0N1BM_W<-~il{%BA~t9kFudo%C%zV_Go z`5b}u^Dp1Xw;e+ML}9!@HtX~yAp0amJ|LXz5pF$7C5lbbPnB&i5t4m!=a#u6$fm`~ zW{O*zHcttM+uUh6JiX^^rnxPgR*GkU&j3FI{0#6}o)xno=fG|j<*^9- zu8Ecm1dZB)a^>|pB8XTdRX|mXG8*-}g0dh`qr+Y(gDu*4H#rf!QK6BdO_8Ic)e8lu zGN7K%!jQ5k(z4l$L|CD%Uf`LBS-@#WL~X@smuUv9(4f~M>gj;)Fuw=plLygyhiS>0 zz6h&1bN&f{HFh*?tDesR0ATH^P@)6oFRQ@s(-3?ESYN{X%txHe_th704gq=~D?c>P z`EpxO!DXa0Kn8#36R{-%c)%#(4G56RXaYor)Ki_VFEDW--b~y+4Rr)0LEMGBV2Z%g z4wF8Hh55q#PqIZD=MXeT-fn)$(9>#?ghr$BaUZf6m>61Gb+B3(vFvn}j_9OFQ-e_@ z1$V*0*ltaE$%|aHXj_&7D9P&d_GbXULcyq1{IsnwjUnrU9dO0+=ViKlgVt04sKzbO zg|tVdR`Hgs(68{ZuM5Sp(E?;cAfLWyI)B&NB`$Eu>XRd@ zc4TQBk8&z@LgBZJ5aM-8&o)n3+w)C_D-H!A`Sg(p; zr~Z*_)}edEa+i7AYzvOhrRvQNlR>rHkF%~T0~xvQ{W-`;bu8gPhdB5T$m7@E_}2L& z`^qUu#ItZ8Eeys4R0L)p#(tOF9qXQ=-=%o!dwZRf?b>|@(|%@^bruXD+*u$YzDYHD zLpWM=^%PDby^3at3E;eN@gYCCR64#?8eF{g<6DFL@*umc=Ro;va23$(lV}1QK#nT{ zn(=xx(ytRe17}!*HVfOP1Ue5V|3it&Dau589nBVCdqAG#7ap#iEM7TYyz*QA>L7d7 z0F2WL#zLE}05h|&;3kO;V9w#R@Nbr^@&7c;L}+i;I)O+sV`)lvt!JHXmk1fh=EOM( zzdiVE1fJ07JbtdGq7TPYFZiepO?mVp7>v3wDN@FD#XSfS$Z311p!?DX}6_lE??Ve}7@!~-U None: - print(f"Hi, {name}!") - ``` - """ - - def __init__(self, documentation: str, /) -> None: - self.documentation = documentation - - def __repr__(self) -> str: - return f"Doc({self.documentation!r})" - - def __hash__(self) -> int: - return hash(self.documentation) - - def __eq__(self, other: object) -> bool: - if not isinstance(other, Doc): - return NotImplemented - return self.documentation == other.documentation diff --git a/.env/Lib/site-packages/annotated_doc/py.typed b/.env/Lib/site-packages/annotated_doc/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/INSTALLER b/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/METADATA b/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/METADATA deleted file mode 100644 index 3ac05cf..0000000 --- a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/METADATA +++ /dev/null @@ -1,295 +0,0 @@ -Metadata-Version: 2.3 -Name: annotated-types -Version: 0.7.0 -Summary: Reusable constraint types to use with typing.Annotated -Project-URL: Homepage, https://github.com/annotated-types/annotated-types -Project-URL: Source, https://github.com/annotated-types/annotated-types -Project-URL: Changelog, https://github.com/annotated-types/annotated-types/releases -Author-email: Adrian Garcia Badaracco <1755071+adriangb@users.noreply.github.com>, Samuel Colvin , Zac Hatfield-Dodds -License-File: LICENSE -Classifier: Development Status :: 4 - Beta -Classifier: Environment :: Console -Classifier: Environment :: MacOS X -Classifier: Intended Audience :: Developers -Classifier: Intended Audience :: Information Technology -Classifier: License :: OSI Approved :: MIT License -Classifier: Operating System :: POSIX :: Linux -Classifier: Operating System :: Unix -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Topic :: Software Development :: Libraries :: Python Modules -Classifier: Typing :: Typed -Requires-Python: >=3.8 -Requires-Dist: typing-extensions>=4.0.0; python_version < '3.9' -Description-Content-Type: text/markdown - -# annotated-types - -[![CI](https://github.com/annotated-types/annotated-types/workflows/CI/badge.svg?event=push)](https://github.com/annotated-types/annotated-types/actions?query=event%3Apush+branch%3Amain+workflow%3ACI) -[![pypi](https://img.shields.io/pypi/v/annotated-types.svg)](https://pypi.python.org/pypi/annotated-types) -[![versions](https://img.shields.io/pypi/pyversions/annotated-types.svg)](https://github.com/annotated-types/annotated-types) -[![license](https://img.shields.io/github/license/annotated-types/annotated-types.svg)](https://github.com/annotated-types/annotated-types/blob/main/LICENSE) - -[PEP-593](https://peps.python.org/pep-0593/) added `typing.Annotated` as a way of -adding context-specific metadata to existing types, and specifies that -`Annotated[T, x]` _should_ be treated as `T` by any tool or library without special -logic for `x`. - -This package provides metadata objects which can be used to represent common -constraints such as upper and lower bounds on scalar values and collection sizes, -a `Predicate` marker for runtime checks, and -descriptions of how we intend these metadata to be interpreted. In some cases, -we also note alternative representations which do not require this package. - -## Install - -```bash -pip install annotated-types -``` - -## Examples - -```python -from typing import Annotated -from annotated_types import Gt, Len, Predicate - -class MyClass: - age: Annotated[int, Gt(18)] # Valid: 19, 20, ... - # Invalid: 17, 18, "19", 19.0, ... - factors: list[Annotated[int, Predicate(is_prime)]] # Valid: 2, 3, 5, 7, 11, ... - # Invalid: 4, 8, -2, 5.0, "prime", ... - - my_list: Annotated[list[int], Len(0, 10)] # Valid: [], [10, 20, 30, 40, 50] - # Invalid: (1, 2), ["abc"], [0] * 20 -``` - -## Documentation - -_While `annotated-types` avoids runtime checks for performance, users should not -construct invalid combinations such as `MultipleOf("non-numeric")` or `Annotated[int, Len(3)]`. -Downstream implementors may choose to raise an error, emit a warning, silently ignore -a metadata item, etc., if the metadata objects described below are used with an -incompatible type - or for any other reason!_ - -### Gt, Ge, Lt, Le - -Express inclusive and/or exclusive bounds on orderable values - which may be numbers, -dates, times, strings, sets, etc. Note that the boundary value need not be of the -same type that was annotated, so long as they can be compared: `Annotated[int, Gt(1.5)]` -is fine, for example, and implies that the value is an integer x such that `x > 1.5`. - -We suggest that implementors may also interpret `functools.partial(operator.le, 1.5)` -as being equivalent to `Gt(1.5)`, for users who wish to avoid a runtime dependency on -the `annotated-types` package. - -To be explicit, these types have the following meanings: - -* `Gt(x)` - value must be "Greater Than" `x` - equivalent to exclusive minimum -* `Ge(x)` - value must be "Greater than or Equal" to `x` - equivalent to inclusive minimum -* `Lt(x)` - value must be "Less Than" `x` - equivalent to exclusive maximum -* `Le(x)` - value must be "Less than or Equal" to `x` - equivalent to inclusive maximum - -### Interval - -`Interval(gt, ge, lt, le)` allows you to specify an upper and lower bound with a single -metadata object. `None` attributes should be ignored, and non-`None` attributes -treated as per the single bounds above. - -### MultipleOf - -`MultipleOf(multiple_of=x)` might be interpreted in two ways: - -1. Python semantics, implying `value % multiple_of == 0`, or -2. [JSONschema semantics](https://json-schema.org/draft/2020-12/json-schema-validation.html#rfc.section.6.2.1), - where `int(value / multiple_of) == value / multiple_of`. - -We encourage users to be aware of these two common interpretations and their -distinct behaviours, especially since very large or non-integer numbers make -it easy to cause silent data corruption due to floating-point imprecision. - -We encourage libraries to carefully document which interpretation they implement. - -### MinLen, MaxLen, Len - -`Len()` implies that `min_length <= len(value) <= max_length` - lower and upper bounds are inclusive. - -As well as `Len()` which can optionally include upper and lower bounds, we also -provide `MinLen(x)` and `MaxLen(y)` which are equivalent to `Len(min_length=x)` -and `Len(max_length=y)` respectively. - -`Len`, `MinLen`, and `MaxLen` may be used with any type which supports `len(value)`. - -Examples of usage: - -* `Annotated[list, MaxLen(10)]` (or `Annotated[list, Len(max_length=10))`) - list must have a length of 10 or less -* `Annotated[str, MaxLen(10)]` - string must have a length of 10 or less -* `Annotated[list, MinLen(3))` (or `Annotated[list, Len(min_length=3))`) - list must have a length of 3 or more -* `Annotated[list, Len(4, 6)]` - list must have a length of 4, 5, or 6 -* `Annotated[list, Len(8, 8)]` - list must have a length of exactly 8 - -#### Changed in v0.4.0 - -* `min_inclusive` has been renamed to `min_length`, no change in meaning -* `max_exclusive` has been renamed to `max_length`, upper bound is now **inclusive** instead of **exclusive** -* The recommendation that slices are interpreted as `Len` has been removed due to ambiguity and different semantic - meaning of the upper bound in slices vs. `Len` - -See [issue #23](https://github.com/annotated-types/annotated-types/issues/23) for discussion. - -### Timezone - -`Timezone` can be used with a `datetime` or a `time` to express which timezones -are allowed. `Annotated[datetime, Timezone(None)]` must be a naive datetime. -`Timezone[...]` ([literal ellipsis](https://docs.python.org/3/library/constants.html#Ellipsis)) -expresses that any timezone-aware datetime is allowed. You may also pass a specific -timezone string or [`tzinfo`](https://docs.python.org/3/library/datetime.html#tzinfo-objects) -object such as `Timezone(timezone.utc)` or `Timezone("Africa/Abidjan")` to express that you only -allow a specific timezone, though we note that this is often a symptom of fragile design. - -#### Changed in v0.x.x - -* `Timezone` accepts [`tzinfo`](https://docs.python.org/3/library/datetime.html#tzinfo-objects) objects instead of - `timezone`, extending compatibility to [`zoneinfo`](https://docs.python.org/3/library/zoneinfo.html) and third party libraries. - -### Unit - -`Unit(unit: str)` expresses that the annotated numeric value is the magnitude of -a quantity with the specified unit. For example, `Annotated[float, Unit("m/s")]` -would be a float representing a velocity in meters per second. - -Please note that `annotated_types` itself makes no attempt to parse or validate -the unit string in any way. That is left entirely to downstream libraries, -such as [`pint`](https://pint.readthedocs.io) or -[`astropy.units`](https://docs.astropy.org/en/stable/units/). - -An example of how a library might use this metadata: - -```python -from annotated_types import Unit -from typing import Annotated, TypeVar, Callable, Any, get_origin, get_args - -# given a type annotated with a unit: -Meters = Annotated[float, Unit("m")] - - -# you can cast the annotation to a specific unit type with any -# callable that accepts a string and returns the desired type -T = TypeVar("T") -def cast_unit(tp: Any, unit_cls: Callable[[str], T]) -> T | None: - if get_origin(tp) is Annotated: - for arg in get_args(tp): - if isinstance(arg, Unit): - return unit_cls(arg.unit) - return None - - -# using `pint` -import pint -pint_unit = cast_unit(Meters, pint.Unit) - - -# using `astropy.units` -import astropy.units as u -astropy_unit = cast_unit(Meters, u.Unit) -``` - -### Predicate - -`Predicate(func: Callable)` expresses that `func(value)` is truthy for valid values. -Users should prefer the statically inspectable metadata above, but if you need -the full power and flexibility of arbitrary runtime predicates... here it is. - -For some common constraints, we provide generic types: - -* `IsLower = Annotated[T, Predicate(str.islower)]` -* `IsUpper = Annotated[T, Predicate(str.isupper)]` -* `IsDigit = Annotated[T, Predicate(str.isdigit)]` -* `IsFinite = Annotated[T, Predicate(math.isfinite)]` -* `IsNotFinite = Annotated[T, Predicate(Not(math.isfinite))]` -* `IsNan = Annotated[T, Predicate(math.isnan)]` -* `IsNotNan = Annotated[T, Predicate(Not(math.isnan))]` -* `IsInfinite = Annotated[T, Predicate(math.isinf)]` -* `IsNotInfinite = Annotated[T, Predicate(Not(math.isinf))]` - -so that you can write e.g. `x: IsFinite[float] = 2.0` instead of the longer -(but exactly equivalent) `x: Annotated[float, Predicate(math.isfinite)] = 2.0`. - -Some libraries might have special logic to handle known or understandable predicates, -for example by checking for `str.isdigit` and using its presence to both call custom -logic to enforce digit-only strings, and customise some generated external schema. -Users are therefore encouraged to avoid indirection like `lambda s: s.lower()`, in -favor of introspectable methods such as `str.lower` or `re.compile("pattern").search`. - -To enable basic negation of commonly used predicates like `math.isnan` without introducing introspection that makes it impossible for implementers to introspect the predicate we provide a `Not` wrapper that simply negates the predicate in an introspectable manner. Several of the predicates listed above are created in this manner. - -We do not specify what behaviour should be expected for predicates that raise -an exception. For example `Annotated[int, Predicate(str.isdigit)]` might silently -skip invalid constraints, or statically raise an error; or it might try calling it -and then propagate or discard the resulting -`TypeError: descriptor 'isdigit' for 'str' objects doesn't apply to a 'int' object` -exception. We encourage libraries to document the behaviour they choose. - -### Doc - -`doc()` can be used to add documentation information in `Annotated`, for function and method parameters, variables, class attributes, return types, and any place where `Annotated` can be used. - -It expects a value that can be statically analyzed, as the main use case is for static analysis, editors, documentation generators, and similar tools. - -It returns a `DocInfo` class with a single attribute `documentation` containing the value passed to `doc()`. - -This is the early adopter's alternative form of the [`typing-doc` proposal](https://github.com/tiangolo/fastapi/blob/typing-doc/typing_doc.md). - -### Integrating downstream types with `GroupedMetadata` - -Implementers may choose to provide a convenience wrapper that groups multiple pieces of metadata. -This can help reduce verbosity and cognitive overhead for users. -For example, an implementer like Pydantic might provide a `Field` or `Meta` type that accepts keyword arguments and transforms these into low-level metadata: - -```python -from dataclasses import dataclass -from typing import Iterator -from annotated_types import GroupedMetadata, Ge - -@dataclass -class Field(GroupedMetadata): - ge: int | None = None - description: str | None = None - - def __iter__(self) -> Iterator[object]: - # Iterating over a GroupedMetadata object should yield annotated-types - # constraint metadata objects which describe it as fully as possible, - # and may include other unknown objects too. - if self.ge is not None: - yield Ge(self.ge) - if self.description is not None: - yield Description(self.description) -``` - -Libraries consuming annotated-types constraints should check for `GroupedMetadata` and unpack it by iterating over the object and treating the results as if they had been "unpacked" in the `Annotated` type. The same logic should be applied to the [PEP 646 `Unpack` type](https://peps.python.org/pep-0646/), so that `Annotated[T, Field(...)]`, `Annotated[T, Unpack[Field(...)]]` and `Annotated[T, *Field(...)]` are all treated consistently. - -Libraries consuming annotated-types should also ignore any metadata they do not recongize that came from unpacking a `GroupedMetadata`, just like they ignore unrecognized metadata in `Annotated` itself. - -Our own `annotated_types.Interval` class is a `GroupedMetadata` which unpacks itself into `Gt`, `Lt`, etc., so this is not an abstract concern. Similarly, `annotated_types.Len` is a `GroupedMetadata` which unpacks itself into `MinLen` (optionally) and `MaxLen`. - -### Consuming metadata - -We intend to not be prescriptive as to _how_ the metadata and constraints are used, but as an example of how one might parse constraints from types annotations see our [implementation in `test_main.py`](https://github.com/annotated-types/annotated-types/blob/f59cf6d1b5255a0fe359b93896759a180bec30ae/tests/test_main.py#L94-L103). - -It is up to the implementer to determine how this metadata is used. -You could use the metadata for runtime type checking, for generating schemas or to generate example data, amongst other use cases. - -## Design & History - -This package was designed at the PyCon 2022 sprints by the maintainers of Pydantic -and Hypothesis, with the goal of making it as easy as possible for end-users to -provide more informative annotations for use by runtime libraries. - -It is deliberately minimal, and following PEP-593 allows considerable downstream -discretion in what (if anything!) they choose to support. Nonetheless, we expect -that staying simple and covering _only_ the most common use-cases will give users -and maintainers the best experience we can. If you'd like more constraints for your -types - follow our lead, by defining them and documenting them downstream! diff --git a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/RECORD b/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/RECORD deleted file mode 100644 index 7045729..0000000 --- a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/RECORD +++ /dev/null @@ -1,10 +0,0 @@ -annotated_types-0.7.0.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -annotated_types-0.7.0.dist-info/METADATA,sha256=7ltqxksJJ0wCYFGBNIQCWTlWQGeAH0hRFdnK3CB895E,15046 -annotated_types-0.7.0.dist-info/RECORD,, -annotated_types-0.7.0.dist-info/WHEEL,sha256=zEMcRr9Kr03x1ozGwg5v9NQBKn3kndp6LSoSlVg-jhU,87 -annotated_types-0.7.0.dist-info/licenses/LICENSE,sha256=_hBJiEsaDZNCkB6I4H8ykl0ksxIdmXK2poBfuYJLCV0,1083 -annotated_types/__init__.py,sha256=RynLsRKUEGI0KimXydlD1fZEfEzWwDo0Uon3zOKhG1Q,13819 -annotated_types/__pycache__/__init__.cpython-312.pyc,, -annotated_types/__pycache__/test_cases.cpython-312.pyc,, -annotated_types/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -annotated_types/test_cases.py,sha256=zHFX6EpcMbGJ8FzBYDbO56bPwx_DYIVSKbZM-4B3_lg,6421 diff --git a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/WHEEL b/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/WHEEL deleted file mode 100644 index 516596c..0000000 --- a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/WHEEL +++ /dev/null @@ -1,4 +0,0 @@ -Wheel-Version: 1.0 -Generator: hatchling 1.24.2 -Root-Is-Purelib: true -Tag: py3-none-any diff --git a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/licenses/LICENSE b/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/licenses/LICENSE deleted file mode 100644 index d99323a..0000000 --- a/.env/Lib/site-packages/annotated_types-0.7.0.dist-info/licenses/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2022 the contributors - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. diff --git a/.env/Lib/site-packages/annotated_types/__init__.py b/.env/Lib/site-packages/annotated_types/__init__.py deleted file mode 100644 index 74e0dee..0000000 --- a/.env/Lib/site-packages/annotated_types/__init__.py +++ /dev/null @@ -1,432 +0,0 @@ -import math -import sys -import types -from dataclasses import dataclass -from datetime import tzinfo -from typing import TYPE_CHECKING, Any, Callable, Iterator, Optional, SupportsFloat, SupportsIndex, TypeVar, Union - -if sys.version_info < (3, 8): - from typing_extensions import Protocol, runtime_checkable -else: - from typing import Protocol, runtime_checkable - -if sys.version_info < (3, 9): - from typing_extensions import Annotated, Literal -else: - from typing import Annotated, Literal - -if sys.version_info < (3, 10): - EllipsisType = type(Ellipsis) - KW_ONLY = {} - SLOTS = {} -else: - from types import EllipsisType - - KW_ONLY = {"kw_only": True} - SLOTS = {"slots": True} - - -__all__ = ( - 'BaseMetadata', - 'GroupedMetadata', - 'Gt', - 'Ge', - 'Lt', - 'Le', - 'Interval', - 'MultipleOf', - 'MinLen', - 'MaxLen', - 'Len', - 'Timezone', - 'Predicate', - 'LowerCase', - 'UpperCase', - 'IsDigits', - 'IsFinite', - 'IsNotFinite', - 'IsNan', - 'IsNotNan', - 'IsInfinite', - 'IsNotInfinite', - 'doc', - 'DocInfo', - '__version__', -) - -__version__ = '0.7.0' - - -T = TypeVar('T') - - -# arguments that start with __ are considered -# positional only -# see https://peps.python.org/pep-0484/#positional-only-arguments - - -class SupportsGt(Protocol): - def __gt__(self: T, __other: T) -> bool: - ... - - -class SupportsGe(Protocol): - def __ge__(self: T, __other: T) -> bool: - ... - - -class SupportsLt(Protocol): - def __lt__(self: T, __other: T) -> bool: - ... - - -class SupportsLe(Protocol): - def __le__(self: T, __other: T) -> bool: - ... - - -class SupportsMod(Protocol): - def __mod__(self: T, __other: T) -> T: - ... - - -class SupportsDiv(Protocol): - def __div__(self: T, __other: T) -> T: - ... - - -class BaseMetadata: - """Base class for all metadata. - - This exists mainly so that implementers - can do `isinstance(..., BaseMetadata)` while traversing field annotations. - """ - - __slots__ = () - - -@dataclass(frozen=True, **SLOTS) -class Gt(BaseMetadata): - """Gt(gt=x) implies that the value must be greater than x. - - It can be used with any type that supports the ``>`` operator, - including numbers, dates and times, strings, sets, and so on. - """ - - gt: SupportsGt - - -@dataclass(frozen=True, **SLOTS) -class Ge(BaseMetadata): - """Ge(ge=x) implies that the value must be greater than or equal to x. - - It can be used with any type that supports the ``>=`` operator, - including numbers, dates and times, strings, sets, and so on. - """ - - ge: SupportsGe - - -@dataclass(frozen=True, **SLOTS) -class Lt(BaseMetadata): - """Lt(lt=x) implies that the value must be less than x. - - It can be used with any type that supports the ``<`` operator, - including numbers, dates and times, strings, sets, and so on. - """ - - lt: SupportsLt - - -@dataclass(frozen=True, **SLOTS) -class Le(BaseMetadata): - """Le(le=x) implies that the value must be less than or equal to x. - - It can be used with any type that supports the ``<=`` operator, - including numbers, dates and times, strings, sets, and so on. - """ - - le: SupportsLe - - -@runtime_checkable -class GroupedMetadata(Protocol): - """A grouping of multiple objects, like typing.Unpack. - - `GroupedMetadata` on its own is not metadata and has no meaning. - All of the constraints and metadata should be fully expressable - in terms of the `BaseMetadata`'s returned by `GroupedMetadata.__iter__()`. - - Concrete implementations should override `GroupedMetadata.__iter__()` - to add their own metadata. - For example: - - >>> @dataclass - >>> class Field(GroupedMetadata): - >>> gt: float | None = None - >>> description: str | None = None - ... - >>> def __iter__(self) -> Iterable[object]: - >>> if self.gt is not None: - >>> yield Gt(self.gt) - >>> if self.description is not None: - >>> yield Description(self.gt) - - Also see the implementation of `Interval` below for an example. - - Parsers should recognize this and unpack it so that it can be used - both with and without unpacking: - - - `Annotated[int, Field(...)]` (parser must unpack Field) - - `Annotated[int, *Field(...)]` (PEP-646) - """ # noqa: trailing-whitespace - - @property - def __is_annotated_types_grouped_metadata__(self) -> Literal[True]: - return True - - def __iter__(self) -> Iterator[object]: - ... - - if not TYPE_CHECKING: - __slots__ = () # allow subclasses to use slots - - def __init_subclass__(cls, *args: Any, **kwargs: Any) -> None: - # Basic ABC like functionality without the complexity of an ABC - super().__init_subclass__(*args, **kwargs) - if cls.__iter__ is GroupedMetadata.__iter__: - raise TypeError("Can't subclass GroupedMetadata without implementing __iter__") - - def __iter__(self) -> Iterator[object]: # noqa: F811 - raise NotImplementedError # more helpful than "None has no attribute..." type errors - - -@dataclass(frozen=True, **KW_ONLY, **SLOTS) -class Interval(GroupedMetadata): - """Interval can express inclusive or exclusive bounds with a single object. - - It accepts keyword arguments ``gt``, ``ge``, ``lt``, and/or ``le``, which - are interpreted the same way as the single-bound constraints. - """ - - gt: Union[SupportsGt, None] = None - ge: Union[SupportsGe, None] = None - lt: Union[SupportsLt, None] = None - le: Union[SupportsLe, None] = None - - def __iter__(self) -> Iterator[BaseMetadata]: - """Unpack an Interval into zero or more single-bounds.""" - if self.gt is not None: - yield Gt(self.gt) - if self.ge is not None: - yield Ge(self.ge) - if self.lt is not None: - yield Lt(self.lt) - if self.le is not None: - yield Le(self.le) - - -@dataclass(frozen=True, **SLOTS) -class MultipleOf(BaseMetadata): - """MultipleOf(multiple_of=x) might be interpreted in two ways: - - 1. Python semantics, implying ``value % multiple_of == 0``, or - 2. JSONschema semantics, where ``int(value / multiple_of) == value / multiple_of`` - - We encourage users to be aware of these two common interpretations, - and libraries to carefully document which they implement. - """ - - multiple_of: Union[SupportsDiv, SupportsMod] - - -@dataclass(frozen=True, **SLOTS) -class MinLen(BaseMetadata): - """ - MinLen() implies minimum inclusive length, - e.g. ``len(value) >= min_length``. - """ - - min_length: Annotated[int, Ge(0)] - - -@dataclass(frozen=True, **SLOTS) -class MaxLen(BaseMetadata): - """ - MaxLen() implies maximum inclusive length, - e.g. ``len(value) <= max_length``. - """ - - max_length: Annotated[int, Ge(0)] - - -@dataclass(frozen=True, **SLOTS) -class Len(GroupedMetadata): - """ - Len() implies that ``min_length <= len(value) <= max_length``. - - Upper bound may be omitted or ``None`` to indicate no upper length bound. - """ - - min_length: Annotated[int, Ge(0)] = 0 - max_length: Optional[Annotated[int, Ge(0)]] = None - - def __iter__(self) -> Iterator[BaseMetadata]: - """Unpack a Len into zone or more single-bounds.""" - if self.min_length > 0: - yield MinLen(self.min_length) - if self.max_length is not None: - yield MaxLen(self.max_length) - - -@dataclass(frozen=True, **SLOTS) -class Timezone(BaseMetadata): - """Timezone(tz=...) requires a datetime to be aware (or ``tz=None``, naive). - - ``Annotated[datetime, Timezone(None)]`` must be a naive datetime. - ``Timezone[...]`` (the ellipsis literal) expresses that the datetime must be - tz-aware but any timezone is allowed. - - You may also pass a specific timezone string or tzinfo object such as - ``Timezone(timezone.utc)`` or ``Timezone("Africa/Abidjan")`` to express that - you only allow a specific timezone, though we note that this is often - a symptom of poor design. - """ - - tz: Union[str, tzinfo, EllipsisType, None] - - -@dataclass(frozen=True, **SLOTS) -class Unit(BaseMetadata): - """Indicates that the value is a physical quantity with the specified unit. - - It is intended for usage with numeric types, where the value represents the - magnitude of the quantity. For example, ``distance: Annotated[float, Unit('m')]`` - or ``speed: Annotated[float, Unit('m/s')]``. - - Interpretation of the unit string is left to the discretion of the consumer. - It is suggested to follow conventions established by python libraries that work - with physical quantities, such as - - - ``pint`` : - - ``astropy.units``: - - For indicating a quantity with a certain dimensionality but without a specific unit - it is recommended to use square brackets, e.g. `Annotated[float, Unit('[time]')]`. - Note, however, ``annotated_types`` itself makes no use of the unit string. - """ - - unit: str - - -@dataclass(frozen=True, **SLOTS) -class Predicate(BaseMetadata): - """``Predicate(func: Callable)`` implies `func(value)` is truthy for valid values. - - Users should prefer statically inspectable metadata, but if you need the full - power and flexibility of arbitrary runtime predicates... here it is. - - We provide a few predefined predicates for common string constraints: - ``IsLower = Predicate(str.islower)``, ``IsUpper = Predicate(str.isupper)``, and - ``IsDigits = Predicate(str.isdigit)``. Users are encouraged to use methods which - can be given special handling, and avoid indirection like ``lambda s: s.lower()``. - - Some libraries might have special logic to handle certain predicates, e.g. by - checking for `str.isdigit` and using its presence to both call custom logic to - enforce digit-only strings, and customise some generated external schema. - - We do not specify what behaviour should be expected for predicates that raise - an exception. For example `Annotated[int, Predicate(str.isdigit)]` might silently - skip invalid constraints, or statically raise an error; or it might try calling it - and then propagate or discard the resulting exception. - """ - - func: Callable[[Any], bool] - - def __repr__(self) -> str: - if getattr(self.func, "__name__", "") == "": - return f"{self.__class__.__name__}({self.func!r})" - if isinstance(self.func, (types.MethodType, types.BuiltinMethodType)) and ( - namespace := getattr(self.func.__self__, "__name__", None) - ): - return f"{self.__class__.__name__}({namespace}.{self.func.__name__})" - if isinstance(self.func, type(str.isascii)): # method descriptor - return f"{self.__class__.__name__}({self.func.__qualname__})" - return f"{self.__class__.__name__}({self.func.__name__})" - - -@dataclass -class Not: - func: Callable[[Any], bool] - - def __call__(self, __v: Any) -> bool: - return not self.func(__v) - - -_StrType = TypeVar("_StrType", bound=str) - -LowerCase = Annotated[_StrType, Predicate(str.islower)] -""" -Return True if the string is a lowercase string, False otherwise. - -A string is lowercase if all cased characters in the string are lowercase and there is at least one cased character in the string. -""" # noqa: E501 -UpperCase = Annotated[_StrType, Predicate(str.isupper)] -""" -Return True if the string is an uppercase string, False otherwise. - -A string is uppercase if all cased characters in the string are uppercase and there is at least one cased character in the string. -""" # noqa: E501 -IsDigit = Annotated[_StrType, Predicate(str.isdigit)] -IsDigits = IsDigit # type: ignore # plural for backwards compatibility, see #63 -""" -Return True if the string is a digit string, False otherwise. - -A string is a digit string if all characters in the string are digits and there is at least one character in the string. -""" # noqa: E501 -IsAscii = Annotated[_StrType, Predicate(str.isascii)] -""" -Return True if all characters in the string are ASCII, False otherwise. - -ASCII characters have code points in the range U+0000-U+007F. Empty string is ASCII too. -""" - -_NumericType = TypeVar('_NumericType', bound=Union[SupportsFloat, SupportsIndex]) -IsFinite = Annotated[_NumericType, Predicate(math.isfinite)] -"""Return True if x is neither an infinity nor a NaN, and False otherwise.""" -IsNotFinite = Annotated[_NumericType, Predicate(Not(math.isfinite))] -"""Return True if x is one of infinity or NaN, and False otherwise""" -IsNan = Annotated[_NumericType, Predicate(math.isnan)] -"""Return True if x is a NaN (not a number), and False otherwise.""" -IsNotNan = Annotated[_NumericType, Predicate(Not(math.isnan))] -"""Return True if x is anything but NaN (not a number), and False otherwise.""" -IsInfinite = Annotated[_NumericType, Predicate(math.isinf)] -"""Return True if x is a positive or negative infinity, and False otherwise.""" -IsNotInfinite = Annotated[_NumericType, Predicate(Not(math.isinf))] -"""Return True if x is neither a positive or negative infinity, and False otherwise.""" - -try: - from typing_extensions import DocInfo, doc # type: ignore [attr-defined] -except ImportError: - - @dataclass(frozen=True, **SLOTS) - class DocInfo: # type: ignore [no-redef] - """ " - The return value of doc(), mainly to be used by tools that want to extract the - Annotated documentation at runtime. - """ - - documentation: str - """The documentation string passed to doc().""" - - def doc( - documentation: str, - ) -> DocInfo: - """ - Add documentation to a type annotation inside of Annotated. - - For example: - - >>> def hi(name: Annotated[int, doc("The name of the user")]) -> None: ... - """ - return DocInfo(documentation) diff --git a/.env/Lib/site-packages/annotated_types/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/annotated_types/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 561ba215af9ff68b2176958bcc27f6b35a12c99e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18619 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 diff --git a/.env/Lib/site-packages/annotated_types/__pycache__/test_cases.cpython-312.pyc b/.env/Lib/site-packages/annotated_types/__pycache__/test_cases.cpython-312.pyc deleted file mode 100644 index 7b72f3f44ee38ae8ddd5566277318c567f5964f4..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13232 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 diff --git a/.env/Lib/site-packages/annotated_types/py.typed b/.env/Lib/site-packages/annotated_types/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/annotated_types/test_cases.py b/.env/Lib/site-packages/annotated_types/test_cases.py deleted file mode 100644 index d9164d6..0000000 --- a/.env/Lib/site-packages/annotated_types/test_cases.py +++ /dev/null @@ -1,151 +0,0 @@ -import math -import sys -from datetime import date, datetime, timedelta, timezone -from decimal import Decimal -from typing import Any, Dict, Iterable, Iterator, List, NamedTuple, Set, Tuple - -if sys.version_info < (3, 9): - from typing_extensions import Annotated -else: - from typing import Annotated - -import annotated_types as at - - -class Case(NamedTuple): - """ - A test case for `annotated_types`. - """ - - annotation: Any - valid_cases: Iterable[Any] - invalid_cases: Iterable[Any] - - -def cases() -> Iterable[Case]: - # Gt, Ge, Lt, Le - yield Case(Annotated[int, at.Gt(4)], (5, 6, 1000), (4, 0, -1)) - yield Case(Annotated[float, at.Gt(0.5)], (0.6, 0.7, 0.8, 0.9), (0.5, 0.0, -0.1)) - yield Case( - Annotated[datetime, at.Gt(datetime(2000, 1, 1))], - [datetime(2000, 1, 2), datetime(2000, 1, 3)], - [datetime(2000, 1, 1), datetime(1999, 12, 31)], - ) - yield Case( - Annotated[datetime, at.Gt(date(2000, 1, 1))], - [date(2000, 1, 2), date(2000, 1, 3)], - [date(2000, 1, 1), date(1999, 12, 31)], - ) - yield Case( - Annotated[datetime, at.Gt(Decimal('1.123'))], - [Decimal('1.1231'), Decimal('123')], - [Decimal('1.123'), Decimal('0')], - ) - - yield Case(Annotated[int, at.Ge(4)], (4, 5, 6, 1000, 4), (0, -1)) - yield Case(Annotated[float, at.Ge(0.5)], (0.5, 0.6, 0.7, 0.8, 0.9), (0.4, 0.0, -0.1)) - yield Case( - Annotated[datetime, at.Ge(datetime(2000, 1, 1))], - [datetime(2000, 1, 2), datetime(2000, 1, 3)], - [datetime(1998, 1, 1), datetime(1999, 12, 31)], - ) - - yield Case(Annotated[int, at.Lt(4)], (0, -1), (4, 5, 6, 1000, 4)) - yield Case(Annotated[float, at.Lt(0.5)], (0.4, 0.0, -0.1), (0.5, 0.6, 0.7, 0.8, 0.9)) - yield Case( - Annotated[datetime, at.Lt(datetime(2000, 1, 1))], - [datetime(1999, 12, 31), datetime(1999, 12, 31)], - [datetime(2000, 1, 2), datetime(2000, 1, 3)], - ) - - yield Case(Annotated[int, at.Le(4)], (4, 0, -1), (5, 6, 1000)) - yield Case(Annotated[float, at.Le(0.5)], (0.5, 0.0, -0.1), (0.6, 0.7, 0.8, 0.9)) - yield Case( - Annotated[datetime, at.Le(datetime(2000, 1, 1))], - [datetime(2000, 1, 1), datetime(1999, 12, 31)], - [datetime(2000, 1, 2), datetime(2000, 1, 3)], - ) - - # Interval - yield Case(Annotated[int, at.Interval(gt=4)], (5, 6, 1000), (4, 0, -1)) - yield Case(Annotated[int, at.Interval(gt=4, lt=10)], (5, 6), (4, 10, 1000, 0, -1)) - yield Case(Annotated[float, at.Interval(ge=0.5, le=1)], (0.5, 0.9, 1), (0.49, 1.1)) - yield Case( - Annotated[datetime, at.Interval(gt=datetime(2000, 1, 1), le=datetime(2000, 1, 3))], - [datetime(2000, 1, 2), datetime(2000, 1, 3)], - [datetime(2000, 1, 1), datetime(2000, 1, 4)], - ) - - yield Case(Annotated[int, at.MultipleOf(multiple_of=3)], (0, 3, 9), (1, 2, 4)) - yield Case(Annotated[float, at.MultipleOf(multiple_of=0.5)], (0, 0.5, 1, 1.5), (0.4, 1.1)) - - # lengths - - yield Case(Annotated[str, at.MinLen(3)], ('123', '1234', 'x' * 10), ('', '1', '12')) - yield Case(Annotated[str, at.Len(3)], ('123', '1234', 'x' * 10), ('', '1', '12')) - yield Case(Annotated[List[int], at.MinLen(3)], ([1, 2, 3], [1, 2, 3, 4], [1] * 10), ([], [1], [1, 2])) - yield Case(Annotated[List[int], at.Len(3)], ([1, 2, 3], [1, 2, 3, 4], [1] * 10), ([], [1], [1, 2])) - - yield Case(Annotated[str, at.MaxLen(4)], ('', '1234'), ('12345', 'x' * 10)) - yield Case(Annotated[str, at.Len(0, 4)], ('', '1234'), ('12345', 'x' * 10)) - yield Case(Annotated[List[str], at.MaxLen(4)], ([], ['a', 'bcdef'], ['a', 'b', 'c']), (['a'] * 5, ['b'] * 10)) - yield Case(Annotated[List[str], at.Len(0, 4)], ([], ['a', 'bcdef'], ['a', 'b', 'c']), (['a'] * 5, ['b'] * 10)) - - yield Case(Annotated[str, at.Len(3, 5)], ('123', '12345'), ('', '1', '12', '123456', 'x' * 10)) - yield Case(Annotated[str, at.Len(3, 3)], ('123',), ('12', '1234')) - - yield Case(Annotated[Dict[int, int], at.Len(2, 3)], [{1: 1, 2: 2}], [{}, {1: 1}, {1: 1, 2: 2, 3: 3, 4: 4}]) - yield Case(Annotated[Set[int], at.Len(2, 3)], ({1, 2}, {1, 2, 3}), (set(), {1}, {1, 2, 3, 4})) - yield Case(Annotated[Tuple[int, ...], at.Len(2, 3)], ((1, 2), (1, 2, 3)), ((), (1,), (1, 2, 3, 4))) - - # Timezone - - yield Case( - Annotated[datetime, at.Timezone(None)], [datetime(2000, 1, 1)], [datetime(2000, 1, 1, tzinfo=timezone.utc)] - ) - yield Case( - Annotated[datetime, at.Timezone(...)], [datetime(2000, 1, 1, tzinfo=timezone.utc)], [datetime(2000, 1, 1)] - ) - yield Case( - Annotated[datetime, at.Timezone(timezone.utc)], - [datetime(2000, 1, 1, tzinfo=timezone.utc)], - [datetime(2000, 1, 1), datetime(2000, 1, 1, tzinfo=timezone(timedelta(hours=6)))], - ) - yield Case( - Annotated[datetime, at.Timezone('Europe/London')], - [datetime(2000, 1, 1, tzinfo=timezone(timedelta(0), name='Europe/London'))], - [datetime(2000, 1, 1), datetime(2000, 1, 1, tzinfo=timezone(timedelta(hours=6)))], - ) - - # Quantity - - yield Case(Annotated[float, at.Unit(unit='m')], (5, 4.2), ('5m', '4.2m')) - - # predicate types - - yield Case(at.LowerCase[str], ['abc', 'foobar'], ['', 'A', 'Boom']) - yield Case(at.UpperCase[str], ['ABC', 'DEFO'], ['', 'a', 'abc', 'AbC']) - yield Case(at.IsDigit[str], ['123'], ['', 'ab', 'a1b2']) - yield Case(at.IsAscii[str], ['123', 'foo bar'], ['£100', '😊', 'whatever 👀']) - - yield Case(Annotated[int, at.Predicate(lambda x: x % 2 == 0)], [0, 2, 4], [1, 3, 5]) - - yield Case(at.IsFinite[float], [1.23], [math.nan, math.inf, -math.inf]) - yield Case(at.IsNotFinite[float], [math.nan, math.inf], [1.23]) - yield Case(at.IsNan[float], [math.nan], [1.23, math.inf]) - yield Case(at.IsNotNan[float], [1.23, math.inf], [math.nan]) - yield Case(at.IsInfinite[float], [math.inf], [math.nan, 1.23]) - yield Case(at.IsNotInfinite[float], [math.nan, 1.23], [math.inf]) - - # check stacked predicates - yield Case(at.IsInfinite[Annotated[float, at.Predicate(lambda x: x > 0)]], [math.inf], [-math.inf, 1.23, math.nan]) - - # doc - yield Case(Annotated[int, at.doc("A number")], [1, 2], []) - - # custom GroupedMetadata - class MyCustomGroupedMetadata(at.GroupedMetadata): - def __iter__(self) -> Iterator[at.Predicate]: - yield at.Predicate(lambda x: float(x).is_integer()) - - yield Case(Annotated[float, MyCustomGroupedMetadata()], [0, 2.0], [0.01, 1.5]) diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/INSTALLER b/.env/Lib/site-packages/anyio-4.12.1.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/METADATA b/.env/Lib/site-packages/anyio-4.12.1.dist-info/METADATA deleted file mode 100644 index dbeb198..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/METADATA +++ /dev/null @@ -1,96 +0,0 @@ -Metadata-Version: 2.4 -Name: anyio -Version: 4.12.1 -Summary: High-level concurrency and networking framework on top of asyncio or Trio -Author-email: Alex Grönholm -License-Expression: MIT -Project-URL: Documentation, https://anyio.readthedocs.io/en/latest/ -Project-URL: Changelog, https://anyio.readthedocs.io/en/stable/versionhistory.html -Project-URL: Source code, https://github.com/agronholm/anyio -Project-URL: Issue tracker, https://github.com/agronholm/anyio/issues -Classifier: Development Status :: 5 - Production/Stable -Classifier: Intended Audience :: Developers -Classifier: Framework :: AnyIO -Classifier: Typing :: Typed -Classifier: Programming Language :: Python -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: 3.13 -Classifier: Programming Language :: Python :: 3.14 -Requires-Python: >=3.9 -Description-Content-Type: text/x-rst -License-File: LICENSE -Requires-Dist: exceptiongroup>=1.0.2; python_version < "3.11" -Requires-Dist: idna>=2.8 -Requires-Dist: typing_extensions>=4.5; python_version < "3.13" -Provides-Extra: trio -Requires-Dist: trio>=0.32.0; python_version >= "3.10" and extra == "trio" -Requires-Dist: trio>=0.31.0; python_version < "3.10" and extra == "trio" -Dynamic: license-file - -.. image:: https://github.com/agronholm/anyio/actions/workflows/test.yml/badge.svg - :target: https://github.com/agronholm/anyio/actions/workflows/test.yml - :alt: Build Status -.. image:: https://coveralls.io/repos/github/agronholm/anyio/badge.svg?branch=master - :target: https://coveralls.io/github/agronholm/anyio?branch=master - :alt: Code Coverage -.. image:: https://readthedocs.org/projects/anyio/badge/?version=latest - :target: https://anyio.readthedocs.io/en/latest/?badge=latest - :alt: Documentation -.. image:: https://badges.gitter.im/gitterHQ/gitter.svg - :target: https://gitter.im/python-trio/AnyIO - :alt: Gitter chat - -AnyIO is an asynchronous networking and concurrency library that works on top of either asyncio_ or -Trio_. It implements Trio-like `structured concurrency`_ (SC) on top of asyncio and works in harmony -with the native SC of Trio itself. - -Applications and libraries written against AnyIO's API will run unmodified on either asyncio_ or -Trio_. AnyIO can also be adopted into a library or application incrementally – bit by bit, no full -refactoring necessary. It will blend in with the native libraries of your chosen backend. - -To find out why you might want to use AnyIO's APIs instead of asyncio's, you can read about it -`here `_. - -Documentation -------------- - -View full documentation at: https://anyio.readthedocs.io/ - -Features --------- - -AnyIO offers the following functionality: - -* Task groups (nurseries_ in trio terminology) -* High-level networking (TCP, UDP and UNIX sockets) - - * `Happy eyeballs`_ algorithm for TCP connections (more robust than that of asyncio on Python - 3.8) - * async/await style UDP sockets (unlike asyncio where you still have to use Transports and - Protocols) - -* A versatile API for byte streams and object streams -* Inter-task synchronization and communication (locks, conditions, events, semaphores, object - streams) -* Worker threads -* Subprocesses -* Subinterpreter support for code parallelization (on Python 3.13 and later) -* Asynchronous file I/O (using worker threads) -* Signal handling -* Asynchronous version of the functools_ module - -AnyIO also comes with its own pytest_ plugin which also supports asynchronous fixtures. -It even works with the popular Hypothesis_ library. - -.. _asyncio: https://docs.python.org/3/library/asyncio.html -.. _Trio: https://github.com/python-trio/trio -.. _structured concurrency: https://en.wikipedia.org/wiki/Structured_concurrency -.. _nurseries: https://trio.readthedocs.io/en/stable/reference-core.html#nurseries-and-spawning -.. _Happy eyeballs: https://en.wikipedia.org/wiki/Happy_Eyeballs -.. _pytest: https://docs.pytest.org/en/latest/ -.. _functools: https://docs.python.org/3/library/functools.html -.. _Hypothesis: https://hypothesis.works/ diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/RECORD b/.env/Lib/site-packages/anyio-4.12.1.dist-info/RECORD deleted file mode 100644 index 9be42bb..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/RECORD +++ /dev/null @@ -1,92 +0,0 @@ -anyio-4.12.1.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -anyio-4.12.1.dist-info/METADATA,sha256=DfiDab9Tmmcfy802lOLTMEHJQShkOSbopCwqCYbLuJk,4277 -anyio-4.12.1.dist-info/RECORD,, -anyio-4.12.1.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91 -anyio-4.12.1.dist-info/entry_points.txt,sha256=_d6Yu6uiaZmNe0CydowirE9Cmg7zUL2g08tQpoS3Qvc,39 -anyio-4.12.1.dist-info/licenses/LICENSE,sha256=U2GsncWPLvX9LpsJxoKXwX8ElQkJu8gCO9uC6s8iwrA,1081 -anyio-4.12.1.dist-info/top_level.txt,sha256=QglSMiWX8_5dpoVAEIHdEYzvqFMdSYWmCj6tYw2ITkQ,6 -anyio/__init__.py,sha256=7iDVqMUprUuKNY91FuoKqayAhR-OY136YDPI6P78HHk,6170 -anyio/__pycache__/__init__.cpython-312.pyc,, -anyio/__pycache__/from_thread.cpython-312.pyc,, -anyio/__pycache__/functools.cpython-312.pyc,, -anyio/__pycache__/lowlevel.cpython-312.pyc,, -anyio/__pycache__/pytest_plugin.cpython-312.pyc,, -anyio/__pycache__/to_interpreter.cpython-312.pyc,, -anyio/__pycache__/to_process.cpython-312.pyc,, -anyio/__pycache__/to_thread.cpython-312.pyc,, -anyio/_backends/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -anyio/_backends/__pycache__/__init__.cpython-312.pyc,, -anyio/_backends/__pycache__/_asyncio.cpython-312.pyc,, -anyio/_backends/__pycache__/_trio.cpython-312.pyc,, -anyio/_backends/_asyncio.py,sha256=xG6qv60mgGnL0mK82dxjH2b8hlkMlJ-x2BqIq3qv70Y,98863 -anyio/_backends/_trio.py,sha256=30Rctb7lm8g63ZHljVPVnj5aH-uK6oQvphjwUBoAzuI,41456 -anyio/_core/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -anyio/_core/__pycache__/__init__.cpython-312.pyc,, -anyio/_core/__pycache__/_asyncio_selector_thread.cpython-312.pyc,, -anyio/_core/__pycache__/_contextmanagers.cpython-312.pyc,, -anyio/_core/__pycache__/_eventloop.cpython-312.pyc,, -anyio/_core/__pycache__/_exceptions.cpython-312.pyc,, -anyio/_core/__pycache__/_fileio.cpython-312.pyc,, -anyio/_core/__pycache__/_resources.cpython-312.pyc,, -anyio/_core/__pycache__/_signals.cpython-312.pyc,, -anyio/_core/__pycache__/_sockets.cpython-312.pyc,, -anyio/_core/__pycache__/_streams.cpython-312.pyc,, -anyio/_core/__pycache__/_subprocesses.cpython-312.pyc,, -anyio/_core/__pycache__/_synchronization.cpython-312.pyc,, -anyio/_core/__pycache__/_tasks.cpython-312.pyc,, -anyio/_core/__pycache__/_tempfile.cpython-312.pyc,, -anyio/_core/__pycache__/_testing.cpython-312.pyc,, -anyio/_core/__pycache__/_typedattr.cpython-312.pyc,, -anyio/_core/_asyncio_selector_thread.py,sha256=2PdxFM3cs02Kp6BSppbvmRT7q7asreTW5FgBxEsflBo,5626 -anyio/_core/_contextmanagers.py,sha256=YInBCabiEeS-UaP_Jdxa1CaFC71ETPW8HZTHIM8Rsc8,7215 -anyio/_core/_eventloop.py,sha256=c2EdcBX-xnKwxPcC4Pjn3_qG9I-x4IWFO2R9RqCGjM4,6448 -anyio/_core/_exceptions.py,sha256=Y3aq-Wxd7Q2HqwSg7nZPvRsHEuGazv_qeet6gqEBdPk,4407 -anyio/_core/_fileio.py,sha256=uc7t10Vb-If7GbdWM_zFf-ajUe6uek63fSt7IBLlZW0,25731 -anyio/_core/_resources.py,sha256=NbmU5O5UX3xEyACnkmYX28Fmwdl-f-ny0tHym26e0w0,435 -anyio/_core/_signals.py,sha256=mjTBB2hTKNPRlU0IhnijeQedpWOGERDiMjSlJQsFrug,1016 -anyio/_core/_sockets.py,sha256=RBXHcUqZt5gg_-OOfgHVv8uq2FSKk1uVUzTdpjBoI1o,34977 -anyio/_core/_streams.py,sha256=FczFwIgDpnkK0bODWJXMpsUJYdvAD04kaUaGzJU8DK0,1806 -anyio/_core/_subprocesses.py,sha256=EXm5igL7dj55iYkPlbYVAqtbqxJxjU-6OndSTIx9SRg,8047 -anyio/_core/_synchronization.py,sha256=MgVVqFzvt580tHC31LiOcq1G6aryut--xRG4Ff8KwxQ,20869 -anyio/_core/_tasks.py,sha256=pVB7K6AAulzUM8YgXAeqNZG44nSyZ1bYJjH8GznC00I,5435 -anyio/_core/_tempfile.py,sha256=lHb7CW4FyIlpkf5ADAf4VmLHCKwEHF9nxqNyBCFFUiA,19697 -anyio/_core/_testing.py,sha256=u7MPqGXwpTxqI7hclSdNA30z2GH1Nw258uwKvy_RfBg,2340 -anyio/_core/_typedattr.py,sha256=P4ozZikn3-DbpoYcvyghS_FOYAgbmUxeoU8-L_07pZM,2508 -anyio/abc/__init__.py,sha256=6mWhcl_pGXhrgZVHP_TCfMvIXIOp9mroEFM90fYCU_U,2869 -anyio/abc/__pycache__/__init__.cpython-312.pyc,, -anyio/abc/__pycache__/_eventloop.cpython-312.pyc,, -anyio/abc/__pycache__/_resources.cpython-312.pyc,, -anyio/abc/__pycache__/_sockets.cpython-312.pyc,, -anyio/abc/__pycache__/_streams.cpython-312.pyc,, -anyio/abc/__pycache__/_subprocesses.cpython-312.pyc,, -anyio/abc/__pycache__/_tasks.cpython-312.pyc,, -anyio/abc/__pycache__/_testing.cpython-312.pyc,, -anyio/abc/_eventloop.py,sha256=GlzgB3UJGgG6Kr7olpjOZ-o00PghecXuofVDQ_5611Q,10749 -anyio/abc/_resources.py,sha256=DrYvkNN1hH6Uvv5_5uKySvDsnknGVDe8FCKfko0VtN8,783 -anyio/abc/_sockets.py,sha256=ECTY0jLEF18gryANHR3vFzXzGdZ-xPwELq1QdgOb0Jo,13258 -anyio/abc/_streams.py,sha256=005GKSCXGprxnhucILboSqc2JFovECZk9m3p-qqxXVc,7640 -anyio/abc/_subprocesses.py,sha256=cumAPJTktOQtw63IqG0lDpyZqu_l1EElvQHMiwJgL08,2067 -anyio/abc/_tasks.py,sha256=KC7wrciE48AINOI-AhPutnFhe1ewfP7QnamFlDzqesQ,3721 -anyio/abc/_testing.py,sha256=tBJUzkSfOXJw23fe8qSJ03kJlShOYjjaEyFB6k6MYT8,1821 -anyio/from_thread.py,sha256=L-0w1HxJ6BSb-KuVi57k5Tkc3yzQrx3QK5tAxMPcY-0,19141 -anyio/functools.py,sha256=HWj7GBEmc0Z-mZg3uok7Z7ZJn0rEC_0Pzbt0nYUDaTQ,10973 -anyio/lowlevel.py,sha256=AyKLVK3LaWSoK39LkCKxE4_GDMLKZBNqTrLUgk63y80,5158 -anyio/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -anyio/pytest_plugin.py,sha256=3jAFQn0jv_pyoWE2GBBlHaj9sqXj4e8vob0_hgrsXE8,10244 -anyio/streams/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -anyio/streams/__pycache__/__init__.cpython-312.pyc,, -anyio/streams/__pycache__/buffered.cpython-312.pyc,, -anyio/streams/__pycache__/file.cpython-312.pyc,, -anyio/streams/__pycache__/memory.cpython-312.pyc,, -anyio/streams/__pycache__/stapled.cpython-312.pyc,, -anyio/streams/__pycache__/text.cpython-312.pyc,, -anyio/streams/__pycache__/tls.cpython-312.pyc,, -anyio/streams/buffered.py,sha256=2R3PeJhe4EXrdYqz44Y6-Eg9R6DrmlsYrP36Ir43-po,6263 -anyio/streams/file.py,sha256=4WZ7XGz5WNu39FQHvqbe__TQ0HDP9OOhgO1mk9iVpVU,4470 -anyio/streams/memory.py,sha256=F0zwzvFJKAhX_LRZGoKzzqDC2oMM-f-yyTBrEYEGOaU,10740 -anyio/streams/stapled.py,sha256=T8Xqwf8K6EgURPxbt1N4i7A8BAk-gScv-GRhjLXIf_o,4390 -anyio/streams/text.py,sha256=BcVAGJw1VRvtIqnv-o0Rb0pwH7p8vwlvl21xHq522ag,5765 -anyio/streams/tls.py,sha256=Jpxy0Mfbcp1BxHCwE-YjSSFaLnIBbnnwur-excYThs4,15368 -anyio/to_interpreter.py,sha256=_mLngrMy97TMR6VbW4Y6YzDUk9ZuPcQMPlkuyRh3C9k,7100 -anyio/to_process.py,sha256=J7gAA_YOuoHqnpDAf5fm1Qu6kOmTzdFbiDNvnV755vk,9798 -anyio/to_thread.py,sha256=menEgXYmUV7Fjg_9WqCV95P9MAtQS8BzPGGcWB_QnfQ,2687 diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/WHEEL b/.env/Lib/site-packages/anyio-4.12.1.dist-info/WHEEL deleted file mode 100644 index e7fa31b..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/WHEEL +++ /dev/null @@ -1,5 +0,0 @@ -Wheel-Version: 1.0 -Generator: setuptools (80.9.0) -Root-Is-Purelib: true -Tag: py3-none-any - diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/entry_points.txt b/.env/Lib/site-packages/anyio-4.12.1.dist-info/entry_points.txt deleted file mode 100644 index 44dd9bd..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/entry_points.txt +++ /dev/null @@ -1,2 +0,0 @@ -[pytest11] -anyio = anyio.pytest_plugin diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/licenses/LICENSE b/.env/Lib/site-packages/anyio-4.12.1.dist-info/licenses/LICENSE deleted file mode 100644 index 104eebf..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/licenses/LICENSE +++ /dev/null @@ -1,20 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2018 Alex Grönholm - -Permission is hereby granted, free of charge, to any person obtaining a copy of -this software and associated documentation files (the "Software"), to deal in -the Software without restriction, including without limitation the rights to -use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of -the Software, and to permit persons to whom the Software is furnished to do so, -subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS -FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR -COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER -IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN -CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/.env/Lib/site-packages/anyio-4.12.1.dist-info/top_level.txt b/.env/Lib/site-packages/anyio-4.12.1.dist-info/top_level.txt deleted file mode 100644 index c77c069..0000000 --- a/.env/Lib/site-packages/anyio-4.12.1.dist-info/top_level.txt +++ /dev/null @@ -1 +0,0 @@ -anyio diff --git a/.env/Lib/site-packages/anyio/__init__.py b/.env/Lib/site-packages/anyio/__init__.py deleted file mode 100644 index d23c5a5..0000000 --- a/.env/Lib/site-packages/anyio/__init__.py +++ /dev/null @@ -1,111 +0,0 @@ -from __future__ import annotations - -from ._core._contextmanagers import AsyncContextManagerMixin as AsyncContextManagerMixin -from ._core._contextmanagers import ContextManagerMixin as ContextManagerMixin -from ._core._eventloop import current_time as current_time -from ._core._eventloop import get_all_backends as get_all_backends -from ._core._eventloop import get_available_backends as get_available_backends -from ._core._eventloop import get_cancelled_exc_class as get_cancelled_exc_class -from ._core._eventloop import run as run -from ._core._eventloop import sleep as sleep -from ._core._eventloop import sleep_forever as sleep_forever -from ._core._eventloop import sleep_until as sleep_until -from ._core._exceptions import BrokenResourceError as BrokenResourceError -from ._core._exceptions import BrokenWorkerInterpreter as BrokenWorkerInterpreter -from ._core._exceptions import BrokenWorkerProcess as BrokenWorkerProcess -from ._core._exceptions import BusyResourceError as BusyResourceError -from ._core._exceptions import ClosedResourceError as ClosedResourceError -from ._core._exceptions import ConnectionFailed as ConnectionFailed -from ._core._exceptions import DelimiterNotFound as DelimiterNotFound -from ._core._exceptions import EndOfStream as EndOfStream -from ._core._exceptions import IncompleteRead as IncompleteRead -from ._core._exceptions import NoEventLoopError as NoEventLoopError -from ._core._exceptions import RunFinishedError as RunFinishedError -from ._core._exceptions import TypedAttributeLookupError as TypedAttributeLookupError -from ._core._exceptions import WouldBlock as WouldBlock -from ._core._fileio import AsyncFile as AsyncFile -from ._core._fileio import Path as Path -from ._core._fileio import open_file as open_file -from ._core._fileio import wrap_file as wrap_file -from ._core._resources import aclose_forcefully as aclose_forcefully -from ._core._signals import open_signal_receiver as open_signal_receiver -from ._core._sockets import TCPConnectable as TCPConnectable -from ._core._sockets import UNIXConnectable as UNIXConnectable -from ._core._sockets import as_connectable as as_connectable -from ._core._sockets import connect_tcp as connect_tcp -from ._core._sockets import connect_unix as connect_unix -from ._core._sockets import create_connected_udp_socket as create_connected_udp_socket -from ._core._sockets import ( - create_connected_unix_datagram_socket as create_connected_unix_datagram_socket, -) -from ._core._sockets import create_tcp_listener as create_tcp_listener -from ._core._sockets import create_udp_socket as create_udp_socket -from ._core._sockets import create_unix_datagram_socket as create_unix_datagram_socket -from ._core._sockets import create_unix_listener as create_unix_listener -from ._core._sockets import getaddrinfo as getaddrinfo -from ._core._sockets import getnameinfo as getnameinfo -from ._core._sockets import notify_closing as notify_closing -from ._core._sockets import wait_readable as wait_readable -from ._core._sockets import wait_socket_readable as wait_socket_readable -from ._core._sockets import wait_socket_writable as wait_socket_writable -from ._core._sockets import wait_writable as wait_writable -from ._core._streams import create_memory_object_stream as create_memory_object_stream -from ._core._subprocesses import open_process as open_process -from ._core._subprocesses import run_process as run_process -from ._core._synchronization import CapacityLimiter as CapacityLimiter -from ._core._synchronization import ( - CapacityLimiterStatistics as CapacityLimiterStatistics, -) -from ._core._synchronization import Condition as Condition -from ._core._synchronization import ConditionStatistics as ConditionStatistics -from ._core._synchronization import Event as Event -from ._core._synchronization import EventStatistics as EventStatistics -from ._core._synchronization import Lock as Lock -from ._core._synchronization import LockStatistics as LockStatistics -from ._core._synchronization import ResourceGuard as ResourceGuard -from ._core._synchronization import Semaphore as Semaphore -from ._core._synchronization import SemaphoreStatistics as SemaphoreStatistics -from ._core._tasks import TASK_STATUS_IGNORED as TASK_STATUS_IGNORED -from ._core._tasks import CancelScope as CancelScope -from ._core._tasks import create_task_group as create_task_group -from ._core._tasks import current_effective_deadline as current_effective_deadline -from ._core._tasks import fail_after as fail_after -from ._core._tasks import move_on_after as move_on_after -from ._core._tempfile import NamedTemporaryFile as NamedTemporaryFile -from ._core._tempfile import SpooledTemporaryFile as SpooledTemporaryFile -from ._core._tempfile import TemporaryDirectory as TemporaryDirectory -from ._core._tempfile import TemporaryFile as TemporaryFile -from ._core._tempfile import gettempdir as gettempdir -from ._core._tempfile import gettempdirb as gettempdirb -from ._core._tempfile import mkdtemp as mkdtemp -from ._core._tempfile import mkstemp as mkstemp -from ._core._testing import TaskInfo as TaskInfo -from ._core._testing import get_current_task as get_current_task -from ._core._testing import get_running_tasks as get_running_tasks -from ._core._testing import wait_all_tasks_blocked as wait_all_tasks_blocked -from ._core._typedattr import TypedAttributeProvider as TypedAttributeProvider -from ._core._typedattr import TypedAttributeSet as TypedAttributeSet -from ._core._typedattr import typed_attribute as typed_attribute - -# Re-export imports so they look like they live directly in this package -for __value in list(locals().values()): - if getattr(__value, "__module__", "").startswith("anyio."): - __value.__module__ = __name__ - - -del __value - - -def __getattr__(attr: str) -> type[BrokenWorkerInterpreter]: - """Support deprecated aliases.""" - if attr == "BrokenWorkerIntepreter": - import warnings - - warnings.warn( - "The 'BrokenWorkerIntepreter' alias is deprecated, use 'BrokenWorkerInterpreter' instead.", - DeprecationWarning, - stacklevel=2, - ) - return BrokenWorkerInterpreter - - raise AttributeError(f"module {__name__!r} has no attribute {attr!r}") diff --git a/.env/Lib/site-packages/anyio/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 72485688cb27e34dae8bc79f85ac580b7680ed5a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4622 zcmZ{n$!{FT6~=4GA&2`eZZf5@B#NR$Q5&^O6nBaeH*rXjE>USWr;B8Zy$q^*D3Ndi zBQOF7Fc26y*gz1#hg<^ZkUznPT!I{;#BkzFGJt_32j3LfzW9{yy_qIOMIoqrpZ&z{Y!~96REU1c3lRGzj0^FRu!I$Jgd3Bwn6OH$Qm4c%m8Hya za9qZj%bYT|T$VGJgA+2roB&tI3g!xMrL1JG1Xsx_<|=TttY)qT*T@>?8gQ+wWv+GV z+R0B({^%#Gk?+05JoZjmj_&EP}w5OWK-Rkku80=LOF=2mdK zY-er*cgPOrc5tWcWbObTmWP=;oj2Sr*~NSqd_*2$egk||9%b$VCuNfPi1VgKu1Z$P>&-r`tU#Pcpv=?vXvr$H1rLDdywgUfIih0(@GYX6^=`k!P4sg8O72a}W5e zJj;9v+%Nl?d%@@AIp)*g^YT3N8Sn*pfw>QSQC?&|3%(>TG53Q99)Njb@U2YgrFWgc>-+rwG*l8>0@!K-qW`2qN`e9XMy{J=G&!JGoW zE#GEd1e?-~i4vjK@Gt#oF$O&OeH0A#|3-D6#tODE@%Xq9QlZ_tQr437@YB*l&67Xw zw~EMGj@kvW54zN*QidJXeb5~jkCY{?mG|QE$&c~$^h zE8DhB+cDQ1^>R%c)uc@?tsF;LhI*DZ(vBHWqf(uvVaozXsSKc&!C3dT+EyBd>VhHb zg|>rY^N98_;DQSLtWK+Ot$p%pE0~vky`}UNK1^q{g6_8{RLyHYt$5(Zk!-N@YB$ZJ zjvpxNRdWLt#Zzfo#0h+!ve0egm~w2_#u&4HIN@hKGFFd!*4%n3)XH>Gtef)EzMF9{ zlLcjxL&K~;zKwO9_WevW_lAY6H(`5ru&J!b*0H#gQPyx6YI`jkD!6ZD_Zn0!`&q{t zas2cax~$~goWQhDh|inhCK>&V@{IL@vGS=lGf|<@Ow%{g3a8b2)^T>=-ooyIz2TXT zp;cPhg~hL19Gx$IJMAev>z8JyRt}o#%)m$&7O3b{T`U@5n)b9NDrPlbMIVwbT_y1O-one`wxuH#W-?6#qh*1p34npPehBp??2Re<$q$#|ntCpw@ zRln3%HBPQ+S=#p2eN2stp6RM$u?|u7B&>V?f=@J zYCA0Y7dm*!5KY`wuCI3tf9(klq7g(#tcG_kvv;hkai5I+V*RL@G1GRqGhH~Vm}18( zrW8eoAhgqT9V@Z5mQ4p81&RfQ!U3N$KEbH3XE;z7pDrBAI{JH{rZ&1%lUY;KRj5oU z*UW6x?3nT3@HDyR8fh+v3yc6!0MU@4_S!S$Q#3xM^&K`%{ z=%eUU;o68!4D7gv(`S&4<1imN3eIdX)+kVL%xXOtRS*ofZL4r+4%VdjBG!i#gbXu^ zLb)o#aD6L_;BMr^Zn^30*nWSyn9VpB(aRS(EcVSV;3M%4UcOtWgwV(M-79_=dnfj! znB&;LC{~^J369*l{w_?8wTCBn)*pG3&7Zz}(y#Hx4)`pbBfAAcO{t=v7F z%4V>8VbW4a1!=?^D``5m8K|H?*ZOL@C}ay=g8pWAR~9!_vgh?KJ%vt^cJNZieaUR_ znijk+p2;PY12W5b9NJj6;19oB{$6=>G{dM^-lJd=PI6>nsztH z#ffrFW5v(G6B*fzwDM8HTnnOq#}o3u@A-40=kFD5UsW{xtoEn1U)HSuqvpt$ zZS5}-qO$c{5wEQJwn{YgJg+$Uud4Qsum0-VCw|(UFjtN=pVi)O7oF8dukKlK2Ojrs(b-+k!WAo`=>+Clk$<%6rz}M8Ah+-i z3UnDNHJ!R#L(v2KIX_eu(m#(Ku+9};g z(nQis(n4~Gq?M$N1plEFx`U*XO)^Mw zhh&Ik7?P_gj>Z8|kC1(oWQ=5-WP)T8lBbFA-?+$WhPnIV}anS<6)NtrO<)WeG!-=2Y{b=sPxzAg= z|5AVQ-O0Zt>fV3)am^=dpC@|%ni%*~V&JzcUnB;tU_J09lVQEkR diff --git a/.env/Lib/site-packages/anyio/__pycache__/from_thread.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/from_thread.cpython-312.pyc deleted file mode 100644 index f605a06972dfe695b0e5227c6ca995283bc9817f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 25828 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 diff --git a/.env/Lib/site-packages/anyio/__pycache__/functools.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/functools.cpython-312.pyc deleted file mode 100644 index db17d198d6325b1495e86ce115b3d48bb9a6b20b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15646 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 diff --git a/.env/Lib/site-packages/anyio/__pycache__/pytest_plugin.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/pytest_plugin.cpython-312.pyc deleted file mode 100644 index 99de70cd31311ebc7fa2708bf37dfa5e00b60d71..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14284 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 M9#YQD8G+CL28|#&0RR91 diff --git a/.env/Lib/site-packages/anyio/__pycache__/to_process.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/to_process.cpython-312.pyc deleted file mode 100644 index 480145a2711e6cc77c9b7c19aba4929a2fc5e119..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11955 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 wY2K6`X$o%{ufQ<&UCq5uE@ diff --git a/.env/Lib/site-packages/anyio/__pycache__/to_thread.cpython-312.pyc b/.env/Lib/site-packages/anyio/__pycache__/to_thread.cpython-312.pyc deleted file mode 100644 index 753c6f9756a2a4bc96eadf0d4d9ee0056f599387..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3196 zcmcgu&2t<_74O-Z{m{-YP;*Zr6A@hXAu&mX=Q{uHE-u`&FdOtAJo2jLzGNMJ>zYqczk1cg=s+ID2O zY^a^c=@wf>=qEujDs{`PvXM(sr90Ld1GyYjqH5P|xkes~YTfbHxJ9^q`}lCX{W_?w zE)t@~K${ENCR>x;sn(Q5M8)*f_j?M2mxJ1UD_h%|hW2DIK5EYd6M+rC$=}*zV!hjIUErmRuewPE_Mi)#vLE!VY>^dg>?-V!i*HLWaoH+a0pqO|e@sV}Kzy>O*r zr*+n0B9Ov^Kc5MoN2#0D#%qBe+uu{(nJ;wRabU?-2`t-zfZScBAkUD;~}R z_~QZ$n=i0GaPH$u_$(3^Gl`J5%B`9E+aC!#RGtsAtg_;LOFMr49 z#s4+(w@|C2q{=Z_x*$iLccz9*|*jJPY@VK!{_P8>EOjpkJ=`9DN7YJ??GQDz$vIqPX-C3FhII1!5e64b~ufo2!Lp0)EU@cLjHndWz@}+ z+ekUU0?LC%<`+XvO|Q$>ArRxMQuba~N~$IvLrAKVUQa6HB1~{AN(z}kP~l`6!E8i1 zHz7x1WJ#V~b4&B>On3mXA)YY^F*acWi)laPQJ~FHpNb)KR6a@(dDvgS^X>uw-aY{6 z`8HkUeV_oqndc4kU>gE3l=HF7v-><)YtnxkO+UZ^9YSWMmqh=K@6&J1e`EfYR}BbX zBm`^}3LQfZ+QsAv$-b&%NZ%27C2ADZG<2$29C1_ys&x9boM}MsAHu}}a4Ed4(_=$h zyij-o7+D6+`q?)W7WqKWcuX8`l(0t3lHiSwkx$?23x1k zeRyEhW zkL+Juqe}@!4}Zh4Hvt;vz+Gpk2Nx?Q<}5iLjyEJ=9v8IJ2UeK(z|#RJl5PiXAGu`I z`BFU1Z=ExVwk%gy0?<<_mUboh$7 zrCz|vQ-`M*Qm0Nw%19x#@3<(^4ah=*o5;Cy@1nY?k+`#Bz7YsU7nC4mtKId3lG> zkL=PTd**>X^Zxo@?YVzAFKsz5edc`opYHVgO+az8@;CSBZ>kU7lbe-6jhr~Oed6Ty z>976EJ!6%&tFH`fs6TRF7!;u%l*n_(2W6BhWctOy7)n(#acY3GVRdO@Zycn@Z(CLC IyOuHke}GJ-E&u=k diff --git a/.env/Lib/site-packages/anyio/_backends/__init__.py b/.env/Lib/site-packages/anyio/_backends/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/anyio/_backends/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/anyio/_backends/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 285f74d60284b119685fe66dc4c9349830d8cc0f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 188 zcmX@j%ge<81V;|EWrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxd742*l6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdi&GVn`Kfs+#rpB_nR%Hd@$q^EmA^P_a`RJ4b5iY!Sb=si0&y{j@sXL4k+Fyw G$N~VT>NAA^ diff --git a/.env/Lib/site-packages/anyio/_backends/__pycache__/_asyncio.cpython-312.pyc b/.env/Lib/site-packages/anyio/_backends/__pycache__/_asyncio.cpython-312.pyc deleted file mode 100644 index 2aacbed145e6c29dfa086d0d46fd5c7cdc5aab5d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 138339 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 Kc)<;15;x!$PPqWR?rQaN-}eDJ=LIW}ZaiPj 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 (3, 10): - from typing import ParamSpec -else: - from typing_extensions import ParamSpec - -if sys.version_info >= (3, 11): - from asyncio import Runner - from typing import TypeVarTuple, Unpack -else: - import contextvars - import enum - import signal - from asyncio import coroutines, events, exceptions, tasks - - from exceptiongroup import BaseExceptionGroup - from typing_extensions import TypeVarTuple, Unpack - - class _State(enum.Enum): - CREATED = "created" - INITIALIZED = "initialized" - CLOSED = "closed" - - class Runner: - # Copied from CPython 3.11 - def __init__( - self, - *, - debug: bool | None = None, - loop_factory: Callable[[], AbstractEventLoop] | None = None, - ): - self._state = _State.CREATED - self._debug = debug - self._loop_factory = loop_factory - self._loop: AbstractEventLoop | None = None - self._context = None - self._interrupt_count = 0 - self._set_event_loop = False - - def __enter__(self) -> Runner: - self._lazy_init() - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.close() - - def close(self) -> None: - """Shutdown and close event loop.""" - loop = self._loop - if self._state is not _State.INITIALIZED or loop is None: - return - try: - _cancel_all_tasks(loop) - loop.run_until_complete(loop.shutdown_asyncgens()) - if hasattr(loop, "shutdown_default_executor"): - loop.run_until_complete(loop.shutdown_default_executor()) - else: - loop.run_until_complete(_shutdown_default_executor(loop)) - finally: - if self._set_event_loop: - events.set_event_loop(None) - loop.close() - self._loop = None - self._state = _State.CLOSED - - def get_loop(self) -> AbstractEventLoop: - """Return embedded event loop.""" - self._lazy_init() - return self._loop - - def run(self, coro: Coroutine[T_Retval], *, context=None) -> T_Retval: - """Run a coroutine inside the embedded event loop.""" - if not coroutines.iscoroutine(coro): - raise ValueError(f"a coroutine was expected, got {coro!r}") - - if events._get_running_loop() is not None: - # fail fast with short traceback - raise RuntimeError( - "Runner.run() cannot be called from a running event loop" - ) - - self._lazy_init() - - if context is None: - context = self._context - task = context.run(self._loop.create_task, coro) - - if ( - threading.current_thread() is threading.main_thread() - and signal.getsignal(signal.SIGINT) is signal.default_int_handler - ): - sigint_handler = partial(self._on_sigint, main_task=task) - try: - signal.signal(signal.SIGINT, sigint_handler) - except ValueError: - # `signal.signal` may throw if `threading.main_thread` does - # not support signals (e.g. embedded interpreter with signals - # not registered - see gh-91880) - sigint_handler = None - else: - sigint_handler = None - - self._interrupt_count = 0 - try: - return self._loop.run_until_complete(task) - except exceptions.CancelledError: - if self._interrupt_count > 0: - uncancel = getattr(task, "uncancel", None) - if uncancel is not None and uncancel() == 0: - raise KeyboardInterrupt # noqa: B904 - raise # CancelledError - finally: - if ( - sigint_handler is not None - and signal.getsignal(signal.SIGINT) is sigint_handler - ): - signal.signal(signal.SIGINT, signal.default_int_handler) - - def _lazy_init(self) -> None: - if self._state is _State.CLOSED: - raise RuntimeError("Runner is closed") - if self._state is _State.INITIALIZED: - return - if self._loop_factory is None: - self._loop = events.new_event_loop() - if not self._set_event_loop: - # Call set_event_loop only once to avoid calling - # attach_loop multiple times on child watchers - events.set_event_loop(self._loop) - self._set_event_loop = True - else: - self._loop = self._loop_factory() - if self._debug is not None: - self._loop.set_debug(self._debug) - self._context = contextvars.copy_context() - self._state = _State.INITIALIZED - - def _on_sigint(self, signum, frame, main_task: asyncio.Task) -> None: - self._interrupt_count += 1 - if self._interrupt_count == 1 and not main_task.done(): - main_task.cancel() - # wakeup loop if it is blocked by select() with long timeout - self._loop.call_soon_threadsafe(lambda: None) - return - raise KeyboardInterrupt() - - def _cancel_all_tasks(loop: AbstractEventLoop) -> None: - to_cancel = tasks.all_tasks(loop) - if not to_cancel: - return - - for task in to_cancel: - task.cancel() - - loop.run_until_complete(tasks.gather(*to_cancel, return_exceptions=True)) - - for task in to_cancel: - if task.cancelled(): - continue - if task.exception() is not None: - loop.call_exception_handler( - { - "message": "unhandled exception during asyncio.run() shutdown", - "exception": task.exception(), - "task": task, - } - ) - - async def _shutdown_default_executor(loop: AbstractEventLoop) -> None: - """Schedule the shutdown of the default executor.""" - - def _do_shutdown(future: asyncio.futures.Future) -> None: - try: - loop._default_executor.shutdown(wait=True) # type: ignore[attr-defined] - loop.call_soon_threadsafe(future.set_result, None) - except Exception as ex: - loop.call_soon_threadsafe(future.set_exception, ex) - - loop._executor_shutdown_called = True - if loop._default_executor is None: - return - future = loop.create_future() - thread = threading.Thread(target=_do_shutdown, args=(future,)) - thread.start() - try: - await future - finally: - thread.join() - - -T_Retval = TypeVar("T_Retval") -T_contra = TypeVar("T_contra", contravariant=True) -PosArgsT = TypeVarTuple("PosArgsT") -P = ParamSpec("P") - -_root_task: RunVar[asyncio.Task | None] = RunVar("_root_task") - - -def find_root_task() -> asyncio.Task: - root_task = _root_task.get(None) - if root_task is not None and not root_task.done(): - return root_task - - # Look for a task that has been started via run_until_complete() - for task in all_tasks(): - if task._callbacks and not task.done(): - callbacks = [cb for cb, context in task._callbacks] - for cb in callbacks: - if ( - cb is _run_until_complete_cb - or getattr(cb, "__module__", None) == "uvloop.loop" - ): - _root_task.set(task) - return task - - # Look up the topmost task in the AnyIO task tree, if possible - task = cast(asyncio.Task, current_task()) - state = _task_states.get(task) - if state: - cancel_scope = state.cancel_scope - while cancel_scope and cancel_scope._parent_scope is not None: - cancel_scope = cancel_scope._parent_scope - - if cancel_scope is not None: - return cast(asyncio.Task, cancel_scope._host_task) - - return task - - -def get_callable_name(func: Callable) -> str: - module = getattr(func, "__module__", None) - qualname = getattr(func, "__qualname__", None) - return ".".join([x for x in (module, qualname) if x]) - - -# -# Event loop -# - -_run_vars: WeakKeyDictionary[asyncio.AbstractEventLoop, Any] = WeakKeyDictionary() - - -def _task_started(task: asyncio.Task) -> bool: - """Return ``True`` if the task has been started and has not finished.""" - # The task coro should never be None here, as we never add finished tasks to the - # task list - coro = task.get_coro() - assert coro is not None - try: - return getcoroutinestate(coro) in (CORO_RUNNING, CORO_SUSPENDED) - except AttributeError: - # task coro is async_genenerator_asend https://bugs.python.org/issue37771 - raise Exception(f"Cannot determine if task {task} has started or not") from None - - -# -# Timeouts and cancellation -# - - -def is_anyio_cancellation(exc: CancelledError) -> bool: - # Sometimes third party frameworks catch a CancelledError and raise a new one, so as - # a workaround we have to look at the previous ones in __context__ too for a - # matching cancel message - while True: - if ( - exc.args - and isinstance(exc.args[0], str) - and exc.args[0].startswith("Cancelled via cancel scope ") - ): - return True - - if isinstance(exc.__context__, CancelledError): - exc = exc.__context__ - continue - - return False - - -class CancelScope(BaseCancelScope): - def __new__( - cls, *, deadline: float = math.inf, shield: bool = False - ) -> CancelScope: - return object.__new__(cls) - - def __init__(self, deadline: float = math.inf, shield: bool = False): - self._deadline = deadline - self._shield = shield - self._parent_scope: CancelScope | None = None - self._child_scopes: set[CancelScope] = set() - self._cancel_called = False - self._cancel_reason: str | None = None - self._cancelled_caught = False - self._active = False - self._timeout_handle: asyncio.TimerHandle | None = None - self._cancel_handle: asyncio.Handle | None = None - self._tasks: set[asyncio.Task] = set() - self._host_task: asyncio.Task | None = None - if sys.version_info >= (3, 11): - self._pending_uncancellations: int | None = 0 - else: - self._pending_uncancellations = None - - def __enter__(self) -> CancelScope: - if self._active: - raise RuntimeError( - "Each CancelScope may only be used for a single 'with' block" - ) - - self._host_task = host_task = cast(asyncio.Task, current_task()) - self._tasks.add(host_task) - try: - task_state = _task_states[host_task] - except KeyError: - task_state = TaskState(None, self) - _task_states[host_task] = task_state - else: - self._parent_scope = task_state.cancel_scope - task_state.cancel_scope = self - if self._parent_scope is not None: - # If using an eager task factory, the parent scope may not even contain - # the host task - self._parent_scope._child_scopes.add(self) - self._parent_scope._tasks.discard(host_task) - - self._timeout() - self._active = True - - # Start cancelling the host task if the scope was cancelled before entering - if self._cancel_called: - self._deliver_cancellation(self) - - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - del exc_tb - - if not self._active: - raise RuntimeError("This cancel scope is not active") - if current_task() is not self._host_task: - raise RuntimeError( - "Attempted to exit cancel scope in a different task than it was " - "entered in" - ) - - assert self._host_task is not None - host_task_state = _task_states.get(self._host_task) - if host_task_state is None or host_task_state.cancel_scope is not self: - raise RuntimeError( - "Attempted to exit a cancel scope that isn't the current tasks's " - "current cancel scope" - ) - - try: - self._active = False - if self._timeout_handle: - self._timeout_handle.cancel() - self._timeout_handle = None - - self._tasks.remove(self._host_task) - if self._parent_scope is not None: - self._parent_scope._child_scopes.remove(self) - self._parent_scope._tasks.add(self._host_task) - - host_task_state.cancel_scope = self._parent_scope - - # Restart the cancellation effort in the closest visible, cancelled parent - # scope if necessary - self._restart_cancellation_in_parent() - - # We only swallow the exception iff it was an AnyIO CancelledError, either - # directly as exc_val or inside an exception group and there are no cancelled - # parent cancel scopes visible to us here - if self._cancel_called and not self._parent_cancellation_is_visible_to_us: - # For each level-cancel() call made on the host task, call uncancel() - while self._pending_uncancellations: - self._host_task.uncancel() - self._pending_uncancellations -= 1 - - # Update cancelled_caught and check for exceptions we must not swallow - cannot_swallow_exc_val = False - if exc_val is not None: - for exc in iterate_exceptions(exc_val): - if isinstance(exc, CancelledError) and is_anyio_cancellation( - exc - ): - self._cancelled_caught = True - else: - cannot_swallow_exc_val = True - - return self._cancelled_caught and not cannot_swallow_exc_val - else: - if self._pending_uncancellations: - assert self._parent_scope is not None - assert self._parent_scope._pending_uncancellations is not None - self._parent_scope._pending_uncancellations += ( - self._pending_uncancellations - ) - self._pending_uncancellations = 0 - - return False - finally: - self._host_task = None - del exc_val - - @property - def _effectively_cancelled(self) -> bool: - cancel_scope: CancelScope | None = self - while cancel_scope is not None: - if cancel_scope._cancel_called: - return True - - if cancel_scope.shield: - return False - - cancel_scope = cancel_scope._parent_scope - - return False - - @property - def _parent_cancellation_is_visible_to_us(self) -> bool: - return ( - self._parent_scope is not None - and not self.shield - and self._parent_scope._effectively_cancelled - ) - - def _timeout(self) -> None: - if self._deadline != math.inf: - loop = get_running_loop() - if loop.time() >= self._deadline: - self.cancel("deadline exceeded") - else: - self._timeout_handle = loop.call_at(self._deadline, self._timeout) - - def _deliver_cancellation(self, origin: CancelScope) -> bool: - """ - Deliver cancellation to directly contained tasks and nested cancel scopes. - - Schedule another run at the end if we still have tasks eligible for - cancellation. - - :param origin: the cancel scope that originated the cancellation - :return: ``True`` if the delivery needs to be retried on the next cycle - - """ - should_retry = False - current = current_task() - for task in self._tasks: - should_retry = True - if task._must_cancel: # type: ignore[attr-defined] - continue - - # The task is eligible for cancellation if it has started - if task is not current and (task is self._host_task or _task_started(task)): - waiter = task._fut_waiter # type: ignore[attr-defined] - if not isinstance(waiter, asyncio.Future) or not waiter.done(): - task.cancel(origin._cancel_reason) - if ( - task is origin._host_task - and origin._pending_uncancellations is not None - ): - origin._pending_uncancellations += 1 - - # Deliver cancellation to child scopes that aren't shielded or running their own - # cancellation callbacks - for scope in self._child_scopes: - if not scope._shield and not scope.cancel_called: - should_retry = scope._deliver_cancellation(origin) or should_retry - - # Schedule another callback if there are still tasks left - if origin is self: - if should_retry: - self._cancel_handle = get_running_loop().call_soon( - self._deliver_cancellation, origin - ) - else: - self._cancel_handle = None - - return should_retry - - def _restart_cancellation_in_parent(self) -> None: - """ - Restart the cancellation effort in the closest directly cancelled parent scope. - - """ - scope = self._parent_scope - while scope is not None: - if scope._cancel_called: - if scope._cancel_handle is None: - scope._deliver_cancellation(scope) - - break - - # No point in looking beyond any shielded scope - if scope._shield: - break - - scope = scope._parent_scope - - def cancel(self, reason: str | None = None) -> None: - if not self._cancel_called: - if self._timeout_handle: - self._timeout_handle.cancel() - self._timeout_handle = None - - self._cancel_called = True - self._cancel_reason = f"Cancelled via cancel scope {id(self):x}" - if task := current_task(): - self._cancel_reason += f" by {task}" - - if reason: - self._cancel_reason += f"; reason: {reason}" - - if self._host_task is not None: - self._deliver_cancellation(self) - - @property - def deadline(self) -> float: - return self._deadline - - @deadline.setter - def deadline(self, value: float) -> None: - self._deadline = float(value) - if self._timeout_handle is not None: - self._timeout_handle.cancel() - self._timeout_handle = None - - if self._active and not self._cancel_called: - self._timeout() - - @property - def cancel_called(self) -> bool: - return self._cancel_called - - @property - def cancelled_caught(self) -> bool: - return self._cancelled_caught - - @property - def shield(self) -> bool: - return self._shield - - @shield.setter - def shield(self, value: bool) -> None: - if self._shield != value: - self._shield = value - if not value: - self._restart_cancellation_in_parent() - - -# -# Task states -# - - -class TaskState: - """ - Encapsulates auxiliary task information that cannot be added to the Task instance - itself because there are no guarantees about its implementation. - """ - - __slots__ = "parent_id", "cancel_scope", "__weakref__" - - def __init__(self, parent_id: int | None, cancel_scope: CancelScope | None): - self.parent_id = parent_id - self.cancel_scope = cancel_scope - - -_task_states: WeakKeyDictionary[asyncio.Task, TaskState] = WeakKeyDictionary() - - -# -# Task groups -# - - -class _AsyncioTaskStatus(abc.TaskStatus): - def __init__(self, future: asyncio.Future, parent_id: int): - self._future = future - self._parent_id = parent_id - - def started(self, value: T_contra | None = None) -> None: - try: - self._future.set_result(value) - except asyncio.InvalidStateError: - if not self._future.cancelled(): - raise RuntimeError( - "called 'started' twice on the same task status" - ) from None - - task = cast(asyncio.Task, current_task()) - _task_states[task].parent_id = self._parent_id - - -if sys.version_info >= (3, 12): - _eager_task_factory_code: CodeType | None = asyncio.eager_task_factory.__code__ -else: - _eager_task_factory_code = None - - -class TaskGroup(abc.TaskGroup): - def __init__(self) -> None: - self.cancel_scope: CancelScope = CancelScope() - self._active = False - self._exceptions: list[BaseException] = [] - self._tasks: set[asyncio.Task] = set() - self._on_completed_fut: asyncio.Future[None] | None = None - - async def __aenter__(self) -> TaskGroup: - self.cancel_scope.__enter__() - self._active = True - return self - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - try: - if exc_val is not None: - self.cancel_scope.cancel() - if not isinstance(exc_val, CancelledError): - self._exceptions.append(exc_val) - - loop = get_running_loop() - try: - if self._tasks: - with CancelScope() as wait_scope: - while self._tasks: - self._on_completed_fut = loop.create_future() - - try: - await self._on_completed_fut - except CancelledError as exc: - # Shield the scope against further cancellation attempts, - # as they're not productive (#695) - wait_scope.shield = True - self.cancel_scope.cancel() - - # Set exc_val from the cancellation exception if it was - # previously unset. However, we should not replace a native - # cancellation exception with one raise by a cancel scope. - if exc_val is None or ( - isinstance(exc_val, CancelledError) - and not is_anyio_cancellation(exc) - ): - exc_val = exc - - self._on_completed_fut = None - else: - # If there are no child tasks to wait on, run at least one checkpoint - # anyway - await AsyncIOBackend.cancel_shielded_checkpoint() - - self._active = False - if self._exceptions: - # The exception that got us here should already have been - # added to self._exceptions so it's ok to break exception - # chaining and avoid adding a "During handling of above..." - # for each nesting level. - raise BaseExceptionGroup( - "unhandled errors in a TaskGroup", self._exceptions - ) from None - elif exc_val: - raise exc_val - except BaseException as exc: - if self.cancel_scope.__exit__(type(exc), exc, exc.__traceback__): - return True - - raise - - return self.cancel_scope.__exit__(exc_type, exc_val, exc_tb) - finally: - del exc_val, exc_tb, self._exceptions - - def _spawn( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[Any]], - args: tuple[Unpack[PosArgsT]], - name: object, - task_status_future: asyncio.Future | None = None, - ) -> asyncio.Task: - def task_done(_task: asyncio.Task) -> None: - if sys.version_info >= (3, 14) and self.cancel_scope._host_task is not None: - asyncio.future_discard_from_awaited_by( - _task, self.cancel_scope._host_task - ) - - task_state = _task_states[_task] - assert task_state.cancel_scope is not None - assert _task in task_state.cancel_scope._tasks - task_state.cancel_scope._tasks.remove(_task) - self._tasks.remove(task) - del _task_states[_task] - - if self._on_completed_fut is not None and not self._tasks: - try: - self._on_completed_fut.set_result(None) - except asyncio.InvalidStateError: - pass - - try: - exc = _task.exception() - except CancelledError as e: - while isinstance(e.__context__, CancelledError): - e = e.__context__ - - exc = e - - if exc is not None: - # The future can only be in the cancelled state if the host task was - # cancelled, so return immediately instead of adding one more - # CancelledError to the exceptions list - if task_status_future is not None and task_status_future.cancelled(): - return - - if task_status_future is None or task_status_future.done(): - if not isinstance(exc, CancelledError): - self._exceptions.append(exc) - - if not self.cancel_scope._effectively_cancelled: - self.cancel_scope.cancel() - else: - task_status_future.set_exception(exc) - elif task_status_future is not None and not task_status_future.done(): - task_status_future.set_exception( - RuntimeError("Child exited without calling task_status.started()") - ) - - if not self._active: - raise RuntimeError( - "This task group is not active; no new tasks can be started." - ) - - kwargs = {} - if task_status_future: - parent_id = id(current_task()) - kwargs["task_status"] = _AsyncioTaskStatus( - task_status_future, id(self.cancel_scope._host_task) - ) - else: - parent_id = id(self.cancel_scope._host_task) - - coro = func(*args, **kwargs) - if not iscoroutine(coro): - prefix = f"{func.__module__}." if hasattr(func, "__module__") else "" - raise TypeError( - f"Expected {prefix}{func.__qualname__}() to return a coroutine, but " - f"the return value ({coro!r}) is not a coroutine object" - ) - - name = get_callable_name(func) if name is None else str(name) - loop = asyncio.get_running_loop() - if ( - (factory := loop.get_task_factory()) - and getattr(factory, "__code__", None) is _eager_task_factory_code - and (closure := getattr(factory, "__closure__", None)) - ): - custom_task_constructor = closure[0].cell_contents - task = custom_task_constructor(coro, loop=loop, name=name) - else: - task = create_task(coro, name=name) - - # Make the spawned task inherit the task group's cancel scope - _task_states[task] = TaskState( - parent_id=parent_id, cancel_scope=self.cancel_scope - ) - self.cancel_scope._tasks.add(task) - self._tasks.add(task) - if sys.version_info >= (3, 14) and self.cancel_scope._host_task is not None: - asyncio.future_add_to_awaited_by(task, self.cancel_scope._host_task) - - task.add_done_callback(task_done) - return task - - def start_soon( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[Any]], - *args: Unpack[PosArgsT], - name: object = None, - ) -> None: - self._spawn(func, args, name) - - async def start( - self, func: Callable[..., Awaitable[Any]], *args: object, name: object = None - ) -> Any: - future: asyncio.Future = asyncio.Future() - task = self._spawn(func, args, name, future) - - # If the task raises an exception after sending a start value without a switch - # point between, the task group is cancelled and this method never proceeds to - # process the completed future. That's why we have to have a shielded cancel - # scope here. - try: - return await future - except CancelledError: - # Cancel the task and wait for it to exit before returning - task.cancel() - with CancelScope(shield=True), suppress(CancelledError): - await task - - raise - - -# -# Threads -# - -_Retval_Queue_Type = tuple[Optional[T_Retval], Optional[BaseException]] - - -class WorkerThread(Thread): - MAX_IDLE_TIME = 10 # seconds - - def __init__( - self, - root_task: asyncio.Task, - workers: set[WorkerThread], - idle_workers: deque[WorkerThread], - ): - super().__init__(name="AnyIO worker thread") - self.root_task = root_task - self.workers = workers - self.idle_workers = idle_workers - self.loop = root_task._loop - self.queue: Queue[ - tuple[Context, Callable, tuple, asyncio.Future, CancelScope] | None - ] = Queue(2) - self.idle_since = AsyncIOBackend.current_time() - self.stopping = False - - def _report_result( - self, future: asyncio.Future, result: Any, exc: BaseException | None - ) -> None: - self.idle_since = AsyncIOBackend.current_time() - if not self.stopping: - self.idle_workers.append(self) - - if not future.cancelled(): - if exc is not None: - if isinstance(exc, StopIteration): - new_exc = RuntimeError("coroutine raised StopIteration") - new_exc.__cause__ = exc - exc = new_exc - - future.set_exception(exc) - else: - future.set_result(result) - - def run(self) -> None: - with claim_worker_thread(AsyncIOBackend, self.loop): - while True: - item = self.queue.get() - if item is None: - # Shutdown command received - return - - context, func, args, future, cancel_scope = item - if not future.cancelled(): - result = None - exception: BaseException | None = None - threadlocals.current_cancel_scope = cancel_scope - try: - result = context.run(func, *args) - except BaseException as exc: - exception = exc - finally: - del threadlocals.current_cancel_scope - - if not self.loop.is_closed(): - self.loop.call_soon_threadsafe( - self._report_result, future, result, exception - ) - - del result, exception - - self.queue.task_done() - del item, context, func, args, future, cancel_scope - - def stop(self, f: asyncio.Task | None = None) -> None: - self.stopping = True - self.queue.put_nowait(None) - self.workers.discard(self) - try: - self.idle_workers.remove(self) - except ValueError: - pass - - -_threadpool_idle_workers: RunVar[deque[WorkerThread]] = RunVar( - "_threadpool_idle_workers" -) -_threadpool_workers: RunVar[set[WorkerThread]] = RunVar("_threadpool_workers") - - -# -# Subprocesses -# - - -@dataclass(eq=False) -class StreamReaderWrapper(abc.ByteReceiveStream): - _stream: asyncio.StreamReader - - async def receive(self, max_bytes: int = 65536) -> bytes: - data = await self._stream.read(max_bytes) - if data: - return data - else: - raise EndOfStream - - async def aclose(self) -> None: - self._stream.set_exception(ClosedResourceError()) - await AsyncIOBackend.checkpoint() - - -@dataclass(eq=False) -class StreamWriterWrapper(abc.ByteSendStream): - _stream: asyncio.StreamWriter - _closed: bool = field(init=False, default=False) - - async def send(self, item: bytes) -> None: - await AsyncIOBackend.checkpoint_if_cancelled() - stream_paused = self._stream._protocol._paused # type: ignore[attr-defined] - try: - self._stream.write(item) - await self._stream.drain() - except (ConnectionResetError, BrokenPipeError, RuntimeError) as exc: - # If closed by us and/or the peer: - # * on stdlib, drain() raises ConnectionResetError or BrokenPipeError - # * on uvloop and Winloop, write() eventually starts raising RuntimeError - if self._closed: - raise ClosedResourceError from exc - elif self._stream.is_closing(): - raise BrokenResourceError from exc - - raise - - if not stream_paused: - await AsyncIOBackend.cancel_shielded_checkpoint() - - async def aclose(self) -> None: - self._closed = True - self._stream.close() - await AsyncIOBackend.checkpoint() - - -@dataclass(eq=False) -class Process(abc.Process): - _process: asyncio.subprocess.Process - _stdin: StreamWriterWrapper | None - _stdout: StreamReaderWrapper | None - _stderr: StreamReaderWrapper | None - - async def aclose(self) -> None: - with CancelScope(shield=True) as scope: - if self._stdin: - await self._stdin.aclose() - if self._stdout: - await self._stdout.aclose() - if self._stderr: - await self._stderr.aclose() - - scope.shield = False - try: - await self.wait() - except BaseException: - scope.shield = True - self.kill() - await self.wait() - raise - - async def wait(self) -> int: - return await self._process.wait() - - def terminate(self) -> None: - self._process.terminate() - - def kill(self) -> None: - self._process.kill() - - def send_signal(self, signal: int) -> None: - self._process.send_signal(signal) - - @property - def pid(self) -> int: - return self._process.pid - - @property - def returncode(self) -> int | None: - return self._process.returncode - - @property - def stdin(self) -> abc.ByteSendStream | None: - return self._stdin - - @property - def stdout(self) -> abc.ByteReceiveStream | None: - return self._stdout - - @property - def stderr(self) -> abc.ByteReceiveStream | None: - return self._stderr - - -def _forcibly_shutdown_process_pool_on_exit( - workers: set[Process], _task: object -) -> None: - """ - Forcibly shuts down worker processes belonging to this event loop.""" - child_watcher: asyncio.AbstractChildWatcher | None = None # type: ignore[name-defined] - if sys.version_info < (3, 12): - try: - child_watcher = asyncio.get_event_loop_policy().get_child_watcher() - except NotImplementedError: - pass - - # Close as much as possible (w/o async/await) to avoid warnings - for process in workers.copy(): - if process.returncode is None: - continue - - process._stdin._stream._transport.close() # type: ignore[union-attr] - process._stdout._stream._transport.close() # type: ignore[union-attr] - process._stderr._stream._transport.close() # type: ignore[union-attr] - process.kill() - if child_watcher: - child_watcher.remove_child_handler(process.pid) - - -async def _shutdown_process_pool_on_exit(workers: set[abc.Process]) -> None: - """ - Shuts down worker processes belonging to this event loop. - - NOTE: this only works when the event loop was started using asyncio.run() or - anyio.run(). - - """ - process: abc.Process - try: - await sleep(math.inf) - except asyncio.CancelledError: - workers = workers.copy() - for process in workers: - if process.returncode is None: - process.kill() - - for process in workers: - await process.aclose() - - -# -# Sockets and networking -# - - -class StreamProtocol(asyncio.Protocol): - read_queue: deque[bytes] - read_event: asyncio.Event - write_event: asyncio.Event - exception: Exception | None = None - is_at_eof: bool = False - - def connection_made(self, transport: asyncio.BaseTransport) -> None: - self.read_queue = deque() - self.read_event = asyncio.Event() - self.write_event = asyncio.Event() - self.write_event.set() - cast(asyncio.Transport, transport).set_write_buffer_limits(0) - - def connection_lost(self, exc: Exception | None) -> None: - if exc: - self.exception = BrokenResourceError() - self.exception.__cause__ = exc - - self.read_event.set() - self.write_event.set() - - def data_received(self, data: bytes) -> None: - # ProactorEventloop sometimes sends bytearray instead of bytes - self.read_queue.append(bytes(data)) - self.read_event.set() - - def eof_received(self) -> bool | None: - self.is_at_eof = True - self.read_event.set() - return True - - def pause_writing(self) -> None: - self.write_event = asyncio.Event() - - def resume_writing(self) -> None: - self.write_event.set() - - -class DatagramProtocol(asyncio.DatagramProtocol): - read_queue: deque[tuple[bytes, IPSockAddrType]] - read_event: asyncio.Event - write_event: asyncio.Event - exception: Exception | None = None - - def connection_made(self, transport: asyncio.BaseTransport) -> None: - self.read_queue = deque(maxlen=100) # arbitrary value - self.read_event = asyncio.Event() - self.write_event = asyncio.Event() - self.write_event.set() - - def connection_lost(self, exc: Exception | None) -> None: - self.read_event.set() - self.write_event.set() - - def datagram_received(self, data: bytes, addr: IPSockAddrType) -> None: - addr = convert_ipv6_sockaddr(addr) - self.read_queue.append((data, addr)) - self.read_event.set() - - def error_received(self, exc: Exception) -> None: - self.exception = exc - - def pause_writing(self) -> None: - self.write_event.clear() - - def resume_writing(self) -> None: - self.write_event.set() - - -class SocketStream(abc.SocketStream): - def __init__(self, transport: asyncio.Transport, protocol: StreamProtocol): - self._transport = transport - self._protocol = protocol - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - self._closed = False - - @property - def _raw_socket(self) -> socket.socket: - return self._transport.get_extra_info("socket") - - async def receive(self, max_bytes: int = 65536) -> bytes: - with self._receive_guard: - if ( - not self._protocol.read_event.is_set() - and not self._transport.is_closing() - and not self._protocol.is_at_eof - ): - self._transport.resume_reading() - await self._protocol.read_event.wait() - self._transport.pause_reading() - else: - await AsyncIOBackend.checkpoint() - - try: - chunk = self._protocol.read_queue.popleft() - except IndexError: - if self._closed: - raise ClosedResourceError from None - elif self._protocol.exception: - raise self._protocol.exception from None - else: - raise EndOfStream from None - - if len(chunk) > max_bytes: - # Split the oversized chunk - chunk, leftover = chunk[:max_bytes], chunk[max_bytes:] - self._protocol.read_queue.appendleft(leftover) - - # If the read queue is empty, clear the flag so that the next call will - # block until data is available - if not self._protocol.read_queue: - self._protocol.read_event.clear() - - return chunk - - async def send(self, item: bytes) -> None: - with self._send_guard: - await AsyncIOBackend.checkpoint() - - if self._closed: - raise ClosedResourceError - elif self._protocol.exception is not None: - raise self._protocol.exception - - try: - self._transport.write(item) - except RuntimeError as exc: - if self._transport.is_closing(): - raise BrokenResourceError from exc - else: - raise - - await self._protocol.write_event.wait() - - async def send_eof(self) -> None: - try: - self._transport.write_eof() - except OSError: - pass - - async def aclose(self) -> None: - self._closed = True - if not self._transport.is_closing(): - try: - self._transport.write_eof() - except OSError: - pass - - self._transport.close() - await sleep(0) - self._transport.abort() - - -class _RawSocketMixin: - _receive_future: asyncio.Future | None = None - _send_future: asyncio.Future | None = None - _closing = False - - def __init__(self, raw_socket: socket.socket): - self.__raw_socket = raw_socket - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - @property - def _raw_socket(self) -> socket.socket: - return self.__raw_socket - - def _wait_until_readable(self, loop: asyncio.AbstractEventLoop) -> asyncio.Future: - def callback(f: object) -> None: - del self._receive_future - loop.remove_reader(self.__raw_socket) - - f = self._receive_future = asyncio.Future() - loop.add_reader(self.__raw_socket, f.set_result, None) - f.add_done_callback(callback) - return f - - def _wait_until_writable(self, loop: asyncio.AbstractEventLoop) -> asyncio.Future: - def callback(f: object) -> None: - del self._send_future - loop.remove_writer(self.__raw_socket) - - f = self._send_future = asyncio.Future() - loop.add_writer(self.__raw_socket, f.set_result, None) - f.add_done_callback(callback) - return f - - async def aclose(self) -> None: - if not self._closing: - self._closing = True - if self.__raw_socket.fileno() != -1: - self.__raw_socket.close() - - if self._receive_future: - self._receive_future.set_result(None) - if self._send_future: - self._send_future.set_result(None) - - -class UNIXSocketStream(_RawSocketMixin, abc.UNIXSocketStream): - async def send_eof(self) -> None: - with self._send_guard: - self._raw_socket.shutdown(socket.SHUT_WR) - - async def receive(self, max_bytes: int = 65536) -> bytes: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._receive_guard: - while True: - try: - data = self._raw_socket.recv(max_bytes) - except BlockingIOError: - await self._wait_until_readable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - if not data: - raise EndOfStream - - return data - - async def send(self, item: bytes) -> None: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._send_guard: - view = memoryview(item) - while view: - try: - bytes_sent = self._raw_socket.send(view) - except BlockingIOError: - await self._wait_until_writable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - view = view[bytes_sent:] - - async def receive_fds(self, msglen: int, maxfds: int) -> tuple[bytes, list[int]]: - if not isinstance(msglen, int) or msglen < 0: - raise ValueError("msglen must be a non-negative integer") - if not isinstance(maxfds, int) or maxfds < 1: - raise ValueError("maxfds must be a positive integer") - - loop = get_running_loop() - fds = array.array("i") - await AsyncIOBackend.checkpoint() - with self._receive_guard: - while True: - try: - message, ancdata, flags, addr = self._raw_socket.recvmsg( - msglen, socket.CMSG_LEN(maxfds * fds.itemsize) - ) - except BlockingIOError: - await self._wait_until_readable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - if not message and not ancdata: - raise EndOfStream - - break - - for cmsg_level, cmsg_type, cmsg_data in ancdata: - if cmsg_level != socket.SOL_SOCKET or cmsg_type != socket.SCM_RIGHTS: - raise RuntimeError( - f"Received unexpected ancillary data; message = {message!r}, " - f"cmsg_level = {cmsg_level}, cmsg_type = {cmsg_type}" - ) - - fds.frombytes(cmsg_data[: len(cmsg_data) - (len(cmsg_data) % fds.itemsize)]) - - return message, list(fds) - - async def send_fds(self, message: bytes, fds: Collection[int | IOBase]) -> None: - if not message: - raise ValueError("message must not be empty") - if not fds: - raise ValueError("fds must not be empty") - - loop = get_running_loop() - filenos: list[int] = [] - for fd in fds: - if isinstance(fd, int): - filenos.append(fd) - elif isinstance(fd, IOBase): - filenos.append(fd.fileno()) - - fdarray = array.array("i", filenos) - await AsyncIOBackend.checkpoint() - with self._send_guard: - while True: - try: - # The ignore can be removed after mypy picks up - # https://github.com/python/typeshed/pull/5545 - self._raw_socket.sendmsg( - [message], [(socket.SOL_SOCKET, socket.SCM_RIGHTS, fdarray)] - ) - break - except BlockingIOError: - await self._wait_until_writable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - - -class TCPSocketListener(abc.SocketListener): - _accept_scope: CancelScope | None = None - _closed = False - - def __init__(self, raw_socket: socket.socket): - self.__raw_socket = raw_socket - self._loop = cast(asyncio.BaseEventLoop, get_running_loop()) - self._accept_guard = ResourceGuard("accepting connections from") - - @property - def _raw_socket(self) -> socket.socket: - return self.__raw_socket - - async def accept(self) -> abc.SocketStream: - if self._closed: - raise ClosedResourceError - - with self._accept_guard: - await AsyncIOBackend.checkpoint() - with CancelScope() as self._accept_scope: - try: - client_sock, _addr = await self._loop.sock_accept(self._raw_socket) - except asyncio.CancelledError: - # Workaround for https://bugs.python.org/issue41317 - try: - self._loop.remove_reader(self._raw_socket) - except (ValueError, NotImplementedError): - pass - - if self._closed: - raise ClosedResourceError from None - - raise - finally: - self._accept_scope = None - - client_sock.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) - transport, protocol = await self._loop.connect_accepted_socket( - StreamProtocol, client_sock - ) - return SocketStream(transport, protocol) - - async def aclose(self) -> None: - if self._closed: - return - - self._closed = True - if self._accept_scope: - # Workaround for https://bugs.python.org/issue41317 - try: - self._loop.remove_reader(self._raw_socket) - except (ValueError, NotImplementedError): - pass - - self._accept_scope.cancel() - await sleep(0) - - self._raw_socket.close() - - -class UNIXSocketListener(abc.SocketListener): - def __init__(self, raw_socket: socket.socket): - self.__raw_socket = raw_socket - self._loop = get_running_loop() - self._accept_guard = ResourceGuard("accepting connections from") - self._closed = False - - async def accept(self) -> abc.SocketStream: - await AsyncIOBackend.checkpoint() - with self._accept_guard: - while True: - try: - client_sock, _ = self.__raw_socket.accept() - client_sock.setblocking(False) - return UNIXSocketStream(client_sock) - except BlockingIOError: - f: asyncio.Future = asyncio.Future() - self._loop.add_reader(self.__raw_socket, f.set_result, None) - f.add_done_callback( - lambda _: self._loop.remove_reader(self.__raw_socket) - ) - await f - except OSError as exc: - if self._closed: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - - async def aclose(self) -> None: - self._closed = True - self.__raw_socket.close() - - @property - def _raw_socket(self) -> socket.socket: - return self.__raw_socket - - -class UDPSocket(abc.UDPSocket): - def __init__( - self, transport: asyncio.DatagramTransport, protocol: DatagramProtocol - ): - self._transport = transport - self._protocol = protocol - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - self._closed = False - - @property - def _raw_socket(self) -> socket.socket: - return self._transport.get_extra_info("socket") - - async def aclose(self) -> None: - self._closed = True - if not self._transport.is_closing(): - self._transport.close() - - async def receive(self) -> tuple[bytes, IPSockAddrType]: - with self._receive_guard: - await AsyncIOBackend.checkpoint() - - # If the buffer is empty, ask for more data - if not self._protocol.read_queue and not self._transport.is_closing(): - self._protocol.read_event.clear() - await self._protocol.read_event.wait() - - try: - return self._protocol.read_queue.popleft() - except IndexError: - if self._closed: - raise ClosedResourceError from None - else: - raise BrokenResourceError from None - - async def send(self, item: UDPPacketType) -> None: - with self._send_guard: - await AsyncIOBackend.checkpoint() - await self._protocol.write_event.wait() - if self._closed: - raise ClosedResourceError - elif self._transport.is_closing(): - raise BrokenResourceError - else: - self._transport.sendto(*item) - - -class ConnectedUDPSocket(abc.ConnectedUDPSocket): - def __init__( - self, transport: asyncio.DatagramTransport, protocol: DatagramProtocol - ): - self._transport = transport - self._protocol = protocol - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - self._closed = False - - @property - def _raw_socket(self) -> socket.socket: - return self._transport.get_extra_info("socket") - - async def aclose(self) -> None: - self._closed = True - if not self._transport.is_closing(): - self._transport.close() - - async def receive(self) -> bytes: - with self._receive_guard: - await AsyncIOBackend.checkpoint() - - # If the buffer is empty, ask for more data - if not self._protocol.read_queue and not self._transport.is_closing(): - self._protocol.read_event.clear() - await self._protocol.read_event.wait() - - try: - packet = self._protocol.read_queue.popleft() - except IndexError: - if self._closed: - raise ClosedResourceError from None - else: - raise BrokenResourceError from None - - return packet[0] - - async def send(self, item: bytes) -> None: - with self._send_guard: - await AsyncIOBackend.checkpoint() - await self._protocol.write_event.wait() - if self._closed: - raise ClosedResourceError - elif self._transport.is_closing(): - raise BrokenResourceError - else: - self._transport.sendto(item) - - -class UNIXDatagramSocket(_RawSocketMixin, abc.UNIXDatagramSocket): - async def receive(self) -> UNIXDatagramPacketType: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._receive_guard: - while True: - try: - data = self._raw_socket.recvfrom(65536) - except BlockingIOError: - await self._wait_until_readable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - return data - - async def send(self, item: UNIXDatagramPacketType) -> None: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._send_guard: - while True: - try: - self._raw_socket.sendto(*item) - except BlockingIOError: - await self._wait_until_writable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - return - - -class ConnectedUNIXDatagramSocket(_RawSocketMixin, abc.ConnectedUNIXDatagramSocket): - async def receive(self) -> bytes: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._receive_guard: - while True: - try: - data = self._raw_socket.recv(65536) - except BlockingIOError: - await self._wait_until_readable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - return data - - async def send(self, item: bytes) -> None: - loop = get_running_loop() - await AsyncIOBackend.checkpoint() - with self._send_guard: - while True: - try: - self._raw_socket.send(item) - except BlockingIOError: - await self._wait_until_writable(loop) - except OSError as exc: - if self._closing: - raise ClosedResourceError from None - else: - raise BrokenResourceError from exc - else: - return - - -_read_events: RunVar[dict[int, asyncio.Future[bool]]] = RunVar("read_events") -_write_events: RunVar[dict[int, asyncio.Future[bool]]] = RunVar("write_events") - - -# -# Synchronization -# - - -class Event(BaseEvent): - def __new__(cls) -> Event: - return object.__new__(cls) - - def __init__(self) -> None: - self._event = asyncio.Event() - - def set(self) -> None: - self._event.set() - - def is_set(self) -> bool: - return self._event.is_set() - - async def wait(self) -> None: - if self.is_set(): - await AsyncIOBackend.checkpoint() - else: - await self._event.wait() - - def statistics(self) -> EventStatistics: - return EventStatistics(len(self._event._waiters)) - - -class Lock(BaseLock): - def __new__(cls, *, fast_acquire: bool = False) -> Lock: - return object.__new__(cls) - - def __init__(self, *, fast_acquire: bool = False) -> None: - self._fast_acquire = fast_acquire - self._owner_task: asyncio.Task | None = None - self._waiters: deque[tuple[asyncio.Task, asyncio.Future]] = deque() - - async def acquire(self) -> None: - task = cast(asyncio.Task, current_task()) - if self._owner_task is None and not self._waiters: - await AsyncIOBackend.checkpoint_if_cancelled() - self._owner_task = task - - # Unless on the "fast path", yield control of the event loop so that other - # tasks can run too - if not self._fast_acquire: - try: - await AsyncIOBackend.cancel_shielded_checkpoint() - except CancelledError: - self.release() - raise - - return - - if self._owner_task == task: - raise RuntimeError("Attempted to acquire an already held Lock") - - fut: asyncio.Future[None] = asyncio.Future() - item = task, fut - self._waiters.append(item) - try: - await fut - except CancelledError: - self._waiters.remove(item) - if self._owner_task is task: - self.release() - - raise - - self._waiters.remove(item) - - def acquire_nowait(self) -> None: - task = cast(asyncio.Task, current_task()) - if self._owner_task is None and not self._waiters: - self._owner_task = task - return - - if self._owner_task is task: - raise RuntimeError("Attempted to acquire an already held Lock") - - raise WouldBlock - - def locked(self) -> bool: - return self._owner_task is not None - - def release(self) -> None: - if self._owner_task != current_task(): - raise RuntimeError("The current task is not holding this lock") - - for task, fut in self._waiters: - if not fut.cancelled(): - self._owner_task = task - fut.set_result(None) - return - - self._owner_task = None - - def statistics(self) -> LockStatistics: - task_info = AsyncIOTaskInfo(self._owner_task) if self._owner_task else None - return LockStatistics(self.locked(), task_info, len(self._waiters)) - - -class Semaphore(BaseSemaphore): - def __new__( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> Semaphore: - return object.__new__(cls) - - def __init__( - self, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ): - super().__init__(initial_value, max_value=max_value) - self._value = initial_value - self._max_value = max_value - self._fast_acquire = fast_acquire - self._waiters: deque[asyncio.Future[None]] = deque() - - async def acquire(self) -> None: - if self._value > 0 and not self._waiters: - await AsyncIOBackend.checkpoint_if_cancelled() - self._value -= 1 - - # Unless on the "fast path", yield control of the event loop so that other - # tasks can run too - if not self._fast_acquire: - try: - await AsyncIOBackend.cancel_shielded_checkpoint() - except CancelledError: - self.release() - raise - - return - - fut: asyncio.Future[None] = asyncio.Future() - self._waiters.append(fut) - try: - await fut - except CancelledError: - try: - self._waiters.remove(fut) - except ValueError: - self.release() - - raise - - def acquire_nowait(self) -> None: - if self._value == 0: - raise WouldBlock - - self._value -= 1 - - def release(self) -> None: - if self._max_value is not None and self._value == self._max_value: - raise ValueError("semaphore released too many times") - - for fut in self._waiters: - if not fut.cancelled(): - fut.set_result(None) - self._waiters.remove(fut) - return - - self._value += 1 - - @property - def value(self) -> int: - return self._value - - @property - def max_value(self) -> int | None: - return self._max_value - - def statistics(self) -> SemaphoreStatistics: - return SemaphoreStatistics(len(self._waiters)) - - -class CapacityLimiter(BaseCapacityLimiter): - _total_tokens: float = 0 - - def __new__(cls, total_tokens: float) -> CapacityLimiter: - return object.__new__(cls) - - def __init__(self, total_tokens: float): - self._borrowers: set[Any] = set() - self._wait_queue: OrderedDict[Any, asyncio.Event] = OrderedDict() - self.total_tokens = total_tokens - - async def __aenter__(self) -> None: - await self.acquire() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.release() - - @property - def total_tokens(self) -> float: - return self._total_tokens - - @total_tokens.setter - def total_tokens(self, value: float) -> None: - if not isinstance(value, int) and not math.isinf(value): - raise TypeError("total_tokens must be an int or math.inf") - - if value < 0: - raise ValueError("total_tokens must be >= 0") - - waiters_to_notify = max(value - self._total_tokens, 0) - self._total_tokens = value - - # Notify waiting tasks that they have acquired the limiter - while self._wait_queue and waiters_to_notify: - event = self._wait_queue.popitem(last=False)[1] - event.set() - waiters_to_notify -= 1 - - @property - def borrowed_tokens(self) -> int: - return len(self._borrowers) - - @property - def available_tokens(self) -> float: - return self._total_tokens - len(self._borrowers) - - def _notify_next_waiter(self) -> None: - """Notify the next task in line if this limiter has free capacity now.""" - if self._wait_queue and len(self._borrowers) < self._total_tokens: - event = self._wait_queue.popitem(last=False)[1] - event.set() - - def acquire_nowait(self) -> None: - self.acquire_on_behalf_of_nowait(current_task()) - - def acquire_on_behalf_of_nowait(self, borrower: object) -> None: - if borrower in self._borrowers: - raise RuntimeError( - "this borrower is already holding one of this CapacityLimiter's tokens" - ) - - if self._wait_queue or len(self._borrowers) >= self._total_tokens: - raise WouldBlock - - self._borrowers.add(borrower) - - async def acquire(self) -> None: - return await self.acquire_on_behalf_of(current_task()) - - async def acquire_on_behalf_of(self, borrower: object) -> None: - await AsyncIOBackend.checkpoint_if_cancelled() - try: - self.acquire_on_behalf_of_nowait(borrower) - except WouldBlock: - event = asyncio.Event() - self._wait_queue[borrower] = event - try: - await event.wait() - except BaseException: - self._wait_queue.pop(borrower, None) - if event.is_set(): - self._notify_next_waiter() - - raise - - self._borrowers.add(borrower) - else: - try: - await AsyncIOBackend.cancel_shielded_checkpoint() - except BaseException: - self.release() - raise - - def release(self) -> None: - self.release_on_behalf_of(current_task()) - - def release_on_behalf_of(self, borrower: object) -> None: - try: - self._borrowers.remove(borrower) - except KeyError: - raise RuntimeError( - "this borrower isn't holding any of this CapacityLimiter's tokens" - ) from None - - self._notify_next_waiter() - - def statistics(self) -> CapacityLimiterStatistics: - return CapacityLimiterStatistics( - self.borrowed_tokens, - self.total_tokens, - tuple(self._borrowers), - len(self._wait_queue), - ) - - -_default_thread_limiter: RunVar[CapacityLimiter] = RunVar("_default_thread_limiter") - - -# -# Operating system signals -# - - -class _SignalReceiver: - def __init__(self, signals: tuple[Signals, ...]): - self._signals = signals - self._loop = get_running_loop() - self._signal_queue: deque[Signals] = deque() - self._future: asyncio.Future = asyncio.Future() - self._handled_signals: set[Signals] = set() - - def _deliver(self, signum: Signals) -> None: - self._signal_queue.append(signum) - if not self._future.done(): - self._future.set_result(None) - - def __enter__(self) -> _SignalReceiver: - for sig in set(self._signals): - self._loop.add_signal_handler(sig, self._deliver, sig) - self._handled_signals.add(sig) - - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - for sig in self._handled_signals: - self._loop.remove_signal_handler(sig) - - def __aiter__(self) -> _SignalReceiver: - return self - - async def __anext__(self) -> Signals: - await AsyncIOBackend.checkpoint() - if not self._signal_queue: - self._future = asyncio.Future() - await self._future - - return self._signal_queue.popleft() - - -# -# Testing and debugging -# - - -class AsyncIOTaskInfo(TaskInfo): - def __init__(self, task: asyncio.Task): - task_state = _task_states.get(task) - if task_state is None: - parent_id = None - else: - parent_id = task_state.parent_id - - coro = task.get_coro() - assert coro is not None, "created TaskInfo from a completed Task" - super().__init__(id(task), parent_id, task.get_name(), coro) - self._task = weakref.ref(task) - - def has_pending_cancellation(self) -> bool: - if not (task := self._task()): - # If the task isn't around anymore, it won't have a pending cancellation - return False - - if task._must_cancel: # type: ignore[attr-defined] - return True - elif ( - isinstance(task._fut_waiter, asyncio.Future) # type: ignore[attr-defined] - and task._fut_waiter.cancelled() # type: ignore[attr-defined] - ): - return True - - if task_state := _task_states.get(task): - if cancel_scope := task_state.cancel_scope: - return cancel_scope._effectively_cancelled - - return False - - -class TestRunner(abc.TestRunner): - _send_stream: MemoryObjectSendStream[tuple[Awaitable[Any], asyncio.Future[Any]]] - - def __init__( - self, - *, - debug: bool | None = None, - use_uvloop: bool = False, - loop_factory: Callable[[], AbstractEventLoop] | None = None, - ) -> None: - if use_uvloop and loop_factory is None: - if sys.platform != "win32": - import uvloop - - loop_factory = uvloop.new_event_loop - else: - import winloop - - loop_factory = winloop.new_event_loop - - self._runner = Runner(debug=debug, loop_factory=loop_factory) - self._exceptions: list[BaseException] = [] - self._runner_task: asyncio.Task | None = None - - def __enter__(self) -> TestRunner: - self._runner.__enter__() - self.get_loop().set_exception_handler(self._exception_handler) - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self._runner.__exit__(exc_type, exc_val, exc_tb) - - def get_loop(self) -> AbstractEventLoop: - return self._runner.get_loop() - - def _exception_handler( - self, loop: asyncio.AbstractEventLoop, context: dict[str, Any] - ) -> None: - if isinstance(context.get("exception"), Exception): - self._exceptions.append(context["exception"]) - else: - loop.default_exception_handler(context) - - def _raise_async_exceptions(self) -> None: - # Re-raise any exceptions raised in asynchronous callbacks - if self._exceptions: - exceptions, self._exceptions = self._exceptions, [] - if len(exceptions) == 1: - raise exceptions[0] - elif exceptions: - raise BaseExceptionGroup( - "Multiple exceptions occurred in asynchronous callbacks", exceptions - ) - - async def _run_tests_and_fixtures( - self, - receive_stream: MemoryObjectReceiveStream[ - tuple[Awaitable[T_Retval], asyncio.Future[T_Retval]] - ], - ) -> None: - from _pytest.outcomes import OutcomeException - - with receive_stream, self._send_stream: - async for coro, future in receive_stream: - try: - retval = await coro - except CancelledError as exc: - if not future.cancelled(): - future.cancel(*exc.args) - - raise - except BaseException as exc: - if not future.cancelled(): - future.set_exception(exc) - - if not isinstance(exc, (Exception, OutcomeException)): - raise - else: - if not future.cancelled(): - future.set_result(retval) - - async def _call_in_runner_task( - self, - func: Callable[P, Awaitable[T_Retval]], - *args: P.args, - **kwargs: P.kwargs, - ) -> T_Retval: - if not self._runner_task: - self._send_stream, receive_stream = create_memory_object_stream[ - tuple[Awaitable[Any], asyncio.Future] - ](1) - self._runner_task = self.get_loop().create_task( - self._run_tests_and_fixtures(receive_stream) - ) - - coro = func(*args, **kwargs) - future: asyncio.Future[T_Retval] = self.get_loop().create_future() - self._send_stream.send_nowait((coro, future)) - return await future - - def run_asyncgen_fixture( - self, - fixture_func: Callable[..., AsyncGenerator[T_Retval, Any]], - kwargs: dict[str, Any], - ) -> Iterable[T_Retval]: - asyncgen = fixture_func(**kwargs) - fixturevalue: T_Retval = self.get_loop().run_until_complete( - self._call_in_runner_task(asyncgen.asend, None) - ) - self._raise_async_exceptions() - - yield fixturevalue - - try: - self.get_loop().run_until_complete( - self._call_in_runner_task(asyncgen.asend, None) - ) - except StopAsyncIteration: - self._raise_async_exceptions() - else: - self.get_loop().run_until_complete(asyncgen.aclose()) - raise RuntimeError("Async generator fixture did not stop") - - def run_fixture( - self, - fixture_func: Callable[..., Coroutine[Any, Any, T_Retval]], - kwargs: dict[str, Any], - ) -> T_Retval: - retval = self.get_loop().run_until_complete( - self._call_in_runner_task(fixture_func, **kwargs) - ) - self._raise_async_exceptions() - return retval - - def run_test( - self, test_func: Callable[..., Coroutine[Any, Any, Any]], kwargs: dict[str, Any] - ) -> None: - try: - self.get_loop().run_until_complete( - self._call_in_runner_task(test_func, **kwargs) - ) - except Exception as exc: - self._exceptions.append(exc) - - self._raise_async_exceptions() - - -class AsyncIOBackend(AsyncBackend): - @classmethod - def run( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - kwargs: dict[str, Any], - options: dict[str, Any], - ) -> T_Retval: - @wraps(func) - async def wrapper() -> T_Retval: - task = cast(asyncio.Task, current_task()) - task.set_name(get_callable_name(func)) - _task_states[task] = TaskState(None, None) - - try: - return await func(*args) - finally: - del _task_states[task] - - debug = options.get("debug", None) - loop_factory = options.get("loop_factory", None) - if loop_factory is None and options.get("use_uvloop", False): - if sys.platform != "win32": - import uvloop - - loop_factory = uvloop.new_event_loop - else: - import winloop - - loop_factory = winloop.new_event_loop - - with Runner(debug=debug, loop_factory=loop_factory) as runner: - return runner.run(wrapper()) - - @classmethod - def current_token(cls) -> object: - return get_running_loop() - - @classmethod - def current_time(cls) -> float: - return get_running_loop().time() - - @classmethod - def cancelled_exception_class(cls) -> type[BaseException]: - return CancelledError - - @classmethod - async def checkpoint(cls) -> None: - await sleep(0) - - @classmethod - async def checkpoint_if_cancelled(cls) -> None: - task = current_task() - if task is None: - return - - try: - cancel_scope = _task_states[task].cancel_scope - except KeyError: - return - - while cancel_scope: - if cancel_scope.cancel_called: - await sleep(0) - elif cancel_scope.shield: - break - else: - cancel_scope = cancel_scope._parent_scope - - @classmethod - async def cancel_shielded_checkpoint(cls) -> None: - with CancelScope(shield=True): - await sleep(0) - - @classmethod - async def sleep(cls, delay: float) -> None: - await sleep(delay) - - @classmethod - def create_cancel_scope( - cls, *, deadline: float = math.inf, shield: bool = False - ) -> CancelScope: - return CancelScope(deadline=deadline, shield=shield) - - @classmethod - def current_effective_deadline(cls) -> float: - if (task := current_task()) is None: - return math.inf - - try: - cancel_scope = _task_states[task].cancel_scope - except KeyError: - return math.inf - - deadline = math.inf - while cancel_scope: - deadline = min(deadline, cancel_scope.deadline) - if cancel_scope._cancel_called: - deadline = -math.inf - break - elif cancel_scope.shield: - break - else: - cancel_scope = cancel_scope._parent_scope - - return deadline - - @classmethod - def create_task_group(cls) -> abc.TaskGroup: - return TaskGroup() - - @classmethod - def create_event(cls) -> abc.Event: - return Event() - - @classmethod - def create_lock(cls, *, fast_acquire: bool) -> abc.Lock: - return Lock(fast_acquire=fast_acquire) - - @classmethod - def create_semaphore( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> abc.Semaphore: - return Semaphore(initial_value, max_value=max_value, fast_acquire=fast_acquire) - - @classmethod - def create_capacity_limiter(cls, total_tokens: float) -> abc.CapacityLimiter: - return CapacityLimiter(total_tokens) - - @classmethod - async def run_sync_in_worker_thread( # type: ignore[return] - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - abandon_on_cancel: bool = False, - limiter: abc.CapacityLimiter | None = None, - ) -> T_Retval: - await cls.checkpoint() - - # If this is the first run in this event loop thread, set up the necessary - # variables - try: - idle_workers = _threadpool_idle_workers.get() - workers = _threadpool_workers.get() - except LookupError: - idle_workers = deque() - workers = set() - _threadpool_idle_workers.set(idle_workers) - _threadpool_workers.set(workers) - - async with limiter or cls.current_default_thread_limiter(): - with CancelScope(shield=not abandon_on_cancel) as scope: - future = asyncio.Future[T_Retval]() - root_task = find_root_task() - if not idle_workers: - worker = WorkerThread(root_task, workers, idle_workers) - worker.start() - workers.add(worker) - root_task.add_done_callback( - worker.stop, context=contextvars.Context() - ) - else: - worker = idle_workers.pop() - - # Prune any other workers that have been idle for MAX_IDLE_TIME - # seconds or longer - now = cls.current_time() - while idle_workers: - if ( - now - idle_workers[0].idle_since - < WorkerThread.MAX_IDLE_TIME - ): - break - - expired_worker = idle_workers.popleft() - expired_worker.root_task.remove_done_callback( - expired_worker.stop - ) - expired_worker.stop() - - context = copy_context() - context.run(set_current_async_library, None) - if abandon_on_cancel or scope._parent_scope is None: - worker_scope = scope - else: - worker_scope = scope._parent_scope - - worker.queue.put_nowait((context, func, args, future, worker_scope)) - return await future - - @classmethod - def check_cancelled(cls) -> None: - scope: CancelScope | None = threadlocals.current_cancel_scope - while scope is not None: - if scope.cancel_called: - raise CancelledError(f"Cancelled by cancel scope {id(scope):x}") - - if scope.shield: - return - - scope = scope._parent_scope - - @classmethod - def run_async_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - async def task_wrapper() -> T_Retval: - __tracebackhide__ = True - if scope is not None: - task = cast(asyncio.Task, current_task()) - _task_states[task] = TaskState(None, scope) - scope._tasks.add(task) - try: - return await func(*args) - except CancelledError as exc: - raise concurrent.futures.CancelledError(str(exc)) from None - finally: - if scope is not None: - scope._tasks.discard(task) - - loop = cast( - "AbstractEventLoop", token or threadlocals.current_token.native_token - ) - if loop.is_closed(): - raise RunFinishedError - - context = copy_context() - context.run(set_current_async_library, "asyncio") - scope = getattr(threadlocals, "current_cancel_scope", None) - f: concurrent.futures.Future[T_Retval] = context.run( - asyncio.run_coroutine_threadsafe, task_wrapper(), loop=loop - ) - return f.result() - - @classmethod - def run_sync_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - @wraps(func) - def wrapper() -> None: - try: - set_current_async_library("asyncio") - f.set_result(func(*args)) - except BaseException as exc: - f.set_exception(exc) - if not isinstance(exc, Exception): - raise - - loop = cast( - "AbstractEventLoop", token or threadlocals.current_token.native_token - ) - if loop.is_closed(): - raise RunFinishedError - - f: concurrent.futures.Future[T_Retval] = Future() - loop.call_soon_threadsafe(wrapper) - return f.result() - - @classmethod - async def open_process( - cls, - command: StrOrBytesPath | Sequence[StrOrBytesPath], - *, - stdin: int | IO[Any] | None, - stdout: int | IO[Any] | None, - stderr: int | IO[Any] | None, - **kwargs: Any, - ) -> Process: - await cls.checkpoint() - if isinstance(command, PathLike): - command = os.fspath(command) - - if isinstance(command, (str, bytes)): - process = await asyncio.create_subprocess_shell( - command, - stdin=stdin, - stdout=stdout, - stderr=stderr, - **kwargs, - ) - else: - process = await asyncio.create_subprocess_exec( - *command, - stdin=stdin, - stdout=stdout, - stderr=stderr, - **kwargs, - ) - - stdin_stream = StreamWriterWrapper(process.stdin) if process.stdin else None - stdout_stream = StreamReaderWrapper(process.stdout) if process.stdout else None - stderr_stream = StreamReaderWrapper(process.stderr) if process.stderr else None - return Process(process, stdin_stream, stdout_stream, stderr_stream) - - @classmethod - def setup_process_pool_exit_at_shutdown(cls, workers: set[abc.Process]) -> None: - create_task( - _shutdown_process_pool_on_exit(workers), - name="AnyIO process pool shutdown task", - ) - find_root_task().add_done_callback( - partial(_forcibly_shutdown_process_pool_on_exit, workers) # type:ignore[arg-type] - ) - - @classmethod - async def connect_tcp( - cls, host: str, port: int, local_address: IPSockAddrType | None = None - ) -> abc.SocketStream: - transport, protocol = cast( - tuple[asyncio.Transport, StreamProtocol], - await get_running_loop().create_connection( - StreamProtocol, host, port, local_addr=local_address - ), - ) - transport.pause_reading() - return SocketStream(transport, protocol) - - @classmethod - async def connect_unix(cls, path: str | bytes) -> abc.UNIXSocketStream: - await cls.checkpoint() - loop = get_running_loop() - raw_socket = socket.socket(socket.AF_UNIX) - raw_socket.setblocking(False) - while True: - try: - raw_socket.connect(path) - except BlockingIOError: - f: asyncio.Future = asyncio.Future() - loop.add_writer(raw_socket, f.set_result, None) - f.add_done_callback(lambda _: loop.remove_writer(raw_socket)) - await f - except BaseException: - raw_socket.close() - raise - else: - return UNIXSocketStream(raw_socket) - - @classmethod - def create_tcp_listener(cls, sock: socket.socket) -> SocketListener: - return TCPSocketListener(sock) - - @classmethod - def create_unix_listener(cls, sock: socket.socket) -> SocketListener: - return UNIXSocketListener(sock) - - @classmethod - async def create_udp_socket( - cls, - family: AddressFamily, - local_address: IPSockAddrType | None, - remote_address: IPSockAddrType | None, - reuse_port: bool, - ) -> UDPSocket | ConnectedUDPSocket: - transport, protocol = await get_running_loop().create_datagram_endpoint( - DatagramProtocol, - local_addr=local_address, - remote_addr=remote_address, - family=family, - reuse_port=reuse_port, - ) - if protocol.exception: - transport.close() - raise protocol.exception - - if not remote_address: - return UDPSocket(transport, protocol) - else: - return ConnectedUDPSocket(transport, protocol) - - @classmethod - async def create_unix_datagram_socket( # type: ignore[override] - cls, raw_socket: socket.socket, remote_path: str | bytes | None - ) -> abc.UNIXDatagramSocket | abc.ConnectedUNIXDatagramSocket: - await cls.checkpoint() - loop = get_running_loop() - - if remote_path: - while True: - try: - raw_socket.connect(remote_path) - except BlockingIOError: - f: asyncio.Future = asyncio.Future() - loop.add_writer(raw_socket, f.set_result, None) - f.add_done_callback(lambda _: loop.remove_writer(raw_socket)) - await f - except BaseException: - raw_socket.close() - raise - else: - return ConnectedUNIXDatagramSocket(raw_socket) - else: - return UNIXDatagramSocket(raw_socket) - - @classmethod - async def getaddrinfo( - cls, - host: bytes | str | None, - port: str | int | None, - *, - family: int | AddressFamily = 0, - type: int | SocketKind = 0, - proto: int = 0, - flags: int = 0, - ) -> Sequence[ - tuple[ - AddressFamily, - SocketKind, - int, - str, - tuple[str, int] | tuple[str, int, int, int] | tuple[int, bytes], - ] - ]: - return await get_running_loop().getaddrinfo( - host, port, family=family, type=type, proto=proto, flags=flags - ) - - @classmethod - async def getnameinfo( - cls, sockaddr: IPSockAddrType, flags: int = 0 - ) -> tuple[str, str]: - return await get_running_loop().getnameinfo(sockaddr, flags) - - @classmethod - async def wait_readable(cls, obj: FileDescriptorLike) -> None: - try: - read_events = _read_events.get() - except LookupError: - read_events = {} - _read_events.set(read_events) - - fd = obj if isinstance(obj, int) else obj.fileno() - if read_events.get(fd): - raise BusyResourceError("reading from") - - loop = get_running_loop() - fut: asyncio.Future[bool] = loop.create_future() - - def cb() -> None: - try: - del read_events[fd] - except KeyError: - pass - else: - remove_reader(fd) - - try: - fut.set_result(True) - except asyncio.InvalidStateError: - pass - - try: - loop.add_reader(fd, cb) - except NotImplementedError: - from anyio._core._asyncio_selector_thread import get_selector - - selector = get_selector() - selector.add_reader(fd, cb) - remove_reader = selector.remove_reader - else: - remove_reader = loop.remove_reader - - read_events[fd] = fut - try: - success = await fut - finally: - try: - del read_events[fd] - except KeyError: - pass - else: - remove_reader(fd) - - if not success: - raise ClosedResourceError - - @classmethod - async def wait_writable(cls, obj: FileDescriptorLike) -> None: - try: - write_events = _write_events.get() - except LookupError: - write_events = {} - _write_events.set(write_events) - - fd = obj if isinstance(obj, int) else obj.fileno() - if write_events.get(fd): - raise BusyResourceError("writing to") - - loop = get_running_loop() - fut: asyncio.Future[bool] = loop.create_future() - - def cb() -> None: - try: - del write_events[fd] - except KeyError: - pass - else: - remove_writer(fd) - - try: - fut.set_result(True) - except asyncio.InvalidStateError: - pass - - try: - loop.add_writer(fd, cb) - except NotImplementedError: - from anyio._core._asyncio_selector_thread import get_selector - - selector = get_selector() - selector.add_writer(fd, cb) - remove_writer = selector.remove_writer - else: - remove_writer = loop.remove_writer - - write_events[fd] = fut - try: - success = await fut - finally: - try: - del write_events[fd] - except KeyError: - pass - else: - remove_writer(fd) - - if not success: - raise ClosedResourceError - - @classmethod - def notify_closing(cls, obj: FileDescriptorLike) -> None: - fd = obj if isinstance(obj, int) else obj.fileno() - loop = get_running_loop() - - try: - write_events = _write_events.get() - except LookupError: - pass - else: - try: - fut = write_events.pop(fd) - except KeyError: - pass - else: - try: - fut.set_result(False) - except asyncio.InvalidStateError: - pass - - try: - loop.remove_writer(fd) - except NotImplementedError: - from anyio._core._asyncio_selector_thread import get_selector - - get_selector().remove_writer(fd) - - try: - read_events = _read_events.get() - except LookupError: - pass - else: - try: - fut = read_events.pop(fd) - except KeyError: - pass - else: - try: - fut.set_result(False) - except asyncio.InvalidStateError: - pass - - try: - loop.remove_reader(fd) - except NotImplementedError: - from anyio._core._asyncio_selector_thread import get_selector - - get_selector().remove_reader(fd) - - @classmethod - async def wrap_listener_socket(cls, sock: socket.socket) -> SocketListener: - return TCPSocketListener(sock) - - @classmethod - async def wrap_stream_socket(cls, sock: socket.socket) -> SocketStream: - transport, protocol = await get_running_loop().create_connection( - StreamProtocol, sock=sock - ) - return SocketStream(transport, protocol) - - @classmethod - async def wrap_unix_stream_socket(cls, sock: socket.socket) -> UNIXSocketStream: - return UNIXSocketStream(sock) - - @classmethod - async def wrap_udp_socket(cls, sock: socket.socket) -> UDPSocket: - transport, protocol = await get_running_loop().create_datagram_endpoint( - DatagramProtocol, sock=sock - ) - return UDPSocket(transport, protocol) - - @classmethod - async def wrap_connected_udp_socket(cls, sock: socket.socket) -> ConnectedUDPSocket: - transport, protocol = await get_running_loop().create_datagram_endpoint( - DatagramProtocol, sock=sock - ) - return ConnectedUDPSocket(transport, protocol) - - @classmethod - async def wrap_unix_datagram_socket(cls, sock: socket.socket) -> UNIXDatagramSocket: - return UNIXDatagramSocket(sock) - - @classmethod - async def wrap_connected_unix_datagram_socket( - cls, sock: socket.socket - ) -> ConnectedUNIXDatagramSocket: - return ConnectedUNIXDatagramSocket(sock) - - @classmethod - def current_default_thread_limiter(cls) -> CapacityLimiter: - try: - return _default_thread_limiter.get() - except LookupError: - limiter = CapacityLimiter(40) - _default_thread_limiter.set(limiter) - return limiter - - @classmethod - def open_signal_receiver( - cls, *signals: Signals - ) -> AbstractContextManager[AsyncIterator[Signals]]: - return _SignalReceiver(signals) - - @classmethod - def get_current_task(cls) -> TaskInfo: - return AsyncIOTaskInfo(current_task()) # type: ignore[arg-type] - - @classmethod - def get_running_tasks(cls) -> Sequence[TaskInfo]: - return [AsyncIOTaskInfo(task) for task in all_tasks() if not task.done()] - - @classmethod - async def wait_all_tasks_blocked(cls) -> None: - await cls.checkpoint() - this_task = current_task() - while True: - for task in all_tasks(): - if task is this_task: - continue - - waiter = task._fut_waiter # type: ignore[attr-defined] - if waiter is None or waiter.done(): - await sleep(0.1) - break - else: - return - - @classmethod - def create_test_runner(cls, options: dict[str, Any]) -> TestRunner: - return TestRunner(**options) - - -backend_class = AsyncIOBackend diff --git a/.env/Lib/site-packages/anyio/_backends/_trio.py b/.env/Lib/site-packages/anyio/_backends/_trio.py deleted file mode 100644 index f460a7f..0000000 --- a/.env/Lib/site-packages/anyio/_backends/_trio.py +++ /dev/null @@ -1,1346 +0,0 @@ -from __future__ import annotations - -import array -import math -import os -import socket -import sys -import types -import weakref -from collections.abc import ( - AsyncGenerator, - AsyncIterator, - Awaitable, - Callable, - Collection, - Coroutine, - Iterable, - Sequence, -) -from contextlib import AbstractContextManager -from dataclasses import dataclass -from io import IOBase -from os import PathLike -from signal import Signals -from socket import AddressFamily, SocketKind -from types import TracebackType -from typing import ( - IO, - TYPE_CHECKING, - Any, - Generic, - NoReturn, - TypeVar, - cast, - overload, -) - -import trio.from_thread -import trio.lowlevel -from outcome import Error, Outcome, Value -from trio.lowlevel import ( - current_root_task, - current_task, - notify_closing, - wait_readable, - wait_writable, -) -from trio.socket import SocketType as TrioSocketType -from trio.to_thread import run_sync - -from .. import ( - CapacityLimiterStatistics, - EventStatistics, - LockStatistics, - RunFinishedError, - TaskInfo, - WouldBlock, - abc, -) -from .._core._eventloop import claim_worker_thread -from .._core._exceptions import ( - BrokenResourceError, - BusyResourceError, - ClosedResourceError, - EndOfStream, -) -from .._core._sockets import convert_ipv6_sockaddr -from .._core._streams import create_memory_object_stream -from .._core._synchronization import ( - CapacityLimiter as BaseCapacityLimiter, -) -from .._core._synchronization import Event as BaseEvent -from .._core._synchronization import Lock as BaseLock -from .._core._synchronization import ( - ResourceGuard, - SemaphoreStatistics, -) -from .._core._synchronization import Semaphore as BaseSemaphore -from .._core._tasks import CancelScope as BaseCancelScope -from ..abc import IPSockAddrType, UDPPacketType, UNIXDatagramPacketType -from ..abc._eventloop import AsyncBackend, StrOrBytesPath -from ..streams.memory import MemoryObjectSendStream - -if TYPE_CHECKING: - from _typeshed import FileDescriptorLike - -if sys.version_info >= (3, 10): - from typing import ParamSpec -else: - from typing_extensions import ParamSpec - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from exceptiongroup import BaseExceptionGroup - from typing_extensions import TypeVarTuple, Unpack - -T = TypeVar("T") -T_Retval = TypeVar("T_Retval") -T_SockAddr = TypeVar("T_SockAddr", str, IPSockAddrType) -PosArgsT = TypeVarTuple("PosArgsT") -P = ParamSpec("P") - - -# -# Event loop -# - -RunVar = trio.lowlevel.RunVar - - -# -# Timeouts and cancellation -# - - -class CancelScope(BaseCancelScope): - def __new__( - cls, original: trio.CancelScope | None = None, **kwargs: object - ) -> CancelScope: - return object.__new__(cls) - - def __init__(self, original: trio.CancelScope | None = None, **kwargs: Any) -> None: - self.__original = original or trio.CancelScope(**kwargs) - - def __enter__(self) -> CancelScope: - self.__original.__enter__() - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - return self.__original.__exit__(exc_type, exc_val, exc_tb) - - def cancel(self, reason: str | None = None) -> None: - self.__original.cancel(reason) - - @property - def deadline(self) -> float: - return self.__original.deadline - - @deadline.setter - def deadline(self, value: float) -> None: - self.__original.deadline = value - - @property - def cancel_called(self) -> bool: - return self.__original.cancel_called - - @property - def cancelled_caught(self) -> bool: - return self.__original.cancelled_caught - - @property - def shield(self) -> bool: - return self.__original.shield - - @shield.setter - def shield(self, value: bool) -> None: - self.__original.shield = value - - -# -# Task groups -# - - -class TaskGroup(abc.TaskGroup): - def __init__(self) -> None: - self._active = False - self._nursery_manager = trio.open_nursery(strict_exception_groups=True) - self.cancel_scope = None # type: ignore[assignment] - - async def __aenter__(self) -> TaskGroup: - self._active = True - self._nursery = await self._nursery_manager.__aenter__() - self.cancel_scope = CancelScope(self._nursery.cancel_scope) - return self - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - try: - # trio.Nursery.__exit__ returns bool; .open_nursery has wrong type - return await self._nursery_manager.__aexit__(exc_type, exc_val, exc_tb) # type: ignore[return-value] - except BaseExceptionGroup as exc: - if not exc.split(trio.Cancelled)[1]: - raise trio.Cancelled._create() from exc - - raise - finally: - del exc_val, exc_tb - self._active = False - - def start_soon( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[Any]], - *args: Unpack[PosArgsT], - name: object = None, - ) -> None: - if not self._active: - raise RuntimeError( - "This task group is not active; no new tasks can be started." - ) - - self._nursery.start_soon(func, *args, name=name) - - async def start( - self, func: Callable[..., Awaitable[Any]], *args: object, name: object = None - ) -> Any: - if not self._active: - raise RuntimeError( - "This task group is not active; no new tasks can be started." - ) - - return await self._nursery.start(func, *args, name=name) - - -# -# Subprocesses -# - - -@dataclass(eq=False) -class ReceiveStreamWrapper(abc.ByteReceiveStream): - _stream: trio.abc.ReceiveStream - - async def receive(self, max_bytes: int | None = None) -> bytes: - try: - data = await self._stream.receive_some(max_bytes) - except trio.ClosedResourceError as exc: - raise ClosedResourceError from exc.__cause__ - except trio.BrokenResourceError as exc: - raise BrokenResourceError from exc.__cause__ - - if data: - return bytes(data) - else: - raise EndOfStream - - async def aclose(self) -> None: - await self._stream.aclose() - - -@dataclass(eq=False) -class SendStreamWrapper(abc.ByteSendStream): - _stream: trio.abc.SendStream - - async def send(self, item: bytes) -> None: - try: - await self._stream.send_all(item) - except trio.ClosedResourceError as exc: - raise ClosedResourceError from exc.__cause__ - except trio.BrokenResourceError as exc: - raise BrokenResourceError from exc.__cause__ - - async def aclose(self) -> None: - await self._stream.aclose() - - -@dataclass(eq=False) -class Process(abc.Process): - _process: trio.Process - _stdin: abc.ByteSendStream | None - _stdout: abc.ByteReceiveStream | None - _stderr: abc.ByteReceiveStream | None - - async def aclose(self) -> None: - with CancelScope(shield=True): - if self._stdin: - await self._stdin.aclose() - if self._stdout: - await self._stdout.aclose() - if self._stderr: - await self._stderr.aclose() - - try: - await self.wait() - except BaseException: - self.kill() - with CancelScope(shield=True): - await self.wait() - raise - - async def wait(self) -> int: - return await self._process.wait() - - def terminate(self) -> None: - self._process.terminate() - - def kill(self) -> None: - self._process.kill() - - def send_signal(self, signal: Signals) -> None: - self._process.send_signal(signal) - - @property - def pid(self) -> int: - return self._process.pid - - @property - def returncode(self) -> int | None: - return self._process.returncode - - @property - def stdin(self) -> abc.ByteSendStream | None: - return self._stdin - - @property - def stdout(self) -> abc.ByteReceiveStream | None: - return self._stdout - - @property - def stderr(self) -> abc.ByteReceiveStream | None: - return self._stderr - - -class _ProcessPoolShutdownInstrument(trio.abc.Instrument): - def after_run(self) -> None: - super().after_run() - - -current_default_worker_process_limiter: trio.lowlevel.RunVar = RunVar( - "current_default_worker_process_limiter" -) - - -async def _shutdown_process_pool(workers: set[abc.Process]) -> None: - try: - await trio.sleep(math.inf) - except trio.Cancelled: - for process in workers: - if process.returncode is None: - process.kill() - - with CancelScope(shield=True): - for process in workers: - await process.aclose() - - -# -# Sockets and networking -# - - -class _TrioSocketMixin(Generic[T_SockAddr]): - def __init__(self, trio_socket: TrioSocketType) -> None: - self._trio_socket = trio_socket - self._closed = False - - def _check_closed(self) -> None: - if self._closed: - raise ClosedResourceError - if self._trio_socket.fileno() < 0: - raise BrokenResourceError - - @property - def _raw_socket(self) -> socket.socket: - return self._trio_socket._sock # type: ignore[attr-defined] - - async def aclose(self) -> None: - if self._trio_socket.fileno() >= 0: - self._closed = True - self._trio_socket.close() - - def _convert_socket_error(self, exc: BaseException) -> NoReturn: - if isinstance(exc, trio.ClosedResourceError): - raise ClosedResourceError from exc - elif self._trio_socket.fileno() < 0 and self._closed: - raise ClosedResourceError from None - elif isinstance(exc, OSError): - raise BrokenResourceError from exc - else: - raise exc - - -class SocketStream(_TrioSocketMixin, abc.SocketStream): - def __init__(self, trio_socket: TrioSocketType) -> None: - super().__init__(trio_socket) - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - async def receive(self, max_bytes: int = 65536) -> bytes: - with self._receive_guard: - try: - data = await self._trio_socket.recv(max_bytes) - except BaseException as exc: - self._convert_socket_error(exc) - - if data: - return data - else: - raise EndOfStream - - async def send(self, item: bytes) -> None: - with self._send_guard: - view = memoryview(item) - while view: - try: - bytes_sent = await self._trio_socket.send(view) - except BaseException as exc: - self._convert_socket_error(exc) - - view = view[bytes_sent:] - - async def send_eof(self) -> None: - self._trio_socket.shutdown(socket.SHUT_WR) - - -class UNIXSocketStream(SocketStream, abc.UNIXSocketStream): - async def receive_fds(self, msglen: int, maxfds: int) -> tuple[bytes, list[int]]: - if not isinstance(msglen, int) or msglen < 0: - raise ValueError("msglen must be a non-negative integer") - if not isinstance(maxfds, int) or maxfds < 1: - raise ValueError("maxfds must be a positive integer") - - fds = array.array("i") - await trio.lowlevel.checkpoint() - with self._receive_guard: - while True: - try: - message, ancdata, flags, addr = await self._trio_socket.recvmsg( - msglen, socket.CMSG_LEN(maxfds * fds.itemsize) - ) - except BaseException as exc: - self._convert_socket_error(exc) - else: - if not message and not ancdata: - raise EndOfStream - - break - - for cmsg_level, cmsg_type, cmsg_data in ancdata: - if cmsg_level != socket.SOL_SOCKET or cmsg_type != socket.SCM_RIGHTS: - raise RuntimeError( - f"Received unexpected ancillary data; message = {message!r}, " - f"cmsg_level = {cmsg_level}, cmsg_type = {cmsg_type}" - ) - - fds.frombytes(cmsg_data[: len(cmsg_data) - (len(cmsg_data) % fds.itemsize)]) - - return message, list(fds) - - async def send_fds(self, message: bytes, fds: Collection[int | IOBase]) -> None: - if not message: - raise ValueError("message must not be empty") - if not fds: - raise ValueError("fds must not be empty") - - filenos: list[int] = [] - for fd in fds: - if isinstance(fd, int): - filenos.append(fd) - elif isinstance(fd, IOBase): - filenos.append(fd.fileno()) - - fdarray = array.array("i", filenos) - await trio.lowlevel.checkpoint() - with self._send_guard: - while True: - try: - await self._trio_socket.sendmsg( - [message], - [ - ( - socket.SOL_SOCKET, - socket.SCM_RIGHTS, - fdarray, - ) - ], - ) - break - except BaseException as exc: - self._convert_socket_error(exc) - - -class TCPSocketListener(_TrioSocketMixin, abc.SocketListener): - def __init__(self, raw_socket: socket.socket): - super().__init__(trio.socket.from_stdlib_socket(raw_socket)) - self._accept_guard = ResourceGuard("accepting connections from") - - async def accept(self) -> SocketStream: - with self._accept_guard: - try: - trio_socket, _addr = await self._trio_socket.accept() - except BaseException as exc: - self._convert_socket_error(exc) - - trio_socket.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) - return SocketStream(trio_socket) - - -class UNIXSocketListener(_TrioSocketMixin, abc.SocketListener): - def __init__(self, raw_socket: socket.socket): - super().__init__(trio.socket.from_stdlib_socket(raw_socket)) - self._accept_guard = ResourceGuard("accepting connections from") - - async def accept(self) -> UNIXSocketStream: - with self._accept_guard: - try: - trio_socket, _addr = await self._trio_socket.accept() - except BaseException as exc: - self._convert_socket_error(exc) - - return UNIXSocketStream(trio_socket) - - -class UDPSocket(_TrioSocketMixin[IPSockAddrType], abc.UDPSocket): - def __init__(self, trio_socket: TrioSocketType) -> None: - super().__init__(trio_socket) - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - async def receive(self) -> tuple[bytes, IPSockAddrType]: - with self._receive_guard: - try: - data, addr = await self._trio_socket.recvfrom(65536) - return data, convert_ipv6_sockaddr(addr) - except BaseException as exc: - self._convert_socket_error(exc) - - async def send(self, item: UDPPacketType) -> None: - with self._send_guard: - try: - await self._trio_socket.sendto(*item) - except BaseException as exc: - self._convert_socket_error(exc) - - -class ConnectedUDPSocket(_TrioSocketMixin[IPSockAddrType], abc.ConnectedUDPSocket): - def __init__(self, trio_socket: TrioSocketType) -> None: - super().__init__(trio_socket) - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - async def receive(self) -> bytes: - with self._receive_guard: - try: - return await self._trio_socket.recv(65536) - except BaseException as exc: - self._convert_socket_error(exc) - - async def send(self, item: bytes) -> None: - with self._send_guard: - try: - await self._trio_socket.send(item) - except BaseException as exc: - self._convert_socket_error(exc) - - -class UNIXDatagramSocket(_TrioSocketMixin[str], abc.UNIXDatagramSocket): - def __init__(self, trio_socket: TrioSocketType) -> None: - super().__init__(trio_socket) - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - async def receive(self) -> UNIXDatagramPacketType: - with self._receive_guard: - try: - data, addr = await self._trio_socket.recvfrom(65536) - return data, addr - except BaseException as exc: - self._convert_socket_error(exc) - - async def send(self, item: UNIXDatagramPacketType) -> None: - with self._send_guard: - try: - await self._trio_socket.sendto(*item) - except BaseException as exc: - self._convert_socket_error(exc) - - -class ConnectedUNIXDatagramSocket( - _TrioSocketMixin[str], abc.ConnectedUNIXDatagramSocket -): - def __init__(self, trio_socket: TrioSocketType) -> None: - super().__init__(trio_socket) - self._receive_guard = ResourceGuard("reading from") - self._send_guard = ResourceGuard("writing to") - - async def receive(self) -> bytes: - with self._receive_guard: - try: - return await self._trio_socket.recv(65536) - except BaseException as exc: - self._convert_socket_error(exc) - - async def send(self, item: bytes) -> None: - with self._send_guard: - try: - await self._trio_socket.send(item) - except BaseException as exc: - self._convert_socket_error(exc) - - -# -# Synchronization -# - - -class Event(BaseEvent): - def __new__(cls) -> Event: - return object.__new__(cls) - - def __init__(self) -> None: - self.__original = trio.Event() - - def is_set(self) -> bool: - return self.__original.is_set() - - async def wait(self) -> None: - return await self.__original.wait() - - def statistics(self) -> EventStatistics: - orig_statistics = self.__original.statistics() - return EventStatistics(tasks_waiting=orig_statistics.tasks_waiting) - - def set(self) -> None: - self.__original.set() - - -class Lock(BaseLock): - def __new__(cls, *, fast_acquire: bool = False) -> Lock: - return object.__new__(cls) - - def __init__(self, *, fast_acquire: bool = False) -> None: - self._fast_acquire = fast_acquire - self.__original = trio.Lock() - - @staticmethod - def _convert_runtime_error_msg(exc: RuntimeError) -> None: - if exc.args == ("attempt to re-acquire an already held Lock",): - exc.args = ("Attempted to acquire an already held Lock",) - - async def acquire(self) -> None: - if not self._fast_acquire: - try: - await self.__original.acquire() - except RuntimeError as exc: - self._convert_runtime_error_msg(exc) - raise - - return - - # This is the "fast path" where we don't let other tasks run - await trio.lowlevel.checkpoint_if_cancelled() - try: - self.__original.acquire_nowait() - except trio.WouldBlock: - await self.__original._lot.park() - except RuntimeError as exc: - self._convert_runtime_error_msg(exc) - raise - - def acquire_nowait(self) -> None: - try: - self.__original.acquire_nowait() - except trio.WouldBlock: - raise WouldBlock from None - except RuntimeError as exc: - self._convert_runtime_error_msg(exc) - raise - - def locked(self) -> bool: - return self.__original.locked() - - def release(self) -> None: - self.__original.release() - - def statistics(self) -> LockStatistics: - orig_statistics = self.__original.statistics() - owner = TrioTaskInfo(orig_statistics.owner) if orig_statistics.owner else None - return LockStatistics( - orig_statistics.locked, owner, orig_statistics.tasks_waiting - ) - - -class Semaphore(BaseSemaphore): - def __new__( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> Semaphore: - return object.__new__(cls) - - def __init__( - self, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> None: - super().__init__(initial_value, max_value=max_value, fast_acquire=fast_acquire) - self.__original = trio.Semaphore(initial_value, max_value=max_value) - - async def acquire(self) -> None: - if not self._fast_acquire: - await self.__original.acquire() - return - - # This is the "fast path" where we don't let other tasks run - await trio.lowlevel.checkpoint_if_cancelled() - try: - self.__original.acquire_nowait() - except trio.WouldBlock: - await self.__original._lot.park() - - def acquire_nowait(self) -> None: - try: - self.__original.acquire_nowait() - except trio.WouldBlock: - raise WouldBlock from None - - @property - def max_value(self) -> int | None: - return self.__original.max_value - - @property - def value(self) -> int: - return self.__original.value - - def release(self) -> None: - self.__original.release() - - def statistics(self) -> SemaphoreStatistics: - orig_statistics = self.__original.statistics() - return SemaphoreStatistics(orig_statistics.tasks_waiting) - - -class CapacityLimiter(BaseCapacityLimiter): - def __new__( - cls, - total_tokens: float | None = None, - *, - original: trio.CapacityLimiter | None = None, - ) -> CapacityLimiter: - return object.__new__(cls) - - def __init__( - self, - total_tokens: float | None = None, - *, - original: trio.CapacityLimiter | None = None, - ) -> None: - if original is not None: - self.__original = original - else: - assert total_tokens is not None - self.__original = trio.CapacityLimiter(total_tokens) - - async def __aenter__(self) -> None: - return await self.__original.__aenter__() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - await self.__original.__aexit__(exc_type, exc_val, exc_tb) - - @property - def total_tokens(self) -> float: - return self.__original.total_tokens - - @total_tokens.setter - def total_tokens(self, value: float) -> None: - self.__original.total_tokens = value - - @property - def borrowed_tokens(self) -> int: - return self.__original.borrowed_tokens - - @property - def available_tokens(self) -> float: - return self.__original.available_tokens - - def acquire_nowait(self) -> None: - self.__original.acquire_nowait() - - def acquire_on_behalf_of_nowait(self, borrower: object) -> None: - self.__original.acquire_on_behalf_of_nowait(borrower) - - async def acquire(self) -> None: - await self.__original.acquire() - - async def acquire_on_behalf_of(self, borrower: object) -> None: - await self.__original.acquire_on_behalf_of(borrower) - - def release(self) -> None: - return self.__original.release() - - def release_on_behalf_of(self, borrower: object) -> None: - return self.__original.release_on_behalf_of(borrower) - - def statistics(self) -> CapacityLimiterStatistics: - orig = self.__original.statistics() - return CapacityLimiterStatistics( - borrowed_tokens=orig.borrowed_tokens, - total_tokens=orig.total_tokens, - borrowers=tuple(orig.borrowers), - tasks_waiting=orig.tasks_waiting, - ) - - -_capacity_limiter_wrapper: trio.lowlevel.RunVar = RunVar("_capacity_limiter_wrapper") - - -# -# Signal handling -# - - -class _SignalReceiver: - _iterator: AsyncIterator[int] - - def __init__(self, signals: tuple[Signals, ...]): - self._signals = signals - - def __enter__(self) -> _SignalReceiver: - self._cm = trio.open_signal_receiver(*self._signals) - self._iterator = self._cm.__enter__() - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool | None: - return self._cm.__exit__(exc_type, exc_val, exc_tb) - - def __aiter__(self) -> _SignalReceiver: - return self - - async def __anext__(self) -> Signals: - signum = await self._iterator.__anext__() - return Signals(signum) - - -# -# Testing and debugging -# - - -class TestRunner(abc.TestRunner): - def __init__(self, **options: Any) -> None: - from queue import Queue - - self._call_queue: Queue[Callable[[], object]] = Queue() - self._send_stream: MemoryObjectSendStream | None = None - self._options = options - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: types.TracebackType | None, - ) -> None: - if self._send_stream: - self._send_stream.close() - while self._send_stream is not None: - self._call_queue.get()() - - async def _run_tests_and_fixtures(self) -> None: - self._send_stream, receive_stream = create_memory_object_stream(1) - with receive_stream: - async for coro, outcome_holder in receive_stream: - try: - retval = await coro - except BaseException as exc: - outcome_holder.append(Error(exc)) - else: - outcome_holder.append(Value(retval)) - - def _main_task_finished(self, outcome: object) -> None: - self._send_stream = None - - def _call_in_runner_task( - self, - func: Callable[P, Awaitable[T_Retval]], - *args: P.args, - **kwargs: P.kwargs, - ) -> T_Retval: - if self._send_stream is None: - trio.lowlevel.start_guest_run( - self._run_tests_and_fixtures, - run_sync_soon_threadsafe=self._call_queue.put, - done_callback=self._main_task_finished, - **self._options, - ) - while self._send_stream is None: - self._call_queue.get()() - - outcome_holder: list[Outcome] = [] - self._send_stream.send_nowait((func(*args, **kwargs), outcome_holder)) - while not outcome_holder: - self._call_queue.get()() - - return outcome_holder[0].unwrap() - - def run_asyncgen_fixture( - self, - fixture_func: Callable[..., AsyncGenerator[T_Retval, Any]], - kwargs: dict[str, Any], - ) -> Iterable[T_Retval]: - asyncgen = fixture_func(**kwargs) - fixturevalue: T_Retval = self._call_in_runner_task(asyncgen.asend, None) - - yield fixturevalue - - try: - self._call_in_runner_task(asyncgen.asend, None) - except StopAsyncIteration: - pass - else: - self._call_in_runner_task(asyncgen.aclose) - raise RuntimeError("Async generator fixture did not stop") - - def run_fixture( - self, - fixture_func: Callable[..., Coroutine[Any, Any, T_Retval]], - kwargs: dict[str, Any], - ) -> T_Retval: - return self._call_in_runner_task(fixture_func, **kwargs) - - def run_test( - self, test_func: Callable[..., Coroutine[Any, Any, Any]], kwargs: dict[str, Any] - ) -> None: - self._call_in_runner_task(test_func, **kwargs) - - -class TrioTaskInfo(TaskInfo): - def __init__(self, task: trio.lowlevel.Task): - parent_id = None - if task.parent_nursery and task.parent_nursery.parent_task: - parent_id = id(task.parent_nursery.parent_task) - - super().__init__(id(task), parent_id, task.name, task.coro) - self._task = weakref.proxy(task) - - def has_pending_cancellation(self) -> bool: - try: - return self._task._cancel_status.effectively_cancelled - except ReferenceError: - # If the task is no longer around, it surely doesn't have a cancellation - # pending - return False - - -class TrioBackend(AsyncBackend): - @classmethod - def run( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - kwargs: dict[str, Any], - options: dict[str, Any], - ) -> T_Retval: - return trio.run(func, *args) - - @classmethod - def current_token(cls) -> object: - return trio.lowlevel.current_trio_token() - - @classmethod - def current_time(cls) -> float: - return trio.current_time() - - @classmethod - def cancelled_exception_class(cls) -> type[BaseException]: - return trio.Cancelled - - @classmethod - async def checkpoint(cls) -> None: - await trio.lowlevel.checkpoint() - - @classmethod - async def checkpoint_if_cancelled(cls) -> None: - await trio.lowlevel.checkpoint_if_cancelled() - - @classmethod - async def cancel_shielded_checkpoint(cls) -> None: - await trio.lowlevel.cancel_shielded_checkpoint() - - @classmethod - async def sleep(cls, delay: float) -> None: - await trio.sleep(delay) - - @classmethod - def create_cancel_scope( - cls, *, deadline: float = math.inf, shield: bool = False - ) -> abc.CancelScope: - return CancelScope(deadline=deadline, shield=shield) - - @classmethod - def current_effective_deadline(cls) -> float: - return trio.current_effective_deadline() - - @classmethod - def create_task_group(cls) -> abc.TaskGroup: - return TaskGroup() - - @classmethod - def create_event(cls) -> abc.Event: - return Event() - - @classmethod - def create_lock(cls, *, fast_acquire: bool) -> Lock: - return Lock(fast_acquire=fast_acquire) - - @classmethod - def create_semaphore( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> abc.Semaphore: - return Semaphore(initial_value, max_value=max_value, fast_acquire=fast_acquire) - - @classmethod - def create_capacity_limiter(cls, total_tokens: float) -> CapacityLimiter: - return CapacityLimiter(total_tokens) - - @classmethod - async def run_sync_in_worker_thread( - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - abandon_on_cancel: bool = False, - limiter: abc.CapacityLimiter | None = None, - ) -> T_Retval: - def wrapper() -> T_Retval: - with claim_worker_thread(TrioBackend, token): - return func(*args) - - token = TrioBackend.current_token() - return await run_sync( - wrapper, - abandon_on_cancel=abandon_on_cancel, - limiter=cast(trio.CapacityLimiter, limiter), - ) - - @classmethod - def check_cancelled(cls) -> None: - trio.from_thread.check_cancelled() - - @classmethod - def run_async_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - trio_token = cast("trio.lowlevel.TrioToken | None", token) - try: - return trio.from_thread.run(func, *args, trio_token=trio_token) - except trio.RunFinishedError: - raise RunFinishedError from None - - @classmethod - def run_sync_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - trio_token = cast("trio.lowlevel.TrioToken | None", token) - try: - return trio.from_thread.run_sync(func, *args, trio_token=trio_token) - except trio.RunFinishedError: - raise RunFinishedError from None - - @classmethod - async def open_process( - cls, - command: StrOrBytesPath | Sequence[StrOrBytesPath], - *, - stdin: int | IO[Any] | None, - stdout: int | IO[Any] | None, - stderr: int | IO[Any] | None, - **kwargs: Any, - ) -> Process: - def convert_item(item: StrOrBytesPath) -> str: - str_or_bytes = os.fspath(item) - if isinstance(str_or_bytes, str): - return str_or_bytes - else: - return os.fsdecode(str_or_bytes) - - if isinstance(command, (str, bytes, PathLike)): - process = await trio.lowlevel.open_process( - convert_item(command), - stdin=stdin, - stdout=stdout, - stderr=stderr, - shell=True, - **kwargs, - ) - else: - process = await trio.lowlevel.open_process( - [convert_item(item) for item in command], - stdin=stdin, - stdout=stdout, - stderr=stderr, - shell=False, - **kwargs, - ) - - stdin_stream = SendStreamWrapper(process.stdin) if process.stdin else None - stdout_stream = ReceiveStreamWrapper(process.stdout) if process.stdout else None - stderr_stream = ReceiveStreamWrapper(process.stderr) if process.stderr else None - return Process(process, stdin_stream, stdout_stream, stderr_stream) - - @classmethod - def setup_process_pool_exit_at_shutdown(cls, workers: set[abc.Process]) -> None: - trio.lowlevel.spawn_system_task(_shutdown_process_pool, workers) - - @classmethod - async def connect_tcp( - cls, host: str, port: int, local_address: IPSockAddrType | None = None - ) -> SocketStream: - family = socket.AF_INET6 if ":" in host else socket.AF_INET - trio_socket = trio.socket.socket(family) - trio_socket.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) - if local_address: - await trio_socket.bind(local_address) - - try: - await trio_socket.connect((host, port)) - except BaseException: - trio_socket.close() - raise - - return SocketStream(trio_socket) - - @classmethod - async def connect_unix(cls, path: str | bytes) -> abc.UNIXSocketStream: - trio_socket = trio.socket.socket(socket.AF_UNIX) - try: - await trio_socket.connect(path) - except BaseException: - trio_socket.close() - raise - - return UNIXSocketStream(trio_socket) - - @classmethod - def create_tcp_listener(cls, sock: socket.socket) -> abc.SocketListener: - return TCPSocketListener(sock) - - @classmethod - def create_unix_listener(cls, sock: socket.socket) -> abc.SocketListener: - return UNIXSocketListener(sock) - - @classmethod - async def create_udp_socket( - cls, - family: socket.AddressFamily, - local_address: IPSockAddrType | None, - remote_address: IPSockAddrType | None, - reuse_port: bool, - ) -> UDPSocket | ConnectedUDPSocket: - trio_socket = trio.socket.socket(family=family, type=socket.SOCK_DGRAM) - - if reuse_port: - trio_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1) - - if local_address: - await trio_socket.bind(local_address) - - if remote_address: - await trio_socket.connect(remote_address) - return ConnectedUDPSocket(trio_socket) - else: - return UDPSocket(trio_socket) - - @classmethod - @overload - async def create_unix_datagram_socket( - cls, raw_socket: socket.socket, remote_path: None - ) -> abc.UNIXDatagramSocket: ... - - @classmethod - @overload - async def create_unix_datagram_socket( - cls, raw_socket: socket.socket, remote_path: str | bytes - ) -> abc.ConnectedUNIXDatagramSocket: ... - - @classmethod - async def create_unix_datagram_socket( - cls, raw_socket: socket.socket, remote_path: str | bytes | None - ) -> abc.UNIXDatagramSocket | abc.ConnectedUNIXDatagramSocket: - trio_socket = trio.socket.from_stdlib_socket(raw_socket) - - if remote_path: - await trio_socket.connect(remote_path) - return ConnectedUNIXDatagramSocket(trio_socket) - else: - return UNIXDatagramSocket(trio_socket) - - @classmethod - async def getaddrinfo( - cls, - host: bytes | str | None, - port: str | int | None, - *, - family: int | AddressFamily = 0, - type: int | SocketKind = 0, - proto: int = 0, - flags: int = 0, - ) -> Sequence[ - tuple[ - AddressFamily, - SocketKind, - int, - str, - tuple[str, int] | tuple[str, int, int, int] | tuple[int, bytes], - ] - ]: - return await trio.socket.getaddrinfo(host, port, family, type, proto, flags) - - @classmethod - async def getnameinfo( - cls, sockaddr: IPSockAddrType, flags: int = 0 - ) -> tuple[str, str]: - return await trio.socket.getnameinfo(sockaddr, flags) - - @classmethod - async def wait_readable(cls, obj: FileDescriptorLike) -> None: - try: - await wait_readable(obj) - except trio.ClosedResourceError as exc: - raise ClosedResourceError().with_traceback(exc.__traceback__) from None - except trio.BusyResourceError: - raise BusyResourceError("reading from") from None - - @classmethod - async def wait_writable(cls, obj: FileDescriptorLike) -> None: - try: - await wait_writable(obj) - except trio.ClosedResourceError as exc: - raise ClosedResourceError().with_traceback(exc.__traceback__) from None - except trio.BusyResourceError: - raise BusyResourceError("writing to") from None - - @classmethod - def notify_closing(cls, obj: FileDescriptorLike) -> None: - notify_closing(obj) - - @classmethod - async def wrap_listener_socket(cls, sock: socket.socket) -> abc.SocketListener: - return TCPSocketListener(sock) - - @classmethod - async def wrap_stream_socket(cls, sock: socket.socket) -> SocketStream: - trio_sock = trio.socket.from_stdlib_socket(sock) - return SocketStream(trio_sock) - - @classmethod - async def wrap_unix_stream_socket(cls, sock: socket.socket) -> UNIXSocketStream: - trio_sock = trio.socket.from_stdlib_socket(sock) - return UNIXSocketStream(trio_sock) - - @classmethod - async def wrap_udp_socket(cls, sock: socket.socket) -> UDPSocket: - trio_sock = trio.socket.from_stdlib_socket(sock) - return UDPSocket(trio_sock) - - @classmethod - async def wrap_connected_udp_socket(cls, sock: socket.socket) -> ConnectedUDPSocket: - trio_sock = trio.socket.from_stdlib_socket(sock) - return ConnectedUDPSocket(trio_sock) - - @classmethod - async def wrap_unix_datagram_socket(cls, sock: socket.socket) -> UNIXDatagramSocket: - trio_sock = trio.socket.from_stdlib_socket(sock) - return UNIXDatagramSocket(trio_sock) - - @classmethod - async def wrap_connected_unix_datagram_socket( - cls, sock: socket.socket - ) -> ConnectedUNIXDatagramSocket: - trio_sock = trio.socket.from_stdlib_socket(sock) - return ConnectedUNIXDatagramSocket(trio_sock) - - @classmethod - def current_default_thread_limiter(cls) -> CapacityLimiter: - try: - return _capacity_limiter_wrapper.get() - except LookupError: - limiter = CapacityLimiter( - original=trio.to_thread.current_default_thread_limiter() - ) - _capacity_limiter_wrapper.set(limiter) - return limiter - - @classmethod - def open_signal_receiver( - cls, *signals: Signals - ) -> AbstractContextManager[AsyncIterator[Signals]]: - return _SignalReceiver(signals) - - @classmethod - def get_current_task(cls) -> TaskInfo: - task = current_task() - return TrioTaskInfo(task) - - @classmethod - def get_running_tasks(cls) -> Sequence[TaskInfo]: - root_task = current_root_task() - assert root_task - task_infos = [TrioTaskInfo(root_task)] - nurseries = root_task.child_nurseries - while nurseries: - new_nurseries: list[trio.Nursery] = [] - for nursery in nurseries: - for task in nursery.child_tasks: - task_infos.append(TrioTaskInfo(task)) - new_nurseries.extend(task.child_nurseries) - - nurseries = new_nurseries - - return task_infos - - @classmethod - async def wait_all_tasks_blocked(cls) -> None: - from trio.testing import wait_all_tasks_blocked - - await wait_all_tasks_blocked() - - @classmethod - def create_test_runner(cls, options: dict[str, Any]) -> TestRunner: - return TestRunner(**options) - - -backend_class = TrioBackend diff --git a/.env/Lib/site-packages/anyio/_core/__init__.py b/.env/Lib/site-packages/anyio/_core/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 7e570fc81667b54d7ff7205b70340e08a6256bda..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 184 zcmX@j%ge<81V;|EWrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxd74B>m6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdi&GYO7p3aQ$7kkcmc+;F6;%G>u*uC&Da}c>D`ExO!wAI1AjU^#Mn=XWW*`dy Dy~r`| diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_asyncio_selector_thread.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_asyncio_selector_thread.cpython-312.pyc deleted file mode 100644 index b460088203a51fb34e4b8139b5a7151dfd9fa4f0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8404 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 DEGrwn diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_fileio.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_fileio.cpython-312.pyc deleted file mode 100644 index a796a0a3105063d9d2272a887ae482103feab4fa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 43371 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 gjSs_BZm>87$8QB=yo1jCM714M+tenn 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 S!5o&OM6i_KEJYaOvj0D;>)HSS diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_streams.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_streams.cpython-312.pyc deleted file mode 100644 index 888507a421065e364be651744d074b2ac57f77aa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2333 zcmb7FO>7fa5PoZ~H}={|Oj-gYls*BXF@guKknP- zq_$FtR;cQ+H(IGuxmMxOV~-qqsMH=D!Xa9LREb`=1qE7(Q)hP9R!AkJuBGRBGqZ2L znfJ}??}LLXf#=uzU)w_iggn7b@5#o3w0R4JJH#TE>AGLE-1p%G-|OXXC~Z{RW8%x+Ohs&lwJuK zr1ZK1e(JY4G6hL_V?xt3Ox^YZ!3;}_bLC95xL1bD6|JVjFvSV1QgWF`C`3oHCi?{3 zl$r1?B3}kp*h`y#N8DSMM5M;$Ch~vVv+zD4d|cpL(F6WUPhLsd#nz$ zS!5l7J=SL>HF&+{GEdNk5r7}_wr~aMpU_D=rbSHoe>~~4gyYq~G*f}R=7Pf(nN!M# zGNKRD06$di02yU?CPRe?)vJcnVjpqtbB=c$j(nJK7>49En2A7lj+e`E+H)N29Hu=f z$zj@bf^c|SDZ*So=P-jU!~z-Iu|Hm3J6*0q8}Oid*|&Up!7v-lg9%e@wyPIxllx#> zv`rRNHRdf=&)f6W08TsFgj|d|s12`e`;)rqb2gdGy+CWW!vS6Q*pjYy4!ovrTH>*IT0rfr47Tsq1_U z(EJ4b4}Tmxj$?;L3pQe)!E&2CQ-(9yjf3RGDQP2J%v_Sz_Z>)t}X#WDeWs-pa diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_subprocesses.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_subprocesses.cpython-312.pyc deleted file mode 100644 index 50408dbc5aa3c1b6ff19612e7c986ffaaabf3bb0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9624 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 l5&3FjU^BAS!oXJKs*QobY^cEKDu~1r_NeE0wW diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_synchronization.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_synchronization.cpython-312.pyc deleted file mode 100644 index c1137709951d8992d8fdddb9900ef1c9b48b83c9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 32934 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 diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_tasks.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_tasks.cpython-312.pyc deleted file mode 100644 index 2bd33991aeb3c6d13f3c527c49b5a8feb42782e9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7683 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 diff --git a/.env/Lib/site-packages/anyio/_core/__pycache__/_typedattr.cpython-312.pyc b/.env/Lib/site-packages/anyio/_core/__pycache__/_typedattr.cpython-312.pyc deleted file mode 100644 index f42e03b8eac54e6f988b20291f6a4115e47b58dc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3829 zcma)9TWl2989sB_``YVE?3kp2Lm-5uwxLO(t<|)pCMlFlBQBL#j3(omV?20vW^>N0 z&90WnLRRXCREjE9j?^YXeW()g($~IJJoKSb`v65z%!sN~ZB^yX#F0Qe_5aUY);3kt zBkj?-{PW+=|NZ~Jf9~li5*UBH^^WtG0wMpzO)~Uop{)M~7AwRiw&IeyQd1OMtFBtt zYFb^d>2j~RnR>RCt>2K%YW!_181nQoZ(=NWf>)BYT_1WwzKz&;!V z+?j3$?1b;nH5;#S?(;Bv*1+?yaMn1&X3VA=Sdo6wjLK?+DC@(pSRp}dHX(_LXMlhE z&}kMlxkpXf^qjYvjQZ1;nH7xEreHP=d|Eg891TzjY9?BUanLLqQGPpj$KpQR%{vwoQ{&8=pE}`8PYEYrV-3@qGcUoh>9rhx zzhU{D?Kh-AlsDdJg%!CoQt^Bc1opsBK){ei@~=|=^=~i!g!jX4lMBFn9LoA`n12RN zV5Fwlq^8=_Q!-DoZMGE6gv%$FKnnKVSpKo;u zwS3nF$Ki{Q7v%?&v&@d(YCqB$M0o;3x$1sY7dF*Gju zS?LRcHCgz9dCK~4Vew04+`p@J$j^tmv8Ip~sb)@v6~pM7 zB@81}jCK*DSVlK(@1TyLbw>!m2z3JR-eyl!vtjNM17tzmYqrC~q6kbLhy^E@1t0)% zgpG{A*9!X$!||NJNZe}}2t3Ob9DNezk&kEASy(U(>3_ozm~`N|7x@4z_;#4uyPk@P z@uwJX141zc)8j>Qf6v&**6ml0+?u|-=h$-T{>bwmOn#JKKDs*i{AYuE?hNj^AwC`a z`tS0$N`F0Y{QmHc)nR&L zo3EDtEucu`pFa1rd^B0gZ^L4R6re<0hSk>5Q82w-)l|~a5&)hdyGW4waW~=H6w*n~CK@I;>s{DS$k0wj)4g?8p9Zu5q)j>@ z)5ye|Z*P8cXcFG60=Uz_N3~pN)v!}Y4`WHODUqB=Vc8-?a6|;Ah;wGB=f_4tU7Yd3 z{HXcZ05X)7s@oODO-De!dt0_ukoTL-|+P-%azAp_K?1j zFNh8_mEm7k?^Jf(AKriaz-xDhU;opQ+kJ05cy{|Qde;hY|3s95;kC*L`C!L{_IqVQ zJ5^Qr3vkwcG4ACrM1A-uO}hLG7cP#{=32{?Ak9CHT8K8+v32~%!V z#$`s?hg+PGo%k_KOXQw@@UTSl39lcMVr-Bd`kfgYslmKL`||elFgfdypxv z9{T<|8P1I0QR$%yt4ItoPh7~+p6VbrEi+B{8v;#B)R(aH-VbWy$_aK`VY3LRYJjs7I>Yu*HUj;hMS>h;6 zYnq}ce None: - self._thread = threading.Thread(target=self.run, name="AnyIO socket selector") - self._selector = DefaultSelector() - self._send, self._receive = socket.socketpair() - self._send.setblocking(False) - self._receive.setblocking(False) - # This somewhat reduces the amount of memory wasted queueing up data - # for wakeups. With these settings, maximum number of 1-byte sends - # before getting BlockingIOError: - # Linux 4.8: 6 - # macOS (darwin 15.5): 1 - # Windows 10: 525347 - # Windows you're weird. (And on Windows setting SNDBUF to 0 makes send - # blocking, even on non-blocking sockets, so don't do that.) - self._receive.setsockopt(socket.SOL_SOCKET, socket.SO_RCVBUF, 1) - self._send.setsockopt(socket.SOL_SOCKET, socket.SO_SNDBUF, 1) - # On Windows this is a TCP socket so this might matter. On other - # platforms this fails b/c AF_UNIX sockets aren't actually TCP. - try: - self._send.setsockopt(socket.IPPROTO_TCP, socket.TCP_NODELAY, 1) - except OSError: - pass - - self._selector.register(self._receive, EVENT_READ) - self._closed = False - - def start(self) -> None: - self._thread.start() - threading._register_atexit(self._stop) # type: ignore[attr-defined] - - def _stop(self) -> None: - global _selector - self._closed = True - self._notify_self() - self._send.close() - self._thread.join() - self._selector.unregister(self._receive) - self._receive.close() - self._selector.close() - _selector = None - assert not self._selector.get_map(), ( - "selector still has registered file descriptors after shutdown" - ) - - def _notify_self(self) -> None: - try: - self._send.send(b"\x00") - except BlockingIOError: - pass - - def add_reader(self, fd: FileDescriptorLike, callback: Callable[[], Any]) -> None: - loop = asyncio.get_running_loop() - try: - key = self._selector.get_key(fd) - except KeyError: - self._selector.register(fd, EVENT_READ, {EVENT_READ: (loop, callback)}) - else: - if EVENT_READ in key.data: - raise ValueError( - "this file descriptor is already registered for reading" - ) - - key.data[EVENT_READ] = loop, callback - self._selector.modify(fd, key.events | EVENT_READ, key.data) - - self._notify_self() - - def add_writer(self, fd: FileDescriptorLike, callback: Callable[[], Any]) -> None: - loop = asyncio.get_running_loop() - try: - key = self._selector.get_key(fd) - except KeyError: - self._selector.register(fd, EVENT_WRITE, {EVENT_WRITE: (loop, callback)}) - else: - if EVENT_WRITE in key.data: - raise ValueError( - "this file descriptor is already registered for writing" - ) - - key.data[EVENT_WRITE] = loop, callback - self._selector.modify(fd, key.events | EVENT_WRITE, key.data) - - self._notify_self() - - def remove_reader(self, fd: FileDescriptorLike) -> bool: - try: - key = self._selector.get_key(fd) - except KeyError: - return False - - if new_events := key.events ^ EVENT_READ: - del key.data[EVENT_READ] - self._selector.modify(fd, new_events, key.data) - else: - self._selector.unregister(fd) - - return True - - def remove_writer(self, fd: FileDescriptorLike) -> bool: - try: - key = self._selector.get_key(fd) - except KeyError: - return False - - if new_events := key.events ^ EVENT_WRITE: - del key.data[EVENT_WRITE] - self._selector.modify(fd, new_events, key.data) - else: - self._selector.unregister(fd) - - return True - - def run(self) -> None: - while not self._closed: - for key, events in self._selector.select(): - if key.fileobj is self._receive: - try: - while self._receive.recv(4096): - pass - except BlockingIOError: - pass - - continue - - if events & EVENT_READ: - loop, callback = key.data[EVENT_READ] - self.remove_reader(key.fd) - try: - loop.call_soon_threadsafe(callback) - except RuntimeError: - pass # the loop was already closed - - if events & EVENT_WRITE: - loop, callback = key.data[EVENT_WRITE] - self.remove_writer(key.fd) - try: - loop.call_soon_threadsafe(callback) - except RuntimeError: - pass # the loop was already closed - - -def get_selector() -> Selector: - global _selector - - with _selector_lock: - if _selector is None: - _selector = Selector() - _selector.start() - - return _selector diff --git a/.env/Lib/site-packages/anyio/_core/_contextmanagers.py b/.env/Lib/site-packages/anyio/_core/_contextmanagers.py deleted file mode 100644 index 302f32b..0000000 --- a/.env/Lib/site-packages/anyio/_core/_contextmanagers.py +++ /dev/null @@ -1,200 +0,0 @@ -from __future__ import annotations - -from abc import abstractmethod -from contextlib import AbstractAsyncContextManager, AbstractContextManager -from inspect import isasyncgen, iscoroutine, isgenerator -from types import TracebackType -from typing import Protocol, TypeVar, cast, final - -_T_co = TypeVar("_T_co", covariant=True) -_ExitT_co = TypeVar("_ExitT_co", covariant=True, bound="bool | None") - - -class _SupportsCtxMgr(Protocol[_T_co, _ExitT_co]): - def __contextmanager__(self) -> AbstractContextManager[_T_co, _ExitT_co]: ... - - -class _SupportsAsyncCtxMgr(Protocol[_T_co, _ExitT_co]): - def __asynccontextmanager__( - self, - ) -> AbstractAsyncContextManager[_T_co, _ExitT_co]: ... - - -class ContextManagerMixin: - """ - Mixin class providing context manager functionality via a generator-based - implementation. - - This class allows you to implement a context manager via :meth:`__contextmanager__` - which should return a generator. The mechanics are meant to mirror those of - :func:`@contextmanager `. - - .. note:: Classes using this mix-in are not reentrant as context managers, meaning - that once you enter it, you can't re-enter before first exiting it. - - .. seealso:: :doc:`contextmanagers` - """ - - __cm: AbstractContextManager[object, bool | None] | None = None - - @final - def __enter__(self: _SupportsCtxMgr[_T_co, bool | None]) -> _T_co: - # Needed for mypy to assume self still has the __cm member - assert isinstance(self, ContextManagerMixin) - if self.__cm is not None: - raise RuntimeError( - f"this {self.__class__.__qualname__} has already been entered" - ) - - cm = self.__contextmanager__() - if not isinstance(cm, AbstractContextManager): - if isgenerator(cm): - raise TypeError( - "__contextmanager__() returned a generator object instead of " - "a context manager. Did you forget to add the @contextmanager " - "decorator?" - ) - - raise TypeError( - f"__contextmanager__() did not return a context manager object, " - f"but {cm.__class__!r}" - ) - - if cm is self: - raise TypeError( - f"{self.__class__.__qualname__}.__contextmanager__() returned " - f"self. Did you forget to add the @contextmanager decorator and a " - f"'yield' statement?" - ) - - value = cm.__enter__() - self.__cm = cm - return value - - @final - def __exit__( - self: _SupportsCtxMgr[object, _ExitT_co], - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> _ExitT_co: - # Needed for mypy to assume self still has the __cm member - assert isinstance(self, ContextManagerMixin) - if self.__cm is None: - raise RuntimeError( - f"this {self.__class__.__qualname__} has not been entered yet" - ) - - # Prevent circular references - cm = self.__cm - del self.__cm - - return cast(_ExitT_co, cm.__exit__(exc_type, exc_val, exc_tb)) - - @abstractmethod - def __contextmanager__(self) -> AbstractContextManager[object, bool | None]: - """ - Implement your context manager logic here. - - This method **must** be decorated with - :func:`@contextmanager `. - - .. note:: Remember that the ``yield`` will raise any exception raised in the - enclosed context block, so use a ``finally:`` block to clean up resources! - - :return: a context manager object - """ - - -class AsyncContextManagerMixin: - """ - Mixin class providing async context manager functionality via a generator-based - implementation. - - This class allows you to implement a context manager via - :meth:`__asynccontextmanager__`. The mechanics are meant to mirror those of - :func:`@asynccontextmanager `. - - .. note:: Classes using this mix-in are not reentrant as context managers, meaning - that once you enter it, you can't re-enter before first exiting it. - - .. seealso:: :doc:`contextmanagers` - """ - - __cm: AbstractAsyncContextManager[object, bool | None] | None = None - - @final - async def __aenter__(self: _SupportsAsyncCtxMgr[_T_co, bool | None]) -> _T_co: - # Needed for mypy to assume self still has the __cm member - assert isinstance(self, AsyncContextManagerMixin) - if self.__cm is not None: - raise RuntimeError( - f"this {self.__class__.__qualname__} has already been entered" - ) - - cm = self.__asynccontextmanager__() - if not isinstance(cm, AbstractAsyncContextManager): - if isasyncgen(cm): - raise TypeError( - "__asynccontextmanager__() returned an async generator instead of " - "an async context manager. Did you forget to add the " - "@asynccontextmanager decorator?" - ) - elif iscoroutine(cm): - cm.close() - raise TypeError( - "__asynccontextmanager__() returned a coroutine object instead of " - "an async context manager. Did you forget to add the " - "@asynccontextmanager decorator and a 'yield' statement?" - ) - - raise TypeError( - f"__asynccontextmanager__() did not return an async context manager, " - f"but {cm.__class__!r}" - ) - - if cm is self: - raise TypeError( - f"{self.__class__.__qualname__}.__asynccontextmanager__() returned " - f"self. Did you forget to add the @asynccontextmanager decorator and a " - f"'yield' statement?" - ) - - value = await cm.__aenter__() - self.__cm = cm - return value - - @final - async def __aexit__( - self: _SupportsAsyncCtxMgr[object, _ExitT_co], - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> _ExitT_co: - assert isinstance(self, AsyncContextManagerMixin) - if self.__cm is None: - raise RuntimeError( - f"this {self.__class__.__qualname__} has not been entered yet" - ) - - # Prevent circular references - cm = self.__cm - del self.__cm - - return cast(_ExitT_co, await cm.__aexit__(exc_type, exc_val, exc_tb)) - - @abstractmethod - def __asynccontextmanager__( - self, - ) -> AbstractAsyncContextManager[object, bool | None]: - """ - Implement your async context manager logic here. - - This method **must** be decorated with - :func:`@asynccontextmanager `. - - .. note:: Remember that the ``yield`` will raise any exception raised in the - enclosed context block, so use a ``finally:`` block to clean up resources! - - :return: an async context manager object - """ diff --git a/.env/Lib/site-packages/anyio/_core/_eventloop.py b/.env/Lib/site-packages/anyio/_core/_eventloop.py deleted file mode 100644 index 59a69cc..0000000 --- a/.env/Lib/site-packages/anyio/_core/_eventloop.py +++ /dev/null @@ -1,234 +0,0 @@ -from __future__ import annotations - -import math -import sys -import threading -from collections.abc import Awaitable, Callable, Generator -from contextlib import contextmanager -from contextvars import Token -from importlib import import_module -from typing import TYPE_CHECKING, Any, TypeVar - -from ._exceptions import NoEventLoopError - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -sniffio: Any -try: - import sniffio -except ModuleNotFoundError: - sniffio = None - -if TYPE_CHECKING: - from ..abc import AsyncBackend - -# This must be updated when new backends are introduced -BACKENDS = "asyncio", "trio" - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") - -threadlocals = threading.local() -loaded_backends: dict[str, type[AsyncBackend]] = {} - - -def run( - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - *args: Unpack[PosArgsT], - backend: str = "asyncio", - backend_options: dict[str, Any] | None = None, -) -> T_Retval: - """ - Run the given coroutine function in an asynchronous event loop. - - The current thread must not be already running an event loop. - - :param func: a coroutine function - :param args: positional arguments to ``func`` - :param backend: name of the asynchronous event loop implementation – currently - either ``asyncio`` or ``trio`` - :param backend_options: keyword arguments to call the backend ``run()`` - implementation with (documented :ref:`here `) - :return: the return value of the coroutine function - :raises RuntimeError: if an asynchronous event loop is already running in this - thread - :raises LookupError: if the named backend is not found - - """ - if asynclib_name := current_async_library(): - raise RuntimeError(f"Already running {asynclib_name} in this thread") - - try: - async_backend = get_async_backend(backend) - except ImportError as exc: - raise LookupError(f"No such backend: {backend}") from exc - - token = None - if asynclib_name is None: - # Since we're in control of the event loop, we can cache the name of the async - # library - token = set_current_async_library(backend) - - try: - backend_options = backend_options or {} - return async_backend.run(func, args, {}, backend_options) - finally: - reset_current_async_library(token) - - -async def sleep(delay: float) -> None: - """ - Pause the current task for the specified duration. - - :param delay: the duration, in seconds - - """ - return await get_async_backend().sleep(delay) - - -async def sleep_forever() -> None: - """ - Pause the current task until it's cancelled. - - This is a shortcut for ``sleep(math.inf)``. - - .. versionadded:: 3.1 - - """ - await sleep(math.inf) - - -async def sleep_until(deadline: float) -> None: - """ - Pause the current task until the given time. - - :param deadline: the absolute time to wake up at (according to the internal - monotonic clock of the event loop) - - .. versionadded:: 3.1 - - """ - now = current_time() - await sleep(max(deadline - now, 0)) - - -def current_time() -> float: - """ - Return the current value of the event loop's internal clock. - - :return: the clock value (seconds) - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().current_time() - - -def get_all_backends() -> tuple[str, ...]: - """Return a tuple of the names of all built-in backends.""" - return BACKENDS - - -def get_available_backends() -> tuple[str, ...]: - """ - Test for the availability of built-in backends. - - :return a tuple of the built-in backend names that were successfully imported - - .. versionadded:: 4.12 - - """ - available_backends: list[str] = [] - for backend_name in get_all_backends(): - try: - get_async_backend(backend_name) - except ImportError: - continue - - available_backends.append(backend_name) - - return tuple(available_backends) - - -def get_cancelled_exc_class() -> type[BaseException]: - """ - Return the current async library's cancellation exception class. - - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().cancelled_exception_class() - - -# -# Private API -# - - -@contextmanager -def claim_worker_thread( - backend_class: type[AsyncBackend], token: object -) -> Generator[Any, None, None]: - from ..lowlevel import EventLoopToken - - threadlocals.current_token = EventLoopToken(backend_class, token) - try: - yield - finally: - del threadlocals.current_token - - -def get_async_backend(asynclib_name: str | None = None) -> type[AsyncBackend]: - if asynclib_name is None: - asynclib_name = current_async_library() - if not asynclib_name: - raise NoEventLoopError( - f"Not currently running on any asynchronous event loop. " - f"Available async backends: {', '.join(get_all_backends())}" - ) - - # We use our own dict instead of sys.modules to get the already imported back-end - # class because the appropriate modules in sys.modules could potentially be only - # partially initialized - try: - return loaded_backends[asynclib_name] - except KeyError: - module = import_module(f"anyio._backends._{asynclib_name}") - loaded_backends[asynclib_name] = module.backend_class - return module.backend_class - - -def current_async_library() -> str | None: - if sniffio is None: - # If sniffio is not installed, we assume we're either running asyncio or nothing - import asyncio - - try: - asyncio.get_running_loop() - return "asyncio" - except RuntimeError: - pass - else: - try: - return sniffio.current_async_library() - except sniffio.AsyncLibraryNotFoundError: - pass - - return None - - -def set_current_async_library(asynclib_name: str | None) -> Token | None: - # no-op if sniffio is not installed - if sniffio is None: - return None - - return sniffio.current_async_library_cvar.set(asynclib_name) - - -def reset_current_async_library(token: Token | None) -> None: - if token is not None: - sniffio.current_async_library_cvar.reset(token) diff --git a/.env/Lib/site-packages/anyio/_core/_exceptions.py b/.env/Lib/site-packages/anyio/_core/_exceptions.py deleted file mode 100644 index 3776bed..0000000 --- a/.env/Lib/site-packages/anyio/_core/_exceptions.py +++ /dev/null @@ -1,156 +0,0 @@ -from __future__ import annotations - -import sys -from collections.abc import Generator -from textwrap import dedent -from typing import Any - -if sys.version_info < (3, 11): - from exceptiongroup import BaseExceptionGroup - - -class BrokenResourceError(Exception): - """ - Raised when trying to use a resource that has been rendered unusable due to external - causes (e.g. a send stream whose peer has disconnected). - """ - - -class BrokenWorkerProcess(Exception): - """ - Raised by :meth:`~anyio.to_process.run_sync` if the worker process terminates abruptly or - otherwise misbehaves. - """ - - -class BrokenWorkerInterpreter(Exception): - """ - Raised by :meth:`~anyio.to_interpreter.run_sync` if an unexpected exception is - raised in the subinterpreter. - """ - - def __init__(self, excinfo: Any): - # This was adapted from concurrent.futures.interpreter.ExecutionFailed - msg = excinfo.formatted - if not msg: - if excinfo.type and excinfo.msg: - msg = f"{excinfo.type.__name__}: {excinfo.msg}" - else: - msg = excinfo.type.__name__ or excinfo.msg - - super().__init__(msg) - self.excinfo = excinfo - - def __str__(self) -> str: - try: - formatted = self.excinfo.errdisplay - except Exception: - return super().__str__() - else: - return dedent( - f""" - {super().__str__()} - - Uncaught in the interpreter: - - {formatted} - """.strip() - ) - - -class BusyResourceError(Exception): - """ - Raised when two tasks are trying to read from or write to the same resource - concurrently. - """ - - def __init__(self, action: str): - super().__init__(f"Another task is already {action} this resource") - - -class ClosedResourceError(Exception): - """Raised when trying to use a resource that has been closed.""" - - -class ConnectionFailed(OSError): - """ - Raised when a connection attempt fails. - - .. note:: This class inherits from :exc:`OSError` for backwards compatibility. - """ - - -def iterate_exceptions( - exception: BaseException, -) -> Generator[BaseException, None, None]: - if isinstance(exception, BaseExceptionGroup): - for exc in exception.exceptions: - yield from iterate_exceptions(exc) - else: - yield exception - - -class DelimiterNotFound(Exception): - """ - Raised during - :meth:`~anyio.streams.buffered.BufferedByteReceiveStream.receive_until` if the - maximum number of bytes has been read without the delimiter being found. - """ - - def __init__(self, max_bytes: int) -> None: - super().__init__( - f"The delimiter was not found among the first {max_bytes} bytes" - ) - - -class EndOfStream(Exception): - """ - Raised when trying to read from a stream that has been closed from the other end. - """ - - -class IncompleteRead(Exception): - """ - Raised during - :meth:`~anyio.streams.buffered.BufferedByteReceiveStream.receive_exactly` or - :meth:`~anyio.streams.buffered.BufferedByteReceiveStream.receive_until` if the - connection is closed before the requested amount of bytes has been read. - """ - - def __init__(self) -> None: - super().__init__( - "The stream was closed before the read operation could be completed" - ) - - -class TypedAttributeLookupError(LookupError): - """ - Raised by :meth:`~anyio.TypedAttributeProvider.extra` when the given typed attribute - is not found and no default value has been given. - """ - - -class WouldBlock(Exception): - """Raised by ``X_nowait`` functions if ``X()`` would block.""" - - -class NoEventLoopError(RuntimeError): - """ - Raised by several functions that require an event loop to be running in the current - thread when there is no running event loop. - - This is also raised by :func:`.from_thread.run` and :func:`.from_thread.run_sync` - if not calling from an AnyIO worker thread, and no ``token`` was passed. - """ - - -class RunFinishedError(RuntimeError): - """ - Raised by :func:`.from_thread.run` and :func:`.from_thread.run_sync` if the event - loop associated with the explicitly passed token has already finished. - """ - - def __init__(self) -> None: - super().__init__( - "The event loop associated with the given token has already finished" - ) diff --git a/.env/Lib/site-packages/anyio/_core/_fileio.py b/.env/Lib/site-packages/anyio/_core/_fileio.py deleted file mode 100644 index 061f0d7..0000000 --- a/.env/Lib/site-packages/anyio/_core/_fileio.py +++ /dev/null @@ -1,797 +0,0 @@ -from __future__ import annotations - -import os -import pathlib -import sys -from collections.abc import ( - AsyncIterator, - Callable, - Iterable, - Iterator, - Sequence, -) -from dataclasses import dataclass -from functools import partial -from os import PathLike -from typing import ( - IO, - TYPE_CHECKING, - Any, - AnyStr, - ClassVar, - Final, - Generic, - overload, -) - -from .. import to_thread -from ..abc import AsyncResource - -if TYPE_CHECKING: - from types import ModuleType - - from _typeshed import OpenBinaryMode, OpenTextMode, ReadableBuffer, WriteableBuffer -else: - ReadableBuffer = OpenBinaryMode = OpenTextMode = WriteableBuffer = object - - -class AsyncFile(AsyncResource, Generic[AnyStr]): - """ - An asynchronous file object. - - This class wraps a standard file object and provides async friendly versions of the - following blocking methods (where available on the original file object): - - * read - * read1 - * readline - * readlines - * readinto - * readinto1 - * write - * writelines - * truncate - * seek - * tell - * flush - - All other methods are directly passed through. - - This class supports the asynchronous context manager protocol which closes the - underlying file at the end of the context block. - - This class also supports asynchronous iteration:: - - async with await open_file(...) as f: - async for line in f: - print(line) - """ - - def __init__(self, fp: IO[AnyStr]) -> None: - self._fp: Any = fp - - def __getattr__(self, name: str) -> object: - return getattr(self._fp, name) - - @property - def wrapped(self) -> IO[AnyStr]: - """The wrapped file object.""" - return self._fp - - async def __aiter__(self) -> AsyncIterator[AnyStr]: - while True: - line = await self.readline() - if line: - yield line - else: - break - - async def aclose(self) -> None: - return await to_thread.run_sync(self._fp.close) - - async def read(self, size: int = -1) -> AnyStr: - return await to_thread.run_sync(self._fp.read, size) - - async def read1(self: AsyncFile[bytes], size: int = -1) -> bytes: - return await to_thread.run_sync(self._fp.read1, size) - - async def readline(self) -> AnyStr: - return await to_thread.run_sync(self._fp.readline) - - async def readlines(self) -> list[AnyStr]: - return await to_thread.run_sync(self._fp.readlines) - - async def readinto(self: AsyncFile[bytes], b: WriteableBuffer) -> int: - return await to_thread.run_sync(self._fp.readinto, b) - - async def readinto1(self: AsyncFile[bytes], b: WriteableBuffer) -> int: - return await to_thread.run_sync(self._fp.readinto1, b) - - @overload - async def write(self: AsyncFile[bytes], b: ReadableBuffer) -> int: ... - - @overload - async def write(self: AsyncFile[str], b: str) -> int: ... - - async def write(self, b: ReadableBuffer | str) -> int: - return await to_thread.run_sync(self._fp.write, b) - - @overload - async def writelines( - self: AsyncFile[bytes], lines: Iterable[ReadableBuffer] - ) -> None: ... - - @overload - async def writelines(self: AsyncFile[str], lines: Iterable[str]) -> None: ... - - async def writelines(self, lines: Iterable[ReadableBuffer] | Iterable[str]) -> None: - return await to_thread.run_sync(self._fp.writelines, lines) - - async def truncate(self, size: int | None = None) -> int: - return await to_thread.run_sync(self._fp.truncate, size) - - async def seek(self, offset: int, whence: int | None = os.SEEK_SET) -> int: - return await to_thread.run_sync(self._fp.seek, offset, whence) - - async def tell(self) -> int: - return await to_thread.run_sync(self._fp.tell) - - async def flush(self) -> None: - return await to_thread.run_sync(self._fp.flush) - - -@overload -async def open_file( - file: str | PathLike[str] | int, - mode: OpenBinaryMode, - buffering: int = ..., - encoding: str | None = ..., - errors: str | None = ..., - newline: str | None = ..., - closefd: bool = ..., - opener: Callable[[str, int], int] | None = ..., -) -> AsyncFile[bytes]: ... - - -@overload -async def open_file( - file: str | PathLike[str] | int, - mode: OpenTextMode = ..., - buffering: int = ..., - encoding: str | None = ..., - errors: str | None = ..., - newline: str | None = ..., - closefd: bool = ..., - opener: Callable[[str, int], int] | None = ..., -) -> AsyncFile[str]: ... - - -async def open_file( - file: str | PathLike[str] | int, - mode: str = "r", - buffering: int = -1, - encoding: str | None = None, - errors: str | None = None, - newline: str | None = None, - closefd: bool = True, - opener: Callable[[str, int], int] | None = None, -) -> AsyncFile[Any]: - """ - Open a file asynchronously. - - The arguments are exactly the same as for the builtin :func:`open`. - - :return: an asynchronous file object - - """ - fp = await to_thread.run_sync( - open, file, mode, buffering, encoding, errors, newline, closefd, opener - ) - return AsyncFile(fp) - - -def wrap_file(file: IO[AnyStr]) -> AsyncFile[AnyStr]: - """ - Wrap an existing file as an asynchronous file. - - :param file: an existing file-like object - :return: an asynchronous file object - - """ - return AsyncFile(file) - - -@dataclass(eq=False) -class _PathIterator(AsyncIterator["Path"]): - iterator: Iterator[PathLike[str]] - - async def __anext__(self) -> Path: - nextval = await to_thread.run_sync( - next, self.iterator, None, abandon_on_cancel=True - ) - if nextval is None: - raise StopAsyncIteration from None - - return Path(nextval) - - -class Path: - """ - An asynchronous version of :class:`pathlib.Path`. - - This class cannot be substituted for :class:`pathlib.Path` or - :class:`pathlib.PurePath`, but it is compatible with the :class:`os.PathLike` - interface. - - It implements the Python 3.10 version of :class:`pathlib.Path` interface, except for - the deprecated :meth:`~pathlib.Path.link_to` method. - - Some methods may be unavailable or have limited functionality, based on the Python - version: - - * :meth:`~pathlib.Path.copy` (available on Python 3.14 or later) - * :meth:`~pathlib.Path.copy_into` (available on Python 3.14 or later) - * :meth:`~pathlib.Path.from_uri` (available on Python 3.13 or later) - * :meth:`~pathlib.PurePath.full_match` (available on Python 3.13 or later) - * :attr:`~pathlib.Path.info` (available on Python 3.14 or later) - * :meth:`~pathlib.Path.is_junction` (available on Python 3.12 or later) - * :meth:`~pathlib.PurePath.match` (the ``case_sensitive`` parameter is only - available on Python 3.13 or later) - * :meth:`~pathlib.Path.move` (available on Python 3.14 or later) - * :meth:`~pathlib.Path.move_into` (available on Python 3.14 or later) - * :meth:`~pathlib.PurePath.relative_to` (the ``walk_up`` parameter is only available - on Python 3.12 or later) - * :meth:`~pathlib.Path.walk` (available on Python 3.12 or later) - - Any methods that do disk I/O need to be awaited on. These methods are: - - * :meth:`~pathlib.Path.absolute` - * :meth:`~pathlib.Path.chmod` - * :meth:`~pathlib.Path.cwd` - * :meth:`~pathlib.Path.exists` - * :meth:`~pathlib.Path.expanduser` - * :meth:`~pathlib.Path.group` - * :meth:`~pathlib.Path.hardlink_to` - * :meth:`~pathlib.Path.home` - * :meth:`~pathlib.Path.is_block_device` - * :meth:`~pathlib.Path.is_char_device` - * :meth:`~pathlib.Path.is_dir` - * :meth:`~pathlib.Path.is_fifo` - * :meth:`~pathlib.Path.is_file` - * :meth:`~pathlib.Path.is_junction` - * :meth:`~pathlib.Path.is_mount` - * :meth:`~pathlib.Path.is_socket` - * :meth:`~pathlib.Path.is_symlink` - * :meth:`~pathlib.Path.lchmod` - * :meth:`~pathlib.Path.lstat` - * :meth:`~pathlib.Path.mkdir` - * :meth:`~pathlib.Path.open` - * :meth:`~pathlib.Path.owner` - * :meth:`~pathlib.Path.read_bytes` - * :meth:`~pathlib.Path.read_text` - * :meth:`~pathlib.Path.readlink` - * :meth:`~pathlib.Path.rename` - * :meth:`~pathlib.Path.replace` - * :meth:`~pathlib.Path.resolve` - * :meth:`~pathlib.Path.rmdir` - * :meth:`~pathlib.Path.samefile` - * :meth:`~pathlib.Path.stat` - * :meth:`~pathlib.Path.symlink_to` - * :meth:`~pathlib.Path.touch` - * :meth:`~pathlib.Path.unlink` - * :meth:`~pathlib.Path.walk` - * :meth:`~pathlib.Path.write_bytes` - * :meth:`~pathlib.Path.write_text` - - Additionally, the following methods return an async iterator yielding - :class:`~.Path` objects: - - * :meth:`~pathlib.Path.glob` - * :meth:`~pathlib.Path.iterdir` - * :meth:`~pathlib.Path.rglob` - """ - - __slots__ = "_path", "__weakref__" - - __weakref__: Any - - def __init__(self, *args: str | PathLike[str]) -> None: - self._path: Final[pathlib.Path] = pathlib.Path(*args) - - def __fspath__(self) -> str: - return self._path.__fspath__() - - def __str__(self) -> str: - return self._path.__str__() - - def __repr__(self) -> str: - return f"{self.__class__.__name__}({self.as_posix()!r})" - - def __bytes__(self) -> bytes: - return self._path.__bytes__() - - def __hash__(self) -> int: - return self._path.__hash__() - - def __eq__(self, other: object) -> bool: - target = other._path if isinstance(other, Path) else other - return self._path.__eq__(target) - - def __lt__(self, other: pathlib.PurePath | Path) -> bool: - target = other._path if isinstance(other, Path) else other - return self._path.__lt__(target) - - def __le__(self, other: pathlib.PurePath | Path) -> bool: - target = other._path if isinstance(other, Path) else other - return self._path.__le__(target) - - def __gt__(self, other: pathlib.PurePath | Path) -> bool: - target = other._path if isinstance(other, Path) else other - return self._path.__gt__(target) - - def __ge__(self, other: pathlib.PurePath | Path) -> bool: - target = other._path if isinstance(other, Path) else other - return self._path.__ge__(target) - - def __truediv__(self, other: str | PathLike[str]) -> Path: - return Path(self._path / other) - - def __rtruediv__(self, other: str | PathLike[str]) -> Path: - return Path(other) / self - - @property - def parts(self) -> tuple[str, ...]: - return self._path.parts - - @property - def drive(self) -> str: - return self._path.drive - - @property - def root(self) -> str: - return self._path.root - - @property - def anchor(self) -> str: - return self._path.anchor - - @property - def parents(self) -> Sequence[Path]: - return tuple(Path(p) for p in self._path.parents) - - @property - def parent(self) -> Path: - return Path(self._path.parent) - - @property - def name(self) -> str: - return self._path.name - - @property - def suffix(self) -> str: - return self._path.suffix - - @property - def suffixes(self) -> list[str]: - return self._path.suffixes - - @property - def stem(self) -> str: - return self._path.stem - - async def absolute(self) -> Path: - path = await to_thread.run_sync(self._path.absolute) - return Path(path) - - def as_posix(self) -> str: - return self._path.as_posix() - - def as_uri(self) -> str: - return self._path.as_uri() - - if sys.version_info >= (3, 13): - parser: ClassVar[ModuleType] = pathlib.Path.parser - - @classmethod - def from_uri(cls, uri: str) -> Path: - return Path(pathlib.Path.from_uri(uri)) - - def full_match( - self, path_pattern: str, *, case_sensitive: bool | None = None - ) -> bool: - return self._path.full_match(path_pattern, case_sensitive=case_sensitive) - - def match( - self, path_pattern: str, *, case_sensitive: bool | None = None - ) -> bool: - return self._path.match(path_pattern, case_sensitive=case_sensitive) - else: - - def match(self, path_pattern: str) -> bool: - return self._path.match(path_pattern) - - if sys.version_info >= (3, 14): - - @property - def info(self) -> Any: # TODO: add return type annotation when Typeshed gets it - return self._path.info - - async def copy( - self, - target: str | os.PathLike[str], - *, - follow_symlinks: bool = True, - preserve_metadata: bool = False, - ) -> Path: - func = partial( - self._path.copy, - follow_symlinks=follow_symlinks, - preserve_metadata=preserve_metadata, - ) - return Path(await to_thread.run_sync(func, pathlib.Path(target))) - - async def copy_into( - self, - target_dir: str | os.PathLike[str], - *, - follow_symlinks: bool = True, - preserve_metadata: bool = False, - ) -> Path: - func = partial( - self._path.copy_into, - follow_symlinks=follow_symlinks, - preserve_metadata=preserve_metadata, - ) - return Path(await to_thread.run_sync(func, pathlib.Path(target_dir))) - - async def move(self, target: str | os.PathLike[str]) -> Path: - # Upstream does not handle anyio.Path properly as a PathLike - target = pathlib.Path(target) - return Path(await to_thread.run_sync(self._path.move, target)) - - async def move_into( - self, - target_dir: str | os.PathLike[str], - ) -> Path: - return Path(await to_thread.run_sync(self._path.move_into, target_dir)) - - def is_relative_to(self, other: str | PathLike[str]) -> bool: - try: - self.relative_to(other) - return True - except ValueError: - return False - - async def chmod(self, mode: int, *, follow_symlinks: bool = True) -> None: - func = partial(os.chmod, follow_symlinks=follow_symlinks) - return await to_thread.run_sync(func, self._path, mode) - - @classmethod - async def cwd(cls) -> Path: - path = await to_thread.run_sync(pathlib.Path.cwd) - return cls(path) - - async def exists(self) -> bool: - return await to_thread.run_sync(self._path.exists, abandon_on_cancel=True) - - async def expanduser(self) -> Path: - return Path( - await to_thread.run_sync(self._path.expanduser, abandon_on_cancel=True) - ) - - if sys.version_info < (3, 12): - # Python 3.11 and earlier - def glob(self, pattern: str) -> AsyncIterator[Path]: - gen = self._path.glob(pattern) - return _PathIterator(gen) - elif (3, 12) <= sys.version_info < (3, 13): - # changed in Python 3.12: - # - The case_sensitive parameter was added. - def glob( - self, - pattern: str, - *, - case_sensitive: bool | None = None, - ) -> AsyncIterator[Path]: - gen = self._path.glob(pattern, case_sensitive=case_sensitive) - return _PathIterator(gen) - elif sys.version_info >= (3, 13): - # Changed in Python 3.13: - # - The recurse_symlinks parameter was added. - # - The pattern parameter accepts a path-like object. - def glob( # type: ignore[misc] # mypy doesn't allow for differing signatures in a conditional block - self, - pattern: str | PathLike[str], - *, - case_sensitive: bool | None = None, - recurse_symlinks: bool = False, - ) -> AsyncIterator[Path]: - gen = self._path.glob( - pattern, # type: ignore[arg-type] - case_sensitive=case_sensitive, - recurse_symlinks=recurse_symlinks, - ) - return _PathIterator(gen) - - async def group(self) -> str: - return await to_thread.run_sync(self._path.group, abandon_on_cancel=True) - - async def hardlink_to( - self, target: str | bytes | PathLike[str] | PathLike[bytes] - ) -> None: - if isinstance(target, Path): - target = target._path - - await to_thread.run_sync(os.link, target, self) - - @classmethod - async def home(cls) -> Path: - home_path = await to_thread.run_sync(pathlib.Path.home) - return cls(home_path) - - def is_absolute(self) -> bool: - return self._path.is_absolute() - - async def is_block_device(self) -> bool: - return await to_thread.run_sync( - self._path.is_block_device, abandon_on_cancel=True - ) - - async def is_char_device(self) -> bool: - return await to_thread.run_sync( - self._path.is_char_device, abandon_on_cancel=True - ) - - async def is_dir(self) -> bool: - return await to_thread.run_sync(self._path.is_dir, abandon_on_cancel=True) - - async def is_fifo(self) -> bool: - return await to_thread.run_sync(self._path.is_fifo, abandon_on_cancel=True) - - async def is_file(self) -> bool: - return await to_thread.run_sync(self._path.is_file, abandon_on_cancel=True) - - if sys.version_info >= (3, 12): - - async def is_junction(self) -> bool: - return await to_thread.run_sync(self._path.is_junction) - - async def is_mount(self) -> bool: - return await to_thread.run_sync( - os.path.ismount, self._path, abandon_on_cancel=True - ) - - def is_reserved(self) -> bool: - return self._path.is_reserved() - - async def is_socket(self) -> bool: - return await to_thread.run_sync(self._path.is_socket, abandon_on_cancel=True) - - async def is_symlink(self) -> bool: - return await to_thread.run_sync(self._path.is_symlink, abandon_on_cancel=True) - - async def iterdir(self) -> AsyncIterator[Path]: - gen = ( - self._path.iterdir() - if sys.version_info < (3, 13) - else await to_thread.run_sync(self._path.iterdir, abandon_on_cancel=True) - ) - async for path in _PathIterator(gen): - yield path - - def joinpath(self, *args: str | PathLike[str]) -> Path: - return Path(self._path.joinpath(*args)) - - async def lchmod(self, mode: int) -> None: - await to_thread.run_sync(self._path.lchmod, mode) - - async def lstat(self) -> os.stat_result: - return await to_thread.run_sync(self._path.lstat, abandon_on_cancel=True) - - async def mkdir( - self, mode: int = 0o777, parents: bool = False, exist_ok: bool = False - ) -> None: - await to_thread.run_sync(self._path.mkdir, mode, parents, exist_ok) - - @overload - async def open( - self, - mode: OpenBinaryMode, - buffering: int = ..., - encoding: str | None = ..., - errors: str | None = ..., - newline: str | None = ..., - ) -> AsyncFile[bytes]: ... - - @overload - async def open( - self, - mode: OpenTextMode = ..., - buffering: int = ..., - encoding: str | None = ..., - errors: str | None = ..., - newline: str | None = ..., - ) -> AsyncFile[str]: ... - - async def open( - self, - mode: str = "r", - buffering: int = -1, - encoding: str | None = None, - errors: str | None = None, - newline: str | None = None, - ) -> AsyncFile[Any]: - fp = await to_thread.run_sync( - self._path.open, mode, buffering, encoding, errors, newline - ) - return AsyncFile(fp) - - async def owner(self) -> str: - return await to_thread.run_sync(self._path.owner, abandon_on_cancel=True) - - async def read_bytes(self) -> bytes: - return await to_thread.run_sync(self._path.read_bytes) - - async def read_text( - self, encoding: str | None = None, errors: str | None = None - ) -> str: - return await to_thread.run_sync(self._path.read_text, encoding, errors) - - if sys.version_info >= (3, 12): - - def relative_to( - self, *other: str | PathLike[str], walk_up: bool = False - ) -> Path: - # relative_to() should work with any PathLike but it doesn't - others = [pathlib.Path(other) for other in other] - return Path(self._path.relative_to(*others, walk_up=walk_up)) - - else: - - def relative_to(self, *other: str | PathLike[str]) -> Path: - return Path(self._path.relative_to(*other)) - - async def readlink(self) -> Path: - target = await to_thread.run_sync(os.readlink, self._path) - return Path(target) - - async def rename(self, target: str | pathlib.PurePath | Path) -> Path: - if isinstance(target, Path): - target = target._path - - await to_thread.run_sync(self._path.rename, target) - return Path(target) - - async def replace(self, target: str | pathlib.PurePath | Path) -> Path: - if isinstance(target, Path): - target = target._path - - await to_thread.run_sync(self._path.replace, target) - return Path(target) - - async def resolve(self, strict: bool = False) -> Path: - func = partial(self._path.resolve, strict=strict) - return Path(await to_thread.run_sync(func, abandon_on_cancel=True)) - - if sys.version_info < (3, 12): - # Pre Python 3.12 - def rglob(self, pattern: str) -> AsyncIterator[Path]: - gen = self._path.rglob(pattern) - return _PathIterator(gen) - elif (3, 12) <= sys.version_info < (3, 13): - # Changed in Python 3.12: - # - The case_sensitive parameter was added. - def rglob( - self, pattern: str, *, case_sensitive: bool | None = None - ) -> AsyncIterator[Path]: - gen = self._path.rglob(pattern, case_sensitive=case_sensitive) - return _PathIterator(gen) - elif sys.version_info >= (3, 13): - # Changed in Python 3.13: - # - The recurse_symlinks parameter was added. - # - The pattern parameter accepts a path-like object. - def rglob( # type: ignore[misc] # mypy doesn't allow for differing signatures in a conditional block - self, - pattern: str | PathLike[str], - *, - case_sensitive: bool | None = None, - recurse_symlinks: bool = False, - ) -> AsyncIterator[Path]: - gen = self._path.rglob( - pattern, # type: ignore[arg-type] - case_sensitive=case_sensitive, - recurse_symlinks=recurse_symlinks, - ) - return _PathIterator(gen) - - async def rmdir(self) -> None: - await to_thread.run_sync(self._path.rmdir) - - async def samefile(self, other_path: str | PathLike[str]) -> bool: - if isinstance(other_path, Path): - other_path = other_path._path - - return await to_thread.run_sync( - self._path.samefile, other_path, abandon_on_cancel=True - ) - - async def stat(self, *, follow_symlinks: bool = True) -> os.stat_result: - func = partial(os.stat, follow_symlinks=follow_symlinks) - return await to_thread.run_sync(func, self._path, abandon_on_cancel=True) - - async def symlink_to( - self, - target: str | bytes | PathLike[str] | PathLike[bytes], - target_is_directory: bool = False, - ) -> None: - if isinstance(target, Path): - target = target._path - - await to_thread.run_sync(self._path.symlink_to, target, target_is_directory) - - async def touch(self, mode: int = 0o666, exist_ok: bool = True) -> None: - await to_thread.run_sync(self._path.touch, mode, exist_ok) - - async def unlink(self, missing_ok: bool = False) -> None: - try: - await to_thread.run_sync(self._path.unlink) - except FileNotFoundError: - if not missing_ok: - raise - - if sys.version_info >= (3, 12): - - async def walk( - self, - top_down: bool = True, - on_error: Callable[[OSError], object] | None = None, - follow_symlinks: bool = False, - ) -> AsyncIterator[tuple[Path, list[str], list[str]]]: - def get_next_value() -> tuple[pathlib.Path, list[str], list[str]] | None: - try: - return next(gen) - except StopIteration: - return None - - gen = self._path.walk(top_down, on_error, follow_symlinks) - while True: - value = await to_thread.run_sync(get_next_value) - if value is None: - return - - root, dirs, paths = value - yield Path(root), dirs, paths - - def with_name(self, name: str) -> Path: - return Path(self._path.with_name(name)) - - def with_stem(self, stem: str) -> Path: - return Path(self._path.with_name(stem + self._path.suffix)) - - def with_suffix(self, suffix: str) -> Path: - return Path(self._path.with_suffix(suffix)) - - def with_segments(self, *pathsegments: str | PathLike[str]) -> Path: - return Path(*pathsegments) - - async def write_bytes(self, data: bytes) -> int: - return await to_thread.run_sync(self._path.write_bytes, data) - - async def write_text( - self, - data: str, - encoding: str | None = None, - errors: str | None = None, - newline: str | None = None, - ) -> int: - # Path.write_text() does not support the "newline" parameter before Python 3.10 - def sync_write_text() -> int: - with self._path.open( - "w", encoding=encoding, errors=errors, newline=newline - ) as fp: - return fp.write(data) - - return await to_thread.run_sync(sync_write_text) - - -PathLike.register(Path) diff --git a/.env/Lib/site-packages/anyio/_core/_resources.py b/.env/Lib/site-packages/anyio/_core/_resources.py deleted file mode 100644 index b9a5344..0000000 --- a/.env/Lib/site-packages/anyio/_core/_resources.py +++ /dev/null @@ -1,18 +0,0 @@ -from __future__ import annotations - -from ..abc import AsyncResource -from ._tasks import CancelScope - - -async def aclose_forcefully(resource: AsyncResource) -> None: - """ - Close an asynchronous resource in a cancelled scope. - - Doing this closes the resource without waiting on anything. - - :param resource: the resource to close - - """ - with CancelScope() as scope: - scope.cancel() - await resource.aclose() diff --git a/.env/Lib/site-packages/anyio/_core/_signals.py b/.env/Lib/site-packages/anyio/_core/_signals.py deleted file mode 100644 index e24c79e..0000000 --- a/.env/Lib/site-packages/anyio/_core/_signals.py +++ /dev/null @@ -1,29 +0,0 @@ -from __future__ import annotations - -from collections.abc import AsyncIterator -from contextlib import AbstractContextManager -from signal import Signals - -from ._eventloop import get_async_backend - - -def open_signal_receiver( - *signals: Signals, -) -> AbstractContextManager[AsyncIterator[Signals]]: - """ - Start receiving operating system signals. - - :param signals: signals to receive (e.g. ``signal.SIGINT``) - :return: an asynchronous context manager for an asynchronous iterator which yields - signal numbers - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - .. warning:: Windows does not support signals natively so it is best to avoid - relying on this in cross-platform applications. - - .. warning:: On asyncio, this permanently replaces any previous signal handler for - the given signals, as set via :meth:`~asyncio.loop.add_signal_handler`. - - """ - return get_async_backend().open_signal_receiver(*signals) diff --git a/.env/Lib/site-packages/anyio/_core/_sockets.py b/.env/Lib/site-packages/anyio/_core/_sockets.py deleted file mode 100644 index 6c99b3a..0000000 --- a/.env/Lib/site-packages/anyio/_core/_sockets.py +++ /dev/null @@ -1,1003 +0,0 @@ -from __future__ import annotations - -import errno -import os -import socket -import ssl -import stat -import sys -from collections.abc import Awaitable -from dataclasses import dataclass -from ipaddress import IPv4Address, IPv6Address, ip_address -from os import PathLike, chmod -from socket import AddressFamily, SocketKind -from typing import TYPE_CHECKING, Any, Literal, cast, overload - -from .. import ConnectionFailed, to_thread -from ..abc import ( - ByteStreamConnectable, - ConnectedUDPSocket, - ConnectedUNIXDatagramSocket, - IPAddressType, - IPSockAddrType, - SocketListener, - SocketStream, - UDPSocket, - UNIXDatagramSocket, - UNIXSocketStream, -) -from ..streams.stapled import MultiListener -from ..streams.tls import TLSConnectable, TLSStream -from ._eventloop import get_async_backend -from ._resources import aclose_forcefully -from ._synchronization import Event -from ._tasks import create_task_group, move_on_after - -if TYPE_CHECKING: - from _typeshed import FileDescriptorLike -else: - FileDescriptorLike = object - -if sys.version_info < (3, 11): - from exceptiongroup import ExceptionGroup - -if sys.version_info >= (3, 12): - from typing import override -else: - from typing_extensions import override - -if sys.version_info < (3, 13): - from typing_extensions import deprecated -else: - from warnings import deprecated - -IPPROTO_IPV6 = getattr(socket, "IPPROTO_IPV6", 41) # https://bugs.python.org/issue29515 - -AnyIPAddressFamily = Literal[ - AddressFamily.AF_UNSPEC, AddressFamily.AF_INET, AddressFamily.AF_INET6 -] -IPAddressFamily = Literal[AddressFamily.AF_INET, AddressFamily.AF_INET6] - - -# tls_hostname given -@overload -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = ..., - ssl_context: ssl.SSLContext | None = ..., - tls_standard_compatible: bool = ..., - tls_hostname: str, - happy_eyeballs_delay: float = ..., -) -> TLSStream: ... - - -# ssl_context given -@overload -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = ..., - ssl_context: ssl.SSLContext, - tls_standard_compatible: bool = ..., - tls_hostname: str | None = ..., - happy_eyeballs_delay: float = ..., -) -> TLSStream: ... - - -# tls=True -@overload -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = ..., - tls: Literal[True], - ssl_context: ssl.SSLContext | None = ..., - tls_standard_compatible: bool = ..., - tls_hostname: str | None = ..., - happy_eyeballs_delay: float = ..., -) -> TLSStream: ... - - -# tls=False -@overload -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = ..., - tls: Literal[False], - ssl_context: ssl.SSLContext | None = ..., - tls_standard_compatible: bool = ..., - tls_hostname: str | None = ..., - happy_eyeballs_delay: float = ..., -) -> SocketStream: ... - - -# No TLS arguments -@overload -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = ..., - happy_eyeballs_delay: float = ..., -) -> SocketStream: ... - - -async def connect_tcp( - remote_host: IPAddressType, - remote_port: int, - *, - local_host: IPAddressType | None = None, - tls: bool = False, - ssl_context: ssl.SSLContext | None = None, - tls_standard_compatible: bool = True, - tls_hostname: str | None = None, - happy_eyeballs_delay: float = 0.25, -) -> SocketStream | TLSStream: - """ - Connect to a host using the TCP protocol. - - This function implements the stateless version of the Happy Eyeballs algorithm (RFC - 6555). If ``remote_host`` is a host name that resolves to multiple IP addresses, - each one is tried until one connection attempt succeeds. If the first attempt does - not connected within 250 milliseconds, a second attempt is started using the next - address in the list, and so on. On IPv6 enabled systems, an IPv6 address (if - available) is tried first. - - When the connection has been established, a TLS handshake will be done if either - ``ssl_context`` or ``tls_hostname`` is not ``None``, or if ``tls`` is ``True``. - - :param remote_host: the IP address or host name to connect to - :param remote_port: port on the target host to connect to - :param local_host: the interface address or name to bind the socket to before - connecting - :param tls: ``True`` to do a TLS handshake with the connected stream and return a - :class:`~anyio.streams.tls.TLSStream` instead - :param ssl_context: the SSL context object to use (if omitted, a default context is - created) - :param tls_standard_compatible: If ``True``, performs the TLS shutdown handshake - before closing the stream and requires that the server does this as well. - Otherwise, :exc:`~ssl.SSLEOFError` may be raised during reads from the stream. - Some protocols, such as HTTP, require this option to be ``False``. - See :meth:`~ssl.SSLContext.wrap_socket` for details. - :param tls_hostname: host name to check the server certificate against (defaults to - the value of ``remote_host``) - :param happy_eyeballs_delay: delay (in seconds) before starting the next connection - attempt - :return: a socket stream object if no TLS handshake was done, otherwise a TLS stream - :raises ConnectionFailed: if the connection fails - - """ - # Placed here due to https://github.com/python/mypy/issues/7057 - connected_stream: SocketStream | None = None - - async def try_connect(remote_host: str, event: Event) -> None: - nonlocal connected_stream - try: - stream = await asynclib.connect_tcp(remote_host, remote_port, local_address) - except OSError as exc: - oserrors.append(exc) - return - else: - if connected_stream is None: - connected_stream = stream - tg.cancel_scope.cancel() - else: - await stream.aclose() - finally: - event.set() - - asynclib = get_async_backend() - local_address: IPSockAddrType | None = None - family = socket.AF_UNSPEC - if local_host: - gai_res = await getaddrinfo(str(local_host), None) - family, *_, local_address = gai_res[0] - - target_host = str(remote_host) - try: - addr_obj = ip_address(remote_host) - except ValueError: - addr_obj = None - - if addr_obj is not None: - if isinstance(addr_obj, IPv6Address): - target_addrs = [(socket.AF_INET6, addr_obj.compressed)] - else: - target_addrs = [(socket.AF_INET, addr_obj.compressed)] - else: - # getaddrinfo() will raise an exception if name resolution fails - gai_res = await getaddrinfo( - target_host, remote_port, family=family, type=socket.SOCK_STREAM - ) - - # Organize the list so that the first address is an IPv6 address (if available) - # and the second one is an IPv4 addresses. The rest can be in whatever order. - v6_found = v4_found = False - target_addrs = [] - for af, *_, sa in gai_res: - if af == socket.AF_INET6 and not v6_found: - v6_found = True - target_addrs.insert(0, (af, sa[0])) - elif af == socket.AF_INET and not v4_found and v6_found: - v4_found = True - target_addrs.insert(1, (af, sa[0])) - else: - target_addrs.append((af, sa[0])) - - oserrors: list[OSError] = [] - try: - async with create_task_group() as tg: - for _af, addr in target_addrs: - event = Event() - tg.start_soon(try_connect, addr, event) - with move_on_after(happy_eyeballs_delay): - await event.wait() - - if connected_stream is None: - cause = ( - oserrors[0] - if len(oserrors) == 1 - else ExceptionGroup("multiple connection attempts failed", oserrors) - ) - raise OSError("All connection attempts failed") from cause - finally: - oserrors.clear() - - if tls or tls_hostname or ssl_context: - try: - return await TLSStream.wrap( - connected_stream, - server_side=False, - hostname=tls_hostname or str(remote_host), - ssl_context=ssl_context, - standard_compatible=tls_standard_compatible, - ) - except BaseException: - await aclose_forcefully(connected_stream) - raise - - return connected_stream - - -async def connect_unix(path: str | bytes | PathLike[Any]) -> UNIXSocketStream: - """ - Connect to the given UNIX socket. - - Not available on Windows. - - :param path: path to the socket - :return: a socket stream object - :raises ConnectionFailed: if the connection fails - - """ - path = os.fspath(path) - return await get_async_backend().connect_unix(path) - - -async def create_tcp_listener( - *, - local_host: IPAddressType | None = None, - local_port: int = 0, - family: AnyIPAddressFamily = socket.AddressFamily.AF_UNSPEC, - backlog: int = 65536, - reuse_port: bool = False, -) -> MultiListener[SocketStream]: - """ - Create a TCP socket listener. - - :param local_port: port number to listen on - :param local_host: IP address of the interface to listen on. If omitted, listen on - all IPv4 and IPv6 interfaces. To listen on all interfaces on a specific address - family, use ``0.0.0.0`` for IPv4 or ``::`` for IPv6. - :param family: address family (used if ``local_host`` was omitted) - :param backlog: maximum number of queued incoming connections (up to a maximum of - 2**16, or 65536) - :param reuse_port: ``True`` to allow multiple sockets to bind to the same - address/port (not supported on Windows) - :return: a multi-listener object containing one or more socket listeners - :raises OSError: if there's an error creating a socket, or binding to one or more - interfaces failed - - """ - asynclib = get_async_backend() - backlog = min(backlog, 65536) - local_host = str(local_host) if local_host is not None else None - - def setup_raw_socket( - fam: AddressFamily, - bind_addr: tuple[str, int] | tuple[str, int, int, int], - *, - v6only: bool = True, - ) -> socket.socket: - sock = socket.socket(fam) - try: - sock.setblocking(False) - - if fam == AddressFamily.AF_INET6: - sock.setsockopt(IPPROTO_IPV6, socket.IPV6_V6ONLY, v6only) - - # For Windows, enable exclusive address use. For others, enable address - # reuse. - if sys.platform == "win32": - sock.setsockopt(socket.SOL_SOCKET, socket.SO_EXCLUSIVEADDRUSE, 1) - else: - sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) - - if reuse_port: - sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT, 1) - - # Workaround for #554 - if fam == socket.AF_INET6 and "%" in bind_addr[0]: - addr, scope_id = bind_addr[0].split("%", 1) - bind_addr = (addr, bind_addr[1], 0, int(scope_id)) - - sock.bind(bind_addr) - sock.listen(backlog) - except BaseException: - sock.close() - raise - - return sock - - # We passing type=0 on non-Windows platforms as a workaround for a uvloop bug - # where we don't get the correct scope ID for IPv6 link-local addresses when passing - # type=socket.SOCK_STREAM to getaddrinfo(): - # https://github.com/MagicStack/uvloop/issues/539 - gai_res = await getaddrinfo( - local_host, - local_port, - family=family, - type=socket.SOCK_STREAM if sys.platform == "win32" else 0, - flags=socket.AI_PASSIVE | socket.AI_ADDRCONFIG, - ) - - # The set comprehension is here to work around a glibc bug: - # https://sourceware.org/bugzilla/show_bug.cgi?id=14969 - sockaddrs = sorted({res for res in gai_res if res[1] == SocketKind.SOCK_STREAM}) - - # Special case for dual-stack binding on the "any" interface - if ( - local_host is None - and family == AddressFamily.AF_UNSPEC - and socket.has_dualstack_ipv6() - and any(fam == AddressFamily.AF_INET6 for fam, *_ in gai_res) - ): - raw_socket = setup_raw_socket( - AddressFamily.AF_INET6, ("::", local_port), v6only=False - ) - listener = asynclib.create_tcp_listener(raw_socket) - return MultiListener([listener]) - - errors: list[OSError] = [] - try: - for _ in range(len(sockaddrs)): - listeners: list[SocketListener] = [] - bound_ephemeral_port = local_port - try: - for fam, *_, sockaddr in sockaddrs: - sockaddr = sockaddr[0], bound_ephemeral_port, *sockaddr[2:] - raw_socket = setup_raw_socket(fam, sockaddr) - - # Store the assigned port if an ephemeral port was requested, so - # we'll bind to the same port on all interfaces - if local_port == 0 and len(gai_res) > 1: - bound_ephemeral_port = raw_socket.getsockname()[1] - - listeners.append(asynclib.create_tcp_listener(raw_socket)) - except BaseException as exc: - for listener in listeners: - await listener.aclose() - - # If an ephemeral port was requested but binding the assigned port - # failed for another interface, rotate the address list and try again - if ( - isinstance(exc, OSError) - and exc.errno == errno.EADDRINUSE - and local_port == 0 - and bound_ephemeral_port - ): - errors.append(exc) - sockaddrs.append(sockaddrs.pop(0)) - continue - - raise - - return MultiListener(listeners) - - raise OSError( - f"Could not create {len(sockaddrs)} listeners with a consistent port" - ) from ExceptionGroup("Several bind attempts failed", errors) - finally: - del errors # Prevent reference cycles - - -async def create_unix_listener( - path: str | bytes | PathLike[Any], - *, - mode: int | None = None, - backlog: int = 65536, -) -> SocketListener: - """ - Create a UNIX socket listener. - - Not available on Windows. - - :param path: path of the socket - :param mode: permissions to set on the socket - :param backlog: maximum number of queued incoming connections (up to a maximum of - 2**16, or 65536) - :return: a listener object - - .. versionchanged:: 3.0 - If a socket already exists on the file system in the given path, it will be - removed first. - - """ - backlog = min(backlog, 65536) - raw_socket = await setup_unix_local_socket(path, mode, socket.SOCK_STREAM) - try: - raw_socket.listen(backlog) - return get_async_backend().create_unix_listener(raw_socket) - except BaseException: - raw_socket.close() - raise - - -async def create_udp_socket( - family: AnyIPAddressFamily = AddressFamily.AF_UNSPEC, - *, - local_host: IPAddressType | None = None, - local_port: int = 0, - reuse_port: bool = False, -) -> UDPSocket: - """ - Create a UDP socket. - - If ``port`` has been given, the socket will be bound to this port on the local - machine, making this socket suitable for providing UDP based services. - - :param family: address family (``AF_INET`` or ``AF_INET6``) – automatically - determined from ``local_host`` if omitted - :param local_host: IP address or host name of the local interface to bind to - :param local_port: local port to bind to - :param reuse_port: ``True`` to allow multiple sockets to bind to the same - address/port (not supported on Windows) - :return: a UDP socket - - """ - if family is AddressFamily.AF_UNSPEC and not local_host: - raise ValueError('Either "family" or "local_host" must be given') - - if local_host: - gai_res = await getaddrinfo( - str(local_host), - local_port, - family=family, - type=socket.SOCK_DGRAM, - flags=socket.AI_PASSIVE | socket.AI_ADDRCONFIG, - ) - family = cast(AnyIPAddressFamily, gai_res[0][0]) - local_address = gai_res[0][-1] - elif family is AddressFamily.AF_INET6: - local_address = ("::", 0) - else: - local_address = ("0.0.0.0", 0) - - sock = await get_async_backend().create_udp_socket( - family, local_address, None, reuse_port - ) - return cast(UDPSocket, sock) - - -async def create_connected_udp_socket( - remote_host: IPAddressType, - remote_port: int, - *, - family: AnyIPAddressFamily = AddressFamily.AF_UNSPEC, - local_host: IPAddressType | None = None, - local_port: int = 0, - reuse_port: bool = False, -) -> ConnectedUDPSocket: - """ - Create a connected UDP socket. - - Connected UDP sockets can only communicate with the specified remote host/port, an - any packets sent from other sources are dropped. - - :param remote_host: remote host to set as the default target - :param remote_port: port on the remote host to set as the default target - :param family: address family (``AF_INET`` or ``AF_INET6``) – automatically - determined from ``local_host`` or ``remote_host`` if omitted - :param local_host: IP address or host name of the local interface to bind to - :param local_port: local port to bind to - :param reuse_port: ``True`` to allow multiple sockets to bind to the same - address/port (not supported on Windows) - :return: a connected UDP socket - - """ - local_address = None - if local_host: - gai_res = await getaddrinfo( - str(local_host), - local_port, - family=family, - type=socket.SOCK_DGRAM, - flags=socket.AI_PASSIVE | socket.AI_ADDRCONFIG, - ) - family = cast(AnyIPAddressFamily, gai_res[0][0]) - local_address = gai_res[0][-1] - - gai_res = await getaddrinfo( - str(remote_host), remote_port, family=family, type=socket.SOCK_DGRAM - ) - family = cast(AnyIPAddressFamily, gai_res[0][0]) - remote_address = gai_res[0][-1] - - sock = await get_async_backend().create_udp_socket( - family, local_address, remote_address, reuse_port - ) - return cast(ConnectedUDPSocket, sock) - - -async def create_unix_datagram_socket( - *, - local_path: None | str | bytes | PathLike[Any] = None, - local_mode: int | None = None, -) -> UNIXDatagramSocket: - """ - Create a UNIX datagram socket. - - Not available on Windows. - - If ``local_path`` has been given, the socket will be bound to this path, making this - socket suitable for receiving datagrams from other processes. Other processes can - send datagrams to this socket only if ``local_path`` is set. - - If a socket already exists on the file system in the ``local_path``, it will be - removed first. - - :param local_path: the path on which to bind to - :param local_mode: permissions to set on the local socket - :return: a UNIX datagram socket - - """ - raw_socket = await setup_unix_local_socket( - local_path, local_mode, socket.SOCK_DGRAM - ) - return await get_async_backend().create_unix_datagram_socket(raw_socket, None) - - -async def create_connected_unix_datagram_socket( - remote_path: str | bytes | PathLike[Any], - *, - local_path: None | str | bytes | PathLike[Any] = None, - local_mode: int | None = None, -) -> ConnectedUNIXDatagramSocket: - """ - Create a connected UNIX datagram socket. - - Connected datagram sockets can only communicate with the specified remote path. - - If ``local_path`` has been given, the socket will be bound to this path, making - this socket suitable for receiving datagrams from other processes. Other processes - can send datagrams to this socket only if ``local_path`` is set. - - If a socket already exists on the file system in the ``local_path``, it will be - removed first. - - :param remote_path: the path to set as the default target - :param local_path: the path on which to bind to - :param local_mode: permissions to set on the local socket - :return: a connected UNIX datagram socket - - """ - remote_path = os.fspath(remote_path) - raw_socket = await setup_unix_local_socket( - local_path, local_mode, socket.SOCK_DGRAM - ) - return await get_async_backend().create_unix_datagram_socket( - raw_socket, remote_path - ) - - -async def getaddrinfo( - host: bytes | str | None, - port: str | int | None, - *, - family: int | AddressFamily = 0, - type: int | SocketKind = 0, - proto: int = 0, - flags: int = 0, -) -> list[tuple[AddressFamily, SocketKind, int, str, tuple[str, int]]]: - """ - Look up a numeric IP address given a host name. - - Internationalized domain names are translated according to the (non-transitional) - IDNA 2008 standard. - - .. note:: 4-tuple IPv6 socket addresses are automatically converted to 2-tuples of - (host, port), unlike what :func:`socket.getaddrinfo` does. - - :param host: host name - :param port: port number - :param family: socket family (`'AF_INET``, ...) - :param type: socket type (``SOCK_STREAM``, ...) - :param proto: protocol number - :param flags: flags to pass to upstream ``getaddrinfo()`` - :return: list of tuples containing (family, type, proto, canonname, sockaddr) - - .. seealso:: :func:`socket.getaddrinfo` - - """ - # Handle unicode hostnames - if isinstance(host, str): - try: - encoded_host: bytes | None = host.encode("ascii") - except UnicodeEncodeError: - import idna - - encoded_host = idna.encode(host, uts46=True) - else: - encoded_host = host - - gai_res = await get_async_backend().getaddrinfo( - encoded_host, port, family=family, type=type, proto=proto, flags=flags - ) - return [ - (family, type, proto, canonname, convert_ipv6_sockaddr(sockaddr)) - for family, type, proto, canonname, sockaddr in gai_res - # filter out IPv6 results when IPv6 is disabled - if not isinstance(sockaddr[0], int) - ] - - -def getnameinfo(sockaddr: IPSockAddrType, flags: int = 0) -> Awaitable[tuple[str, str]]: - """ - Look up the host name of an IP address. - - :param sockaddr: socket address (e.g. (ipaddress, port) for IPv4) - :param flags: flags to pass to upstream ``getnameinfo()`` - :return: a tuple of (host name, service name) - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - .. seealso:: :func:`socket.getnameinfo` - - """ - return get_async_backend().getnameinfo(sockaddr, flags) - - -@deprecated("This function is deprecated; use `wait_readable` instead") -def wait_socket_readable(sock: socket.socket) -> Awaitable[None]: - """ - .. deprecated:: 4.7.0 - Use :func:`wait_readable` instead. - - Wait until the given socket has data to be read. - - .. warning:: Only use this on raw sockets that have not been wrapped by any higher - level constructs like socket streams! - - :param sock: a socket object - :raises ~anyio.ClosedResourceError: if the socket was closed while waiting for the - socket to become readable - :raises ~anyio.BusyResourceError: if another task is already waiting for the socket - to become readable - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().wait_readable(sock.fileno()) - - -@deprecated("This function is deprecated; use `wait_writable` instead") -def wait_socket_writable(sock: socket.socket) -> Awaitable[None]: - """ - .. deprecated:: 4.7.0 - Use :func:`wait_writable` instead. - - Wait until the given socket can be written to. - - This does **NOT** work on Windows when using the asyncio backend with a proactor - event loop (default on py3.8+). - - .. warning:: Only use this on raw sockets that have not been wrapped by any higher - level constructs like socket streams! - - :param sock: a socket object - :raises ~anyio.ClosedResourceError: if the socket was closed while waiting for the - socket to become writable - :raises ~anyio.BusyResourceError: if another task is already waiting for the socket - to become writable - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().wait_writable(sock.fileno()) - - -def wait_readable(obj: FileDescriptorLike) -> Awaitable[None]: - """ - Wait until the given object has data to be read. - - On Unix systems, ``obj`` must either be an integer file descriptor, or else an - object with a ``.fileno()`` method which returns an integer file descriptor. Any - kind of file descriptor can be passed, though the exact semantics will depend on - your kernel. For example, this probably won't do anything useful for on-disk files. - - On Windows systems, ``obj`` must either be an integer ``SOCKET`` handle, or else an - object with a ``.fileno()`` method which returns an integer ``SOCKET`` handle. File - descriptors aren't supported, and neither are handles that refer to anything besides - a ``SOCKET``. - - On backends where this functionality is not natively provided (asyncio - ``ProactorEventLoop`` on Windows), it is provided using a separate selector thread - which is set to shut down when the interpreter shuts down. - - .. warning:: Don't use this on raw sockets that have been wrapped by any higher - level constructs like socket streams! - - :param obj: an object with a ``.fileno()`` method or an integer handle - :raises ~anyio.ClosedResourceError: if the object was closed while waiting for the - object to become readable - :raises ~anyio.BusyResourceError: if another task is already waiting for the object - to become readable - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().wait_readable(obj) - - -def wait_writable(obj: FileDescriptorLike) -> Awaitable[None]: - """ - Wait until the given object can be written to. - - :param obj: an object with a ``.fileno()`` method or an integer handle - :raises ~anyio.ClosedResourceError: if the object was closed while waiting for the - object to become writable - :raises ~anyio.BusyResourceError: if another task is already waiting for the object - to become writable - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - .. seealso:: See the documentation of :func:`wait_readable` for the definition of - ``obj`` and notes on backend compatibility. - - .. warning:: Don't use this on raw sockets that have been wrapped by any higher - level constructs like socket streams! - - """ - return get_async_backend().wait_writable(obj) - - -def notify_closing(obj: FileDescriptorLike) -> None: - """ - Call this before closing a file descriptor (on Unix) or socket (on - Windows). This will cause any `wait_readable` or `wait_writable` - calls on the given object to immediately wake up and raise - `~anyio.ClosedResourceError`. - - This doesn't actually close the object – you still have to do that - yourself afterwards. Also, you want to be careful to make sure no - new tasks start waiting on the object in between when you call this - and when it's actually closed. So to close something properly, you - usually want to do these steps in order: - - 1. Explicitly mark the object as closed, so that any new attempts - to use it will abort before they start. - 2. Call `notify_closing` to wake up any already-existing users. - 3. Actually close the object. - - It's also possible to do them in a different order if that's more - convenient, *but only if* you make sure not to have any checkpoints in - between the steps. This way they all happen in a single atomic - step, so other tasks won't be able to tell what order they happened - in anyway. - - :param obj: an object with a ``.fileno()`` method or an integer handle - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - get_async_backend().notify_closing(obj) - - -# -# Private API -# - - -def convert_ipv6_sockaddr( - sockaddr: tuple[str, int, int, int] | tuple[str, int], -) -> tuple[str, int]: - """ - Convert a 4-tuple IPv6 socket address to a 2-tuple (address, port) format. - - If the scope ID is nonzero, it is added to the address, separated with ``%``. - Otherwise the flow id and scope id are simply cut off from the tuple. - Any other kinds of socket addresses are returned as-is. - - :param sockaddr: the result of :meth:`~socket.socket.getsockname` - :return: the converted socket address - - """ - # This is more complicated than it should be because of MyPy - if isinstance(sockaddr, tuple) and len(sockaddr) == 4: - host, port, flowinfo, scope_id = sockaddr - if scope_id: - # PyPy (as of v7.3.11) leaves the interface name in the result, so - # we discard it and only get the scope ID from the end - # (https://foss.heptapod.net/pypy/pypy/-/issues/3938) - host = host.split("%")[0] - - # Add scope_id to the address - return f"{host}%{scope_id}", port - else: - return host, port - else: - return sockaddr - - -async def setup_unix_local_socket( - path: None | str | bytes | PathLike[Any], - mode: int | None, - socktype: int, -) -> socket.socket: - """ - Create a UNIX local socket object, deleting the socket at the given path if it - exists. - - Not available on Windows. - - :param path: path of the socket - :param mode: permissions to set on the socket - :param socktype: socket.SOCK_STREAM or socket.SOCK_DGRAM - - """ - path_str: str | None - if path is not None: - path_str = os.fsdecode(path) - - # Linux abstract namespace sockets aren't backed by a concrete file so skip stat call - if not path_str.startswith("\0"): - # Copied from pathlib... - try: - stat_result = os.stat(path) - except OSError as e: - if e.errno not in ( - errno.ENOENT, - errno.ENOTDIR, - errno.EBADF, - errno.ELOOP, - ): - raise - else: - if stat.S_ISSOCK(stat_result.st_mode): - os.unlink(path) - else: - path_str = None - - raw_socket = socket.socket(socket.AF_UNIX, socktype) - raw_socket.setblocking(False) - - if path_str is not None: - try: - await to_thread.run_sync(raw_socket.bind, path_str, abandon_on_cancel=True) - if mode is not None: - await to_thread.run_sync(chmod, path_str, mode, abandon_on_cancel=True) - except BaseException: - raw_socket.close() - raise - - return raw_socket - - -@dataclass -class TCPConnectable(ByteStreamConnectable): - """ - Connects to a TCP server at the given host and port. - - :param host: host name or IP address of the server - :param port: TCP port number of the server - """ - - host: str | IPv4Address | IPv6Address - port: int - - def __post_init__(self) -> None: - if self.port < 1 or self.port > 65535: - raise ValueError("TCP port number out of range") - - @override - async def connect(self) -> SocketStream: - try: - return await connect_tcp(self.host, self.port) - except OSError as exc: - raise ConnectionFailed( - f"error connecting to {self.host}:{self.port}: {exc}" - ) from exc - - -@dataclass -class UNIXConnectable(ByteStreamConnectable): - """ - Connects to a UNIX domain socket at the given path. - - :param path: the file system path of the socket - """ - - path: str | bytes | PathLike[str] | PathLike[bytes] - - @override - async def connect(self) -> UNIXSocketStream: - try: - return await connect_unix(self.path) - except OSError as exc: - raise ConnectionFailed(f"error connecting to {self.path!r}: {exc}") from exc - - -def as_connectable( - remote: ByteStreamConnectable - | tuple[str | IPv4Address | IPv6Address, int] - | str - | bytes - | PathLike[str], - /, - *, - tls: bool = False, - ssl_context: ssl.SSLContext | None = None, - tls_hostname: str | None = None, - tls_standard_compatible: bool = True, -) -> ByteStreamConnectable: - """ - Return a byte stream connectable from the given object. - - If a bytestream connectable is given, it is returned unchanged. - If a tuple of (host, port) is given, a TCP connectable is returned. - If a string or bytes path is given, a UNIX connectable is returned. - - If ``tls=True``, the connectable will be wrapped in a - :class:`~.streams.tls.TLSConnectable`. - - :param remote: a connectable, a tuple of (host, port) or a path to a UNIX socket - :param tls: if ``True``, wrap the plaintext connectable in a - :class:`~.streams.tls.TLSConnectable`, using the provided TLS settings) - :param ssl_context: if ``tls=True``, the SSLContext object to use (if not provided, - a secure default will be created) - :param tls_hostname: if ``tls=True``, host name of the server to use for checking - the server certificate (defaults to the host portion of the address for TCP - connectables) - :param tls_standard_compatible: if ``False`` and ``tls=True``, makes the TLS stream - skip the closing handshake when closing the connection, so it won't raise an - exception if the server does the same - - """ - connectable: TCPConnectable | UNIXConnectable | TLSConnectable - if isinstance(remote, ByteStreamConnectable): - return remote - elif isinstance(remote, tuple) and len(remote) == 2: - connectable = TCPConnectable(*remote) - elif isinstance(remote, (str, bytes, PathLike)): - connectable = UNIXConnectable(remote) - else: - raise TypeError(f"cannot convert {remote!r} to a connectable") - - if tls: - if not tls_hostname and isinstance(connectable, TCPConnectable): - tls_hostname = str(connectable.host) - - connectable = TLSConnectable( - connectable, - ssl_context=ssl_context, - hostname=tls_hostname, - standard_compatible=tls_standard_compatible, - ) - - return connectable diff --git a/.env/Lib/site-packages/anyio/_core/_streams.py b/.env/Lib/site-packages/anyio/_core/_streams.py deleted file mode 100644 index 2b9c7df..0000000 --- a/.env/Lib/site-packages/anyio/_core/_streams.py +++ /dev/null @@ -1,52 +0,0 @@ -from __future__ import annotations - -import math -from typing import TypeVar -from warnings import warn - -from ..streams.memory import ( - MemoryObjectReceiveStream, - MemoryObjectSendStream, - _MemoryObjectStreamState, -) - -T_Item = TypeVar("T_Item") - - -class create_memory_object_stream( - tuple[MemoryObjectSendStream[T_Item], MemoryObjectReceiveStream[T_Item]], -): - """ - Create a memory object stream. - - The stream's item type can be annotated like - :func:`create_memory_object_stream[T_Item]`. - - :param max_buffer_size: number of items held in the buffer until ``send()`` starts - blocking - :param item_type: old way of marking the streams with the right generic type for - static typing (does nothing on AnyIO 4) - - .. deprecated:: 4.0 - Use ``create_memory_object_stream[YourItemType](...)`` instead. - :return: a tuple of (send stream, receive stream) - - """ - - def __new__( # type: ignore[misc] - cls, max_buffer_size: float = 0, item_type: object = None - ) -> tuple[MemoryObjectSendStream[T_Item], MemoryObjectReceiveStream[T_Item]]: - if max_buffer_size != math.inf and not isinstance(max_buffer_size, int): - raise ValueError("max_buffer_size must be either an integer or math.inf") - if max_buffer_size < 0: - raise ValueError("max_buffer_size cannot be negative") - if item_type is not None: - warn( - "The item_type argument has been deprecated in AnyIO 4.0. " - "Use create_memory_object_stream[YourItemType](...) instead.", - DeprecationWarning, - stacklevel=2, - ) - - state = _MemoryObjectStreamState[T_Item](max_buffer_size) - return (MemoryObjectSendStream(state), MemoryObjectReceiveStream(state)) diff --git a/.env/Lib/site-packages/anyio/_core/_subprocesses.py b/.env/Lib/site-packages/anyio/_core/_subprocesses.py deleted file mode 100644 index 36d9b30..0000000 --- a/.env/Lib/site-packages/anyio/_core/_subprocesses.py +++ /dev/null @@ -1,202 +0,0 @@ -from __future__ import annotations - -import sys -from collections.abc import AsyncIterable, Iterable, Mapping, Sequence -from io import BytesIO -from os import PathLike -from subprocess import PIPE, CalledProcessError, CompletedProcess -from typing import IO, Any, Union, cast - -from ..abc import Process -from ._eventloop import get_async_backend -from ._tasks import create_task_group - -if sys.version_info >= (3, 10): - from typing import TypeAlias -else: - from typing_extensions import TypeAlias - -StrOrBytesPath: TypeAlias = Union[str, bytes, "PathLike[str]", "PathLike[bytes]"] - - -async def run_process( - command: StrOrBytesPath | Sequence[StrOrBytesPath], - *, - input: bytes | None = None, - stdin: int | IO[Any] | None = None, - stdout: int | IO[Any] | None = PIPE, - stderr: int | IO[Any] | None = PIPE, - check: bool = True, - cwd: StrOrBytesPath | None = None, - env: Mapping[str, str] | None = None, - startupinfo: Any = None, - creationflags: int = 0, - start_new_session: bool = False, - pass_fds: Sequence[int] = (), - user: str | int | None = None, - group: str | int | None = None, - extra_groups: Iterable[str | int] | None = None, - umask: int = -1, -) -> CompletedProcess[bytes]: - """ - Run an external command in a subprocess and wait until it completes. - - .. seealso:: :func:`subprocess.run` - - :param command: either a string to pass to the shell, or an iterable of strings - containing the executable name or path and its arguments - :param input: bytes passed to the standard input of the subprocess - :param stdin: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, - a file-like object, or `None`; ``input`` overrides this - :param stdout: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, - a file-like object, or `None` - :param stderr: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, - :data:`subprocess.STDOUT`, a file-like object, or `None` - :param check: if ``True``, raise :exc:`~subprocess.CalledProcessError` if the - process terminates with a return code other than 0 - :param cwd: If not ``None``, change the working directory to this before running the - command - :param env: if not ``None``, this mapping replaces the inherited environment - variables from the parent process - :param startupinfo: an instance of :class:`subprocess.STARTUPINFO` that can be used - to specify process startup parameters (Windows only) - :param creationflags: flags that can be used to control the creation of the - subprocess (see :class:`subprocess.Popen` for the specifics) - :param start_new_session: if ``true`` the setsid() system call will be made in the - child process prior to the execution of the subprocess. (POSIX only) - :param pass_fds: sequence of file descriptors to keep open between the parent and - child processes. (POSIX only) - :param user: effective user to run the process as (Python >= 3.9, POSIX only) - :param group: effective group to run the process as (Python >= 3.9, POSIX only) - :param extra_groups: supplementary groups to set in the subprocess (Python >= 3.9, - POSIX only) - :param umask: if not negative, this umask is applied in the child process before - running the given command (Python >= 3.9, POSIX only) - :return: an object representing the completed process - :raises ~subprocess.CalledProcessError: if ``check`` is ``True`` and the process - exits with a nonzero return code - - """ - - async def drain_stream(stream: AsyncIterable[bytes], index: int) -> None: - buffer = BytesIO() - async for chunk in stream: - buffer.write(chunk) - - stream_contents[index] = buffer.getvalue() - - if stdin is not None and input is not None: - raise ValueError("only one of stdin and input is allowed") - - async with await open_process( - command, - stdin=PIPE if input else stdin, - stdout=stdout, - stderr=stderr, - cwd=cwd, - env=env, - startupinfo=startupinfo, - creationflags=creationflags, - start_new_session=start_new_session, - pass_fds=pass_fds, - user=user, - group=group, - extra_groups=extra_groups, - umask=umask, - ) as process: - stream_contents: list[bytes | None] = [None, None] - async with create_task_group() as tg: - if process.stdout: - tg.start_soon(drain_stream, process.stdout, 0) - - if process.stderr: - tg.start_soon(drain_stream, process.stderr, 1) - - if process.stdin and input: - await process.stdin.send(input) - await process.stdin.aclose() - - await process.wait() - - output, errors = stream_contents - if check and process.returncode != 0: - raise CalledProcessError(cast(int, process.returncode), command, output, errors) - - return CompletedProcess(command, cast(int, process.returncode), output, errors) - - -async def open_process( - command: StrOrBytesPath | Sequence[StrOrBytesPath], - *, - stdin: int | IO[Any] | None = PIPE, - stdout: int | IO[Any] | None = PIPE, - stderr: int | IO[Any] | None = PIPE, - cwd: StrOrBytesPath | None = None, - env: Mapping[str, str] | None = None, - startupinfo: Any = None, - creationflags: int = 0, - start_new_session: bool = False, - pass_fds: Sequence[int] = (), - user: str | int | None = None, - group: str | int | None = None, - extra_groups: Iterable[str | int] | None = None, - umask: int = -1, -) -> Process: - """ - Start an external command in a subprocess. - - .. seealso:: :class:`subprocess.Popen` - - :param command: either a string to pass to the shell, or an iterable of strings - containing the executable name or path and its arguments - :param stdin: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, a - file-like object, or ``None`` - :param stdout: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, - a file-like object, or ``None`` - :param stderr: one of :data:`subprocess.PIPE`, :data:`subprocess.DEVNULL`, - :data:`subprocess.STDOUT`, a file-like object, or ``None`` - :param cwd: If not ``None``, the working directory is changed before executing - :param env: If env is not ``None``, it must be a mapping that defines the - environment variables for the new process - :param creationflags: flags that can be used to control the creation of the - subprocess (see :class:`subprocess.Popen` for the specifics) - :param startupinfo: an instance of :class:`subprocess.STARTUPINFO` that can be used - to specify process startup parameters (Windows only) - :param start_new_session: if ``true`` the setsid() system call will be made in the - child process prior to the execution of the subprocess. (POSIX only) - :param pass_fds: sequence of file descriptors to keep open between the parent and - child processes. (POSIX only) - :param user: effective user to run the process as (POSIX only) - :param group: effective group to run the process as (POSIX only) - :param extra_groups: supplementary groups to set in the subprocess (POSIX only) - :param umask: if not negative, this umask is applied in the child process before - running the given command (POSIX only) - :return: an asynchronous process object - - """ - kwargs: dict[str, Any] = {} - if user is not None: - kwargs["user"] = user - - if group is not None: - kwargs["group"] = group - - if extra_groups is not None: - kwargs["extra_groups"] = group - - if umask >= 0: - kwargs["umask"] = umask - - return await get_async_backend().open_process( - command, - stdin=stdin, - stdout=stdout, - stderr=stderr, - cwd=cwd, - env=env, - startupinfo=startupinfo, - creationflags=creationflags, - start_new_session=start_new_session, - pass_fds=pass_fds, - **kwargs, - ) diff --git a/.env/Lib/site-packages/anyio/_core/_synchronization.py b/.env/Lib/site-packages/anyio/_core/_synchronization.py deleted file mode 100644 index c0ef27a..0000000 --- a/.env/Lib/site-packages/anyio/_core/_synchronization.py +++ /dev/null @@ -1,753 +0,0 @@ -from __future__ import annotations - -import math -from collections import deque -from collections.abc import Callable -from dataclasses import dataclass -from types import TracebackType -from typing import TypeVar - -from ..lowlevel import checkpoint_if_cancelled -from ._eventloop import get_async_backend -from ._exceptions import BusyResourceError, NoEventLoopError -from ._tasks import CancelScope -from ._testing import TaskInfo, get_current_task - -T = TypeVar("T") - - -@dataclass(frozen=True) -class EventStatistics: - """ - :ivar int tasks_waiting: number of tasks waiting on :meth:`~.Event.wait` - """ - - tasks_waiting: int - - -@dataclass(frozen=True) -class CapacityLimiterStatistics: - """ - :ivar int borrowed_tokens: number of tokens currently borrowed by tasks - :ivar float total_tokens: total number of available tokens - :ivar tuple borrowers: tasks or other objects currently holding tokens borrowed from - this limiter - :ivar int tasks_waiting: number of tasks waiting on - :meth:`~.CapacityLimiter.acquire` or - :meth:`~.CapacityLimiter.acquire_on_behalf_of` - """ - - borrowed_tokens: int - total_tokens: float - borrowers: tuple[object, ...] - tasks_waiting: int - - -@dataclass(frozen=True) -class LockStatistics: - """ - :ivar bool locked: flag indicating if this lock is locked or not - :ivar ~anyio.TaskInfo owner: task currently holding the lock (or ``None`` if the - lock is not held by any task) - :ivar int tasks_waiting: number of tasks waiting on :meth:`~.Lock.acquire` - """ - - locked: bool - owner: TaskInfo | None - tasks_waiting: int - - -@dataclass(frozen=True) -class ConditionStatistics: - """ - :ivar int tasks_waiting: number of tasks blocked on :meth:`~.Condition.wait` - :ivar ~anyio.LockStatistics lock_statistics: statistics of the underlying - :class:`~.Lock` - """ - - tasks_waiting: int - lock_statistics: LockStatistics - - -@dataclass(frozen=True) -class SemaphoreStatistics: - """ - :ivar int tasks_waiting: number of tasks waiting on :meth:`~.Semaphore.acquire` - - """ - - tasks_waiting: int - - -class Event: - def __new__(cls) -> Event: - try: - return get_async_backend().create_event() - except NoEventLoopError: - return EventAdapter() - - def set(self) -> None: - """Set the flag, notifying all listeners.""" - raise NotImplementedError - - def is_set(self) -> bool: - """Return ``True`` if the flag is set, ``False`` if not.""" - raise NotImplementedError - - async def wait(self) -> None: - """ - Wait until the flag has been set. - - If the flag has already been set when this method is called, it returns - immediately. - - """ - raise NotImplementedError - - def statistics(self) -> EventStatistics: - """Return statistics about the current state of this event.""" - raise NotImplementedError - - -class EventAdapter(Event): - _internal_event: Event | None = None - _is_set: bool = False - - def __new__(cls) -> EventAdapter: - return object.__new__(cls) - - @property - def _event(self) -> Event: - if self._internal_event is None: - self._internal_event = get_async_backend().create_event() - if self._is_set: - self._internal_event.set() - - return self._internal_event - - def set(self) -> None: - if self._internal_event is None: - self._is_set = True - else: - self._event.set() - - def is_set(self) -> bool: - if self._internal_event is None: - return self._is_set - - return self._internal_event.is_set() - - async def wait(self) -> None: - await self._event.wait() - - def statistics(self) -> EventStatistics: - if self._internal_event is None: - return EventStatistics(tasks_waiting=0) - - return self._internal_event.statistics() - - -class Lock: - def __new__(cls, *, fast_acquire: bool = False) -> Lock: - try: - return get_async_backend().create_lock(fast_acquire=fast_acquire) - except NoEventLoopError: - return LockAdapter(fast_acquire=fast_acquire) - - async def __aenter__(self) -> None: - await self.acquire() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.release() - - async def acquire(self) -> None: - """Acquire the lock.""" - raise NotImplementedError - - def acquire_nowait(self) -> None: - """ - Acquire the lock, without blocking. - - :raises ~anyio.WouldBlock: if the operation would block - - """ - raise NotImplementedError - - def release(self) -> None: - """Release the lock.""" - raise NotImplementedError - - def locked(self) -> bool: - """Return True if the lock is currently held.""" - raise NotImplementedError - - def statistics(self) -> LockStatistics: - """ - Return statistics about the current state of this lock. - - .. versionadded:: 3.0 - """ - raise NotImplementedError - - -class LockAdapter(Lock): - _internal_lock: Lock | None = None - - def __new__(cls, *, fast_acquire: bool = False) -> LockAdapter: - return object.__new__(cls) - - def __init__(self, *, fast_acquire: bool = False): - self._fast_acquire = fast_acquire - - @property - def _lock(self) -> Lock: - if self._internal_lock is None: - self._internal_lock = get_async_backend().create_lock( - fast_acquire=self._fast_acquire - ) - - return self._internal_lock - - async def __aenter__(self) -> None: - await self._lock.acquire() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - if self._internal_lock is not None: - self._internal_lock.release() - - async def acquire(self) -> None: - """Acquire the lock.""" - await self._lock.acquire() - - def acquire_nowait(self) -> None: - """ - Acquire the lock, without blocking. - - :raises ~anyio.WouldBlock: if the operation would block - - """ - self._lock.acquire_nowait() - - def release(self) -> None: - """Release the lock.""" - self._lock.release() - - def locked(self) -> bool: - """Return True if the lock is currently held.""" - return self._lock.locked() - - def statistics(self) -> LockStatistics: - """ - Return statistics about the current state of this lock. - - .. versionadded:: 3.0 - - """ - if self._internal_lock is None: - return LockStatistics(False, None, 0) - - return self._internal_lock.statistics() - - -class Condition: - _owner_task: TaskInfo | None = None - - def __init__(self, lock: Lock | None = None): - self._lock = lock or Lock() - self._waiters: deque[Event] = deque() - - async def __aenter__(self) -> None: - await self.acquire() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.release() - - def _check_acquired(self) -> None: - if self._owner_task != get_current_task(): - raise RuntimeError("The current task is not holding the underlying lock") - - async def acquire(self) -> None: - """Acquire the underlying lock.""" - await self._lock.acquire() - self._owner_task = get_current_task() - - def acquire_nowait(self) -> None: - """ - Acquire the underlying lock, without blocking. - - :raises ~anyio.WouldBlock: if the operation would block - - """ - self._lock.acquire_nowait() - self._owner_task = get_current_task() - - def release(self) -> None: - """Release the underlying lock.""" - self._lock.release() - - def locked(self) -> bool: - """Return True if the lock is set.""" - return self._lock.locked() - - def notify(self, n: int = 1) -> None: - """Notify exactly n listeners.""" - self._check_acquired() - for _ in range(n): - try: - event = self._waiters.popleft() - except IndexError: - break - - event.set() - - def notify_all(self) -> None: - """Notify all the listeners.""" - self._check_acquired() - for event in self._waiters: - event.set() - - self._waiters.clear() - - async def wait(self) -> None: - """Wait for a notification.""" - await checkpoint_if_cancelled() - self._check_acquired() - event = Event() - self._waiters.append(event) - self.release() - try: - await event.wait() - except BaseException: - if not event.is_set(): - self._waiters.remove(event) - - raise - finally: - with CancelScope(shield=True): - await self.acquire() - - async def wait_for(self, predicate: Callable[[], T]) -> T: - """ - Wait until a predicate becomes true. - - :param predicate: a callable that returns a truthy value when the condition is - met - :return: the result of the predicate - - .. versionadded:: 4.11.0 - - """ - while not (result := predicate()): - await self.wait() - - return result - - def statistics(self) -> ConditionStatistics: - """ - Return statistics about the current state of this condition. - - .. versionadded:: 3.0 - """ - return ConditionStatistics(len(self._waiters), self._lock.statistics()) - - -class Semaphore: - def __new__( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> Semaphore: - try: - return get_async_backend().create_semaphore( - initial_value, max_value=max_value, fast_acquire=fast_acquire - ) - except NoEventLoopError: - return SemaphoreAdapter(initial_value, max_value=max_value) - - def __init__( - self, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ): - if not isinstance(initial_value, int): - raise TypeError("initial_value must be an integer") - if initial_value < 0: - raise ValueError("initial_value must be >= 0") - if max_value is not None: - if not isinstance(max_value, int): - raise TypeError("max_value must be an integer or None") - if max_value < initial_value: - raise ValueError( - "max_value must be equal to or higher than initial_value" - ) - - self._fast_acquire = fast_acquire - - async def __aenter__(self) -> Semaphore: - await self.acquire() - return self - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.release() - - async def acquire(self) -> None: - """Decrement the semaphore value, blocking if necessary.""" - raise NotImplementedError - - def acquire_nowait(self) -> None: - """ - Acquire the underlying lock, without blocking. - - :raises ~anyio.WouldBlock: if the operation would block - - """ - raise NotImplementedError - - def release(self) -> None: - """Increment the semaphore value.""" - raise NotImplementedError - - @property - def value(self) -> int: - """The current value of the semaphore.""" - raise NotImplementedError - - @property - def max_value(self) -> int | None: - """The maximum value of the semaphore.""" - raise NotImplementedError - - def statistics(self) -> SemaphoreStatistics: - """ - Return statistics about the current state of this semaphore. - - .. versionadded:: 3.0 - """ - raise NotImplementedError - - -class SemaphoreAdapter(Semaphore): - _internal_semaphore: Semaphore | None = None - - def __new__( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> SemaphoreAdapter: - return object.__new__(cls) - - def __init__( - self, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> None: - super().__init__(initial_value, max_value=max_value, fast_acquire=fast_acquire) - self._initial_value = initial_value - self._max_value = max_value - - @property - def _semaphore(self) -> Semaphore: - if self._internal_semaphore is None: - self._internal_semaphore = get_async_backend().create_semaphore( - self._initial_value, max_value=self._max_value - ) - - return self._internal_semaphore - - async def acquire(self) -> None: - await self._semaphore.acquire() - - def acquire_nowait(self) -> None: - self._semaphore.acquire_nowait() - - def release(self) -> None: - self._semaphore.release() - - @property - def value(self) -> int: - if self._internal_semaphore is None: - return self._initial_value - - return self._semaphore.value - - @property - def max_value(self) -> int | None: - return self._max_value - - def statistics(self) -> SemaphoreStatistics: - if self._internal_semaphore is None: - return SemaphoreStatistics(tasks_waiting=0) - - return self._semaphore.statistics() - - -class CapacityLimiter: - def __new__(cls, total_tokens: float) -> CapacityLimiter: - try: - return get_async_backend().create_capacity_limiter(total_tokens) - except NoEventLoopError: - return CapacityLimiterAdapter(total_tokens) - - async def __aenter__(self) -> None: - raise NotImplementedError - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - raise NotImplementedError - - @property - def total_tokens(self) -> float: - """ - The total number of tokens available for borrowing. - - This is a read-write property. If the total number of tokens is increased, the - proportionate number of tasks waiting on this limiter will be granted their - tokens. - - .. versionchanged:: 3.0 - The property is now writable. - .. versionchanged:: 4.12 - The value can now be set to 0. - - """ - raise NotImplementedError - - @total_tokens.setter - def total_tokens(self, value: float) -> None: - raise NotImplementedError - - @property - def borrowed_tokens(self) -> int: - """The number of tokens that have currently been borrowed.""" - raise NotImplementedError - - @property - def available_tokens(self) -> float: - """The number of tokens currently available to be borrowed""" - raise NotImplementedError - - def acquire_nowait(self) -> None: - """ - Acquire a token for the current task without waiting for one to become - available. - - :raises ~anyio.WouldBlock: if there are no tokens available for borrowing - - """ - raise NotImplementedError - - def acquire_on_behalf_of_nowait(self, borrower: object) -> None: - """ - Acquire a token without waiting for one to become available. - - :param borrower: the entity borrowing a token - :raises ~anyio.WouldBlock: if there are no tokens available for borrowing - - """ - raise NotImplementedError - - async def acquire(self) -> None: - """ - Acquire a token for the current task, waiting if necessary for one to become - available. - - """ - raise NotImplementedError - - async def acquire_on_behalf_of(self, borrower: object) -> None: - """ - Acquire a token, waiting if necessary for one to become available. - - :param borrower: the entity borrowing a token - - """ - raise NotImplementedError - - def release(self) -> None: - """ - Release the token held by the current task. - - :raises RuntimeError: if the current task has not borrowed a token from this - limiter. - - """ - raise NotImplementedError - - def release_on_behalf_of(self, borrower: object) -> None: - """ - Release the token held by the given borrower. - - :raises RuntimeError: if the borrower has not borrowed a token from this - limiter. - - """ - raise NotImplementedError - - def statistics(self) -> CapacityLimiterStatistics: - """ - Return statistics about the current state of this limiter. - - .. versionadded:: 3.0 - - """ - raise NotImplementedError - - -class CapacityLimiterAdapter(CapacityLimiter): - _internal_limiter: CapacityLimiter | None = None - - def __new__(cls, total_tokens: float) -> CapacityLimiterAdapter: - return object.__new__(cls) - - def __init__(self, total_tokens: float) -> None: - self.total_tokens = total_tokens - - @property - def _limiter(self) -> CapacityLimiter: - if self._internal_limiter is None: - self._internal_limiter = get_async_backend().create_capacity_limiter( - self._total_tokens - ) - - return self._internal_limiter - - async def __aenter__(self) -> None: - await self._limiter.__aenter__() - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - return await self._limiter.__aexit__(exc_type, exc_val, exc_tb) - - @property - def total_tokens(self) -> float: - if self._internal_limiter is None: - return self._total_tokens - - return self._internal_limiter.total_tokens - - @total_tokens.setter - def total_tokens(self, value: float) -> None: - if not isinstance(value, int) and value is not math.inf: - raise TypeError("total_tokens must be an int or math.inf") - elif value < 1: - raise ValueError("total_tokens must be >= 1") - - if self._internal_limiter is None: - self._total_tokens = value - return - - self._limiter.total_tokens = value - - @property - def borrowed_tokens(self) -> int: - if self._internal_limiter is None: - return 0 - - return self._internal_limiter.borrowed_tokens - - @property - def available_tokens(self) -> float: - if self._internal_limiter is None: - return self._total_tokens - - return self._internal_limiter.available_tokens - - def acquire_nowait(self) -> None: - self._limiter.acquire_nowait() - - def acquire_on_behalf_of_nowait(self, borrower: object) -> None: - self._limiter.acquire_on_behalf_of_nowait(borrower) - - async def acquire(self) -> None: - await self._limiter.acquire() - - async def acquire_on_behalf_of(self, borrower: object) -> None: - await self._limiter.acquire_on_behalf_of(borrower) - - def release(self) -> None: - self._limiter.release() - - def release_on_behalf_of(self, borrower: object) -> None: - self._limiter.release_on_behalf_of(borrower) - - def statistics(self) -> CapacityLimiterStatistics: - if self._internal_limiter is None: - return CapacityLimiterStatistics( - borrowed_tokens=0, - total_tokens=self.total_tokens, - borrowers=(), - tasks_waiting=0, - ) - - return self._internal_limiter.statistics() - - -class ResourceGuard: - """ - A context manager for ensuring that a resource is only used by a single task at a - time. - - Entering this context manager while the previous has not exited it yet will trigger - :exc:`BusyResourceError`. - - :param action: the action to guard against (visible in the :exc:`BusyResourceError` - when triggered, e.g. "Another task is already {action} this resource") - - .. versionadded:: 4.1 - """ - - __slots__ = "action", "_guarded" - - def __init__(self, action: str = "using"): - self.action: str = action - self._guarded = False - - def __enter__(self) -> None: - if self._guarded: - raise BusyResourceError(self.action) - - self._guarded = True - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self._guarded = False diff --git a/.env/Lib/site-packages/anyio/_core/_tasks.py b/.env/Lib/site-packages/anyio/_core/_tasks.py deleted file mode 100644 index 0688bfe..0000000 --- a/.env/Lib/site-packages/anyio/_core/_tasks.py +++ /dev/null @@ -1,173 +0,0 @@ -from __future__ import annotations - -import math -from collections.abc import Generator -from contextlib import contextmanager -from types import TracebackType - -from ..abc._tasks import TaskGroup, TaskStatus -from ._eventloop import get_async_backend - - -class _IgnoredTaskStatus(TaskStatus[object]): - def started(self, value: object = None) -> None: - pass - - -TASK_STATUS_IGNORED = _IgnoredTaskStatus() - - -class CancelScope: - """ - Wraps a unit of work that can be made separately cancellable. - - :param deadline: The time (clock value) when this scope is cancelled automatically - :param shield: ``True`` to shield the cancel scope from external cancellation - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - """ - - def __new__( - cls, *, deadline: float = math.inf, shield: bool = False - ) -> CancelScope: - return get_async_backend().create_cancel_scope(shield=shield, deadline=deadline) - - def cancel(self, reason: str | None = None) -> None: - """ - Cancel this scope immediately. - - :param reason: a message describing the reason for the cancellation - - """ - raise NotImplementedError - - @property - def deadline(self) -> float: - """ - The time (clock value) when this scope is cancelled automatically. - - Will be ``float('inf')`` if no timeout has been set. - - """ - raise NotImplementedError - - @deadline.setter - def deadline(self, value: float) -> None: - raise NotImplementedError - - @property - def cancel_called(self) -> bool: - """``True`` if :meth:`cancel` has been called.""" - raise NotImplementedError - - @property - def cancelled_caught(self) -> bool: - """ - ``True`` if this scope suppressed a cancellation exception it itself raised. - - This is typically used to check if any work was interrupted, or to see if the - scope was cancelled due to its deadline being reached. The value will, however, - only be ``True`` if the cancellation was triggered by the scope itself (and not - an outer scope). - - """ - raise NotImplementedError - - @property - def shield(self) -> bool: - """ - ``True`` if this scope is shielded from external cancellation. - - While a scope is shielded, it will not receive cancellations from outside. - - """ - raise NotImplementedError - - @shield.setter - def shield(self, value: bool) -> None: - raise NotImplementedError - - def __enter__(self) -> CancelScope: - raise NotImplementedError - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - raise NotImplementedError - - -@contextmanager -def fail_after( - delay: float | None, shield: bool = False -) -> Generator[CancelScope, None, None]: - """ - Create a context manager which raises a :class:`TimeoutError` if does not finish in - time. - - :param delay: maximum allowed time (in seconds) before raising the exception, or - ``None`` to disable the timeout - :param shield: ``True`` to shield the cancel scope from external cancellation - :return: a context manager that yields a cancel scope - :rtype: :class:`~typing.ContextManager`\\[:class:`~anyio.CancelScope`\\] - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - current_time = get_async_backend().current_time - deadline = (current_time() + delay) if delay is not None else math.inf - with get_async_backend().create_cancel_scope( - deadline=deadline, shield=shield - ) as cancel_scope: - yield cancel_scope - - if cancel_scope.cancelled_caught and current_time() >= cancel_scope.deadline: - raise TimeoutError - - -def move_on_after(delay: float | None, shield: bool = False) -> CancelScope: - """ - Create a cancel scope with a deadline that expires after the given delay. - - :param delay: maximum allowed time (in seconds) before exiting the context block, or - ``None`` to disable the timeout - :param shield: ``True`` to shield the cancel scope from external cancellation - :return: a cancel scope - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - deadline = ( - (get_async_backend().current_time() + delay) if delay is not None else math.inf - ) - return get_async_backend().create_cancel_scope(deadline=deadline, shield=shield) - - -def current_effective_deadline() -> float: - """ - Return the nearest deadline among all the cancel scopes effective for the current - task. - - :return: a clock value from the event loop's internal clock (or ``float('inf')`` if - there is no deadline in effect, or ``float('-inf')`` if the current scope has - been cancelled) - :rtype: float - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().current_effective_deadline() - - -def create_task_group() -> TaskGroup: - """ - Create a task group. - - :return: a task group - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().create_task_group() diff --git a/.env/Lib/site-packages/anyio/_core/_tempfile.py b/.env/Lib/site-packages/anyio/_core/_tempfile.py deleted file mode 100644 index fbb6b14..0000000 --- a/.env/Lib/site-packages/anyio/_core/_tempfile.py +++ /dev/null @@ -1,616 +0,0 @@ -from __future__ import annotations - -import os -import sys -import tempfile -from collections.abc import Iterable -from io import BytesIO, TextIOWrapper -from types import TracebackType -from typing import ( - TYPE_CHECKING, - Any, - AnyStr, - Generic, - overload, -) - -from .. import to_thread -from .._core._fileio import AsyncFile -from ..lowlevel import checkpoint_if_cancelled - -if TYPE_CHECKING: - from _typeshed import OpenBinaryMode, OpenTextMode, ReadableBuffer, WriteableBuffer - - -class TemporaryFile(Generic[AnyStr]): - """ - An asynchronous temporary file that is automatically created and cleaned up. - - This class provides an asynchronous context manager interface to a temporary file. - The file is created using Python's standard `tempfile.TemporaryFile` function in a - background thread, and is wrapped as an asynchronous file using `AsyncFile`. - - :param mode: The mode in which the file is opened. Defaults to "w+b". - :param buffering: The buffering policy (-1 means the default buffering). - :param encoding: The encoding used to decode or encode the file. Only applicable in - text mode. - :param newline: Controls how universal newlines mode works (only applicable in text - mode). - :param suffix: The suffix for the temporary file name. - :param prefix: The prefix for the temporary file name. - :param dir: The directory in which the temporary file is created. - :param errors: The error handling scheme used for encoding/decoding errors. - """ - - _async_file: AsyncFile[AnyStr] - - @overload - def __init__( - self: TemporaryFile[bytes], - mode: OpenBinaryMode = ..., - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - *, - errors: str | None = ..., - ): ... - @overload - def __init__( - self: TemporaryFile[str], - mode: OpenTextMode, - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - *, - errors: str | None = ..., - ): ... - - def __init__( - self, - mode: OpenTextMode | OpenBinaryMode = "w+b", - buffering: int = -1, - encoding: str | None = None, - newline: str | None = None, - suffix: str | None = None, - prefix: str | None = None, - dir: str | None = None, - *, - errors: str | None = None, - ) -> None: - self.mode = mode - self.buffering = buffering - self.encoding = encoding - self.newline = newline - self.suffix: str | None = suffix - self.prefix: str | None = prefix - self.dir: str | None = dir - self.errors = errors - - async def __aenter__(self) -> AsyncFile[AnyStr]: - fp = await to_thread.run_sync( - lambda: tempfile.TemporaryFile( - self.mode, - self.buffering, - self.encoding, - self.newline, - self.suffix, - self.prefix, - self.dir, - errors=self.errors, - ) - ) - self._async_file = AsyncFile(fp) - return self._async_file - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - traceback: TracebackType | None, - ) -> None: - await self._async_file.aclose() - - -class NamedTemporaryFile(Generic[AnyStr]): - """ - An asynchronous named temporary file that is automatically created and cleaned up. - - This class provides an asynchronous context manager for a temporary file with a - visible name in the file system. It uses Python's standard - :func:`~tempfile.NamedTemporaryFile` function and wraps the file object with - :class:`AsyncFile` for asynchronous operations. - - :param mode: The mode in which the file is opened. Defaults to "w+b". - :param buffering: The buffering policy (-1 means the default buffering). - :param encoding: The encoding used to decode or encode the file. Only applicable in - text mode. - :param newline: Controls how universal newlines mode works (only applicable in text - mode). - :param suffix: The suffix for the temporary file name. - :param prefix: The prefix for the temporary file name. - :param dir: The directory in which the temporary file is created. - :param delete: Whether to delete the file when it is closed. - :param errors: The error handling scheme used for encoding/decoding errors. - :param delete_on_close: (Python 3.12+) Whether to delete the file on close. - """ - - _async_file: AsyncFile[AnyStr] - - @overload - def __init__( - self: NamedTemporaryFile[bytes], - mode: OpenBinaryMode = ..., - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - delete: bool = ..., - *, - errors: str | None = ..., - delete_on_close: bool = ..., - ): ... - @overload - def __init__( - self: NamedTemporaryFile[str], - mode: OpenTextMode, - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - delete: bool = ..., - *, - errors: str | None = ..., - delete_on_close: bool = ..., - ): ... - - def __init__( - self, - mode: OpenBinaryMode | OpenTextMode = "w+b", - buffering: int = -1, - encoding: str | None = None, - newline: str | None = None, - suffix: str | None = None, - prefix: str | None = None, - dir: str | None = None, - delete: bool = True, - *, - errors: str | None = None, - delete_on_close: bool = True, - ) -> None: - self._params: dict[str, Any] = { - "mode": mode, - "buffering": buffering, - "encoding": encoding, - "newline": newline, - "suffix": suffix, - "prefix": prefix, - "dir": dir, - "delete": delete, - "errors": errors, - } - if sys.version_info >= (3, 12): - self._params["delete_on_close"] = delete_on_close - - async def __aenter__(self) -> AsyncFile[AnyStr]: - fp = await to_thread.run_sync( - lambda: tempfile.NamedTemporaryFile(**self._params) - ) - self._async_file = AsyncFile(fp) - return self._async_file - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - traceback: TracebackType | None, - ) -> None: - await self._async_file.aclose() - - -class SpooledTemporaryFile(AsyncFile[AnyStr]): - """ - An asynchronous spooled temporary file that starts in memory and is spooled to disk. - - This class provides an asynchronous interface to a spooled temporary file, much like - Python's standard :class:`~tempfile.SpooledTemporaryFile`. It supports asynchronous - write operations and provides a method to force a rollover to disk. - - :param max_size: Maximum size in bytes before the file is rolled over to disk. - :param mode: The mode in which the file is opened. Defaults to "w+b". - :param buffering: The buffering policy (-1 means the default buffering). - :param encoding: The encoding used to decode or encode the file (text mode only). - :param newline: Controls how universal newlines mode works (text mode only). - :param suffix: The suffix for the temporary file name. - :param prefix: The prefix for the temporary file name. - :param dir: The directory in which the temporary file is created. - :param errors: The error handling scheme used for encoding/decoding errors. - """ - - _rolled: bool = False - - @overload - def __init__( - self: SpooledTemporaryFile[bytes], - max_size: int = ..., - mode: OpenBinaryMode = ..., - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - *, - errors: str | None = ..., - ): ... - @overload - def __init__( - self: SpooledTemporaryFile[str], - max_size: int = ..., - mode: OpenTextMode = ..., - buffering: int = ..., - encoding: str | None = ..., - newline: str | None = ..., - suffix: str | None = ..., - prefix: str | None = ..., - dir: str | None = ..., - *, - errors: str | None = ..., - ): ... - - def __init__( - self, - max_size: int = 0, - mode: OpenBinaryMode | OpenTextMode = "w+b", - buffering: int = -1, - encoding: str | None = None, - newline: str | None = None, - suffix: str | None = None, - prefix: str | None = None, - dir: str | None = None, - *, - errors: str | None = None, - ) -> None: - self._tempfile_params: dict[str, Any] = { - "mode": mode, - "buffering": buffering, - "encoding": encoding, - "newline": newline, - "suffix": suffix, - "prefix": prefix, - "dir": dir, - "errors": errors, - } - self._max_size = max_size - if "b" in mode: - super().__init__(BytesIO()) # type: ignore[arg-type] - else: - super().__init__( - TextIOWrapper( # type: ignore[arg-type] - BytesIO(), - encoding=encoding, - errors=errors, - newline=newline, - write_through=True, - ) - ) - - async def aclose(self) -> None: - if not self._rolled: - self._fp.close() - return - - await super().aclose() - - async def _check(self) -> None: - if self._rolled or self._fp.tell() <= self._max_size: - return - - await self.rollover() - - async def rollover(self) -> None: - if self._rolled: - return - - self._rolled = True - buffer = self._fp - buffer.seek(0) - self._fp = await to_thread.run_sync( - lambda: tempfile.TemporaryFile(**self._tempfile_params) - ) - await self.write(buffer.read()) - buffer.close() - - @property - def closed(self) -> bool: - return self._fp.closed - - async def read(self, size: int = -1) -> AnyStr: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.read(size) - - return await super().read(size) # type: ignore[return-value] - - async def read1(self: SpooledTemporaryFile[bytes], size: int = -1) -> bytes: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.read1(size) - - return await super().read1(size) - - async def readline(self) -> AnyStr: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.readline() - - return await super().readline() # type: ignore[return-value] - - async def readlines(self) -> list[AnyStr]: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.readlines() - - return await super().readlines() # type: ignore[return-value] - - async def readinto(self: SpooledTemporaryFile[bytes], b: WriteableBuffer) -> int: - if not self._rolled: - await checkpoint_if_cancelled() - self._fp.readinto(b) - - return await super().readinto(b) - - async def readinto1(self: SpooledTemporaryFile[bytes], b: WriteableBuffer) -> int: - if not self._rolled: - await checkpoint_if_cancelled() - self._fp.readinto(b) - - return await super().readinto1(b) - - async def seek(self, offset: int, whence: int | None = os.SEEK_SET) -> int: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.seek(offset, whence) - - return await super().seek(offset, whence) - - async def tell(self) -> int: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.tell() - - return await super().tell() - - async def truncate(self, size: int | None = None) -> int: - if not self._rolled: - await checkpoint_if_cancelled() - return self._fp.truncate(size) - - return await super().truncate(size) - - @overload - async def write(self: SpooledTemporaryFile[bytes], b: ReadableBuffer) -> int: ... - @overload - async def write(self: SpooledTemporaryFile[str], b: str) -> int: ... - - async def write(self, b: ReadableBuffer | str) -> int: - """ - Asynchronously write data to the spooled temporary file. - - If the file has not yet been rolled over, the data is written synchronously, - and a rollover is triggered if the size exceeds the maximum size. - - :param s: The data to write. - :return: The number of bytes written. - :raises RuntimeError: If the underlying file is not initialized. - - """ - if not self._rolled: - await checkpoint_if_cancelled() - result = self._fp.write(b) - await self._check() - return result - - return await super().write(b) # type: ignore[misc] - - @overload - async def writelines( - self: SpooledTemporaryFile[bytes], lines: Iterable[ReadableBuffer] - ) -> None: ... - @overload - async def writelines( - self: SpooledTemporaryFile[str], lines: Iterable[str] - ) -> None: ... - - async def writelines(self, lines: Iterable[str] | Iterable[ReadableBuffer]) -> None: - """ - Asynchronously write a list of lines to the spooled temporary file. - - If the file has not yet been rolled over, the lines are written synchronously, - and a rollover is triggered if the size exceeds the maximum size. - - :param lines: An iterable of lines to write. - :raises RuntimeError: If the underlying file is not initialized. - - """ - if not self._rolled: - await checkpoint_if_cancelled() - result = self._fp.writelines(lines) - await self._check() - return result - - return await super().writelines(lines) # type: ignore[misc] - - -class TemporaryDirectory(Generic[AnyStr]): - """ - An asynchronous temporary directory that is created and cleaned up automatically. - - This class provides an asynchronous context manager for creating a temporary - directory. It wraps Python's standard :class:`~tempfile.TemporaryDirectory` to - perform directory creation and cleanup operations in a background thread. - - :param suffix: Suffix to be added to the temporary directory name. - :param prefix: Prefix to be added to the temporary directory name. - :param dir: The parent directory where the temporary directory is created. - :param ignore_cleanup_errors: Whether to ignore errors during cleanup - (Python 3.10+). - :param delete: Whether to delete the directory upon closing (Python 3.12+). - """ - - def __init__( - self, - suffix: AnyStr | None = None, - prefix: AnyStr | None = None, - dir: AnyStr | None = None, - *, - ignore_cleanup_errors: bool = False, - delete: bool = True, - ) -> None: - self.suffix: AnyStr | None = suffix - self.prefix: AnyStr | None = prefix - self.dir: AnyStr | None = dir - self.ignore_cleanup_errors = ignore_cleanup_errors - self.delete = delete - - self._tempdir: tempfile.TemporaryDirectory | None = None - - async def __aenter__(self) -> str: - params: dict[str, Any] = { - "suffix": self.suffix, - "prefix": self.prefix, - "dir": self.dir, - } - if sys.version_info >= (3, 10): - params["ignore_cleanup_errors"] = self.ignore_cleanup_errors - - if sys.version_info >= (3, 12): - params["delete"] = self.delete - - self._tempdir = await to_thread.run_sync( - lambda: tempfile.TemporaryDirectory(**params) - ) - return await to_thread.run_sync(self._tempdir.__enter__) - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - traceback: TracebackType | None, - ) -> None: - if self._tempdir is not None: - await to_thread.run_sync( - self._tempdir.__exit__, exc_type, exc_value, traceback - ) - - async def cleanup(self) -> None: - if self._tempdir is not None: - await to_thread.run_sync(self._tempdir.cleanup) - - -@overload -async def mkstemp( - suffix: str | None = None, - prefix: str | None = None, - dir: str | None = None, - text: bool = False, -) -> tuple[int, str]: ... - - -@overload -async def mkstemp( - suffix: bytes | None = None, - prefix: bytes | None = None, - dir: bytes | None = None, - text: bool = False, -) -> tuple[int, bytes]: ... - - -async def mkstemp( - suffix: AnyStr | None = None, - prefix: AnyStr | None = None, - dir: AnyStr | None = None, - text: bool = False, -) -> tuple[int, str | bytes]: - """ - Asynchronously create a temporary file and return an OS-level handle and the file - name. - - This function wraps `tempfile.mkstemp` and executes it in a background thread. - - :param suffix: Suffix to be added to the file name. - :param prefix: Prefix to be added to the file name. - :param dir: Directory in which the temporary file is created. - :param text: Whether the file is opened in text mode. - :return: A tuple containing the file descriptor and the file name. - - """ - return await to_thread.run_sync(tempfile.mkstemp, suffix, prefix, dir, text) - - -@overload -async def mkdtemp( - suffix: str | None = None, - prefix: str | None = None, - dir: str | None = None, -) -> str: ... - - -@overload -async def mkdtemp( - suffix: bytes | None = None, - prefix: bytes | None = None, - dir: bytes | None = None, -) -> bytes: ... - - -async def mkdtemp( - suffix: AnyStr | None = None, - prefix: AnyStr | None = None, - dir: AnyStr | None = None, -) -> str | bytes: - """ - Asynchronously create a temporary directory and return its path. - - This function wraps `tempfile.mkdtemp` and executes it in a background thread. - - :param suffix: Suffix to be added to the directory name. - :param prefix: Prefix to be added to the directory name. - :param dir: Parent directory where the temporary directory is created. - :return: The path of the created temporary directory. - - """ - return await to_thread.run_sync(tempfile.mkdtemp, suffix, prefix, dir) - - -async def gettempdir() -> str: - """ - Asynchronously return the name of the directory used for temporary files. - - This function wraps `tempfile.gettempdir` and executes it in a background thread. - - :return: The path of the temporary directory as a string. - - """ - return await to_thread.run_sync(tempfile.gettempdir) - - -async def gettempdirb() -> bytes: - """ - Asynchronously return the name of the directory used for temporary files in bytes. - - This function wraps `tempfile.gettempdirb` and executes it in a background thread. - - :return: The path of the temporary directory as bytes. - - """ - return await to_thread.run_sync(tempfile.gettempdirb) diff --git a/.env/Lib/site-packages/anyio/_core/_testing.py b/.env/Lib/site-packages/anyio/_core/_testing.py deleted file mode 100644 index 369e65c..0000000 --- a/.env/Lib/site-packages/anyio/_core/_testing.py +++ /dev/null @@ -1,82 +0,0 @@ -from __future__ import annotations - -from collections.abc import Awaitable, Generator -from typing import Any, cast - -from ._eventloop import get_async_backend - - -class TaskInfo: - """ - Represents an asynchronous task. - - :ivar int id: the unique identifier of the task - :ivar parent_id: the identifier of the parent task, if any - :vartype parent_id: Optional[int] - :ivar str name: the description of the task (if any) - :ivar ~collections.abc.Coroutine coro: the coroutine object of the task - """ - - __slots__ = "_name", "id", "parent_id", "name", "coro" - - def __init__( - self, - id: int, - parent_id: int | None, - name: str | None, - coro: Generator[Any, Any, Any] | Awaitable[Any], - ): - func = get_current_task - self._name = f"{func.__module__}.{func.__qualname__}" - self.id: int = id - self.parent_id: int | None = parent_id - self.name: str | None = name - self.coro: Generator[Any, Any, Any] | Awaitable[Any] = coro - - def __eq__(self, other: object) -> bool: - if isinstance(other, TaskInfo): - return self.id == other.id - - return NotImplemented - - def __hash__(self) -> int: - return hash(self.id) - - def __repr__(self) -> str: - return f"{self.__class__.__name__}(id={self.id!r}, name={self.name!r})" - - def has_pending_cancellation(self) -> bool: - """ - Return ``True`` if the task has a cancellation pending, ``False`` otherwise. - - """ - return False - - -def get_current_task() -> TaskInfo: - """ - Return the current task. - - :return: a representation of the current task - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().get_current_task() - - -def get_running_tasks() -> list[TaskInfo]: - """ - Return a list of running tasks in the current event loop. - - :return: a list of task info objects - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return cast("list[TaskInfo]", get_async_backend().get_running_tasks()) - - -async def wait_all_tasks_blocked() -> None: - """Wait until all other tasks are waiting for something.""" - await get_async_backend().wait_all_tasks_blocked() diff --git a/.env/Lib/site-packages/anyio/_core/_typedattr.py b/.env/Lib/site-packages/anyio/_core/_typedattr.py deleted file mode 100644 index f358a44..0000000 --- a/.env/Lib/site-packages/anyio/_core/_typedattr.py +++ /dev/null @@ -1,81 +0,0 @@ -from __future__ import annotations - -from collections.abc import Callable, Mapping -from typing import Any, TypeVar, final, overload - -from ._exceptions import TypedAttributeLookupError - -T_Attr = TypeVar("T_Attr") -T_Default = TypeVar("T_Default") -undefined = object() - - -def typed_attribute() -> Any: - """Return a unique object, used to mark typed attributes.""" - return object() - - -class TypedAttributeSet: - """ - Superclass for typed attribute collections. - - Checks that every public attribute of every subclass has a type annotation. - """ - - def __init_subclass__(cls) -> None: - annotations: dict[str, Any] = getattr(cls, "__annotations__", {}) - for attrname in dir(cls): - if not attrname.startswith("_") and attrname not in annotations: - raise TypeError( - f"Attribute {attrname!r} is missing its type annotation" - ) - - super().__init_subclass__() - - -class TypedAttributeProvider: - """Base class for classes that wish to provide typed extra attributes.""" - - @property - def extra_attributes(self) -> Mapping[T_Attr, Callable[[], T_Attr]]: - """ - A mapping of the extra attributes to callables that return the corresponding - values. - - If the provider wraps another provider, the attributes from that wrapper should - also be included in the returned mapping (but the wrapper may override the - callables from the wrapped instance). - - """ - return {} - - @overload - def extra(self, attribute: T_Attr) -> T_Attr: ... - - @overload - def extra(self, attribute: T_Attr, default: T_Default) -> T_Attr | T_Default: ... - - @final - def extra(self, attribute: Any, default: object = undefined) -> object: - """ - extra(attribute, default=undefined) - - Return the value of the given typed extra attribute. - - :param attribute: the attribute (member of a :class:`~TypedAttributeSet`) to - look for - :param default: the value that should be returned if no value is found for the - attribute - :raises ~anyio.TypedAttributeLookupError: if the search failed and no default - value was given - - """ - try: - getter = self.extra_attributes[attribute] - except KeyError: - if default is undefined: - raise TypedAttributeLookupError("Attribute not found") from None - else: - return default - - return getter() diff --git a/.env/Lib/site-packages/anyio/abc/__init__.py b/.env/Lib/site-packages/anyio/abc/__init__.py deleted file mode 100644 index d560ce3..0000000 --- a/.env/Lib/site-packages/anyio/abc/__init__.py +++ /dev/null @@ -1,58 +0,0 @@ -from __future__ import annotations - -from ._eventloop import AsyncBackend as AsyncBackend -from ._resources import AsyncResource as AsyncResource -from ._sockets import ConnectedUDPSocket as ConnectedUDPSocket -from ._sockets import ConnectedUNIXDatagramSocket as ConnectedUNIXDatagramSocket -from ._sockets import IPAddressType as IPAddressType -from ._sockets import IPSockAddrType as IPSockAddrType -from ._sockets import SocketAttribute as SocketAttribute -from ._sockets import SocketListener as SocketListener -from ._sockets import SocketStream as SocketStream -from ._sockets import UDPPacketType as UDPPacketType -from ._sockets import UDPSocket as UDPSocket -from ._sockets import UNIXDatagramPacketType as UNIXDatagramPacketType -from ._sockets import UNIXDatagramSocket as UNIXDatagramSocket -from ._sockets import UNIXSocketStream as UNIXSocketStream -from ._streams import AnyByteReceiveStream as AnyByteReceiveStream -from ._streams import AnyByteSendStream as AnyByteSendStream -from ._streams import AnyByteStream as AnyByteStream -from ._streams import AnyByteStreamConnectable as AnyByteStreamConnectable -from ._streams import AnyUnreliableByteReceiveStream as AnyUnreliableByteReceiveStream -from ._streams import AnyUnreliableByteSendStream as AnyUnreliableByteSendStream -from ._streams import AnyUnreliableByteStream as AnyUnreliableByteStream -from ._streams import ByteReceiveStream as ByteReceiveStream -from ._streams import ByteSendStream as ByteSendStream -from ._streams import ByteStream as ByteStream -from ._streams import ByteStreamConnectable as ByteStreamConnectable -from ._streams import Listener as Listener -from ._streams import ObjectReceiveStream as ObjectReceiveStream -from ._streams import ObjectSendStream as ObjectSendStream -from ._streams import ObjectStream as ObjectStream -from ._streams import ObjectStreamConnectable as ObjectStreamConnectable -from ._streams import UnreliableObjectReceiveStream as UnreliableObjectReceiveStream -from ._streams import UnreliableObjectSendStream as UnreliableObjectSendStream -from ._streams import UnreliableObjectStream as UnreliableObjectStream -from ._subprocesses import Process as Process -from ._tasks import TaskGroup as TaskGroup -from ._tasks import TaskStatus as TaskStatus -from ._testing import TestRunner as TestRunner - -# Re-exported here, for backwards compatibility -# isort: off -from .._core._synchronization import ( - CapacityLimiter as CapacityLimiter, - Condition as Condition, - Event as Event, - Lock as Lock, - Semaphore as Semaphore, -) -from .._core._tasks import CancelScope as CancelScope -from ..from_thread import BlockingPortal as BlockingPortal - -# Re-export imports so they look like they live directly in this package -for __value in list(locals().values()): - if getattr(__value, "__module__", "").startswith("anyio.abc."): - __value.__module__ = __name__ - -del __value diff --git a/.env/Lib/site-packages/anyio/abc/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/anyio/abc/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 886f35611d5ccdd995793d92a46875dc2a9a5e1d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2359 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 BpXvYr diff --git a/.env/Lib/site-packages/anyio/abc/__pycache__/_sockets.cpython-312.pyc b/.env/Lib/site-packages/anyio/abc/__pycache__/_sockets.cpython-312.pyc deleted file mode 100644 index 5c5ea892582d334b5c1536c70f3514ddf23ede15..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18424 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 diff --git a/.env/Lib/site-packages/anyio/abc/__pycache__/_streams.cpython-312.pyc b/.env/Lib/site-packages/anyio/abc/__pycache__/_streams.cpython-312.pyc deleted file mode 100644 index 5ce7eb5231a4505ee60712153a3a11b5d951f60d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9885 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 aPfk1_r++4I{h7Z)=>q-A`HO^180Nng3MiET diff --git a/.env/Lib/site-packages/anyio/abc/__pycache__/_tasks.cpython-312.pyc b/.env/Lib/site-packages/anyio/abc/__pycache__/_tasks.cpython-312.pyc deleted file mode 100644 index 2b794a49c14193d3221bf7a4f1227088cde486a6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5138 zcmb_g&5snv74M$u`JUZnKlreXaoNa*f%RYuV#Oom7$ZawVS~&%QiesnwcWM54L#kH zsveeEsMc|-pB8~n%_)JR1G}8{?(7YpH3Ra=kzgriV0zEEtrP!k>MJy=^GuhZJI>q{9HSS zcizu;thUuDv(txz&P02nGufW(Otq&>L*y>A z;g+MfTe`Aj7_x*N%9u6Xo;Ho3sJ~{CuP4l_=y?^4tYY45d$u#zp2OU-=FE&ZqArVW z&8_=8JGvxBi}Vr`+79J8tGv?jCSw-RW%9Gdbh5;m%w)+*vUx z_B?}O+1O`w-yHgO3hUEd8UML{uCe<8E!Z&Q8V`an;*l2yYN3&fODEnrc}_$;p5!Yk zlH7?pB3ch!bX87la4+I3zKF{wx$hH&6($xjO000_%3{AO8WtGeJ%7qR`OQ-&&z@O0 z9p_I3{kVKyhEeE*eq0W(3h9R&D?Xr8J(x%YDbeijaM{R7;ykDkI~Uhe_F}J#eT(k} zU2OCqm&~c1&l}nEYQCxDxD)0FbBns6D zxuzB!td%{fp3SWmSRWa}Xq&Fl&WXHRaLw!1&2(Vf0;lNa#yM`uExXqBLfh)+8gRsZGasF}EzeL*KYzF5Ey9TT1w=iiWbUE}U#Fy{Cj!OYeqm$j)(RT?F10 zvDED^o%0+SqVKBb2({D{!PTX2c`HlGi^Qv>9KHrN5Bgqsgs(V9>Yxh#u9y+krovT!w9F9uzz+<4w1%Y> zw7FxX_KOQ)ATGs4DWaYXVv9a8>=jIsyQpbUbBG$!bK12zHFRwDeMVD8!y1LSY}*0v z2-}XUw%rNco=r{(O4FR@l}a-C)~=|h)UmTOvV+m$!Qt(x*Py(Xq)R# zIFkTS!Oghp2hO?-gRrNxtDz0o0E$AUj@I&kF3izbWGe!rmiD5A!!(C^0VW_u;dgxo z^kmMu=ex;*dfOHjiGYEREHnmjiCZ_D42UgSE%xmo5zY(end)`Bp^VsSD475R$RsLO z0pOwZgkqjzKtkoYLP7?({ffvu{LmqBtqbWoB8a$u=unfL2@oYR@|>Q}dJj3QL7;ykHGFSQq4xE(-%aHxSn%26afCdWLZa zl?M^?!>~Kgc;K=Q?_+dNWotUTrv&R^b{COM$`tGTs)s{+0nX&a}G_F|bt*ybf zBd~>MB4iW^Q5q~SvxAg# z5cip)T8@td)mo-w!;NTEqsesfMWLhv2#%E0Ww0!4o#YX=0pBOoBL5#|xhTvQ-nR2F z^B-)G49m-uPe1{j(h8|DcFYmx(19N#Ak#7v$gBzECC^FXm)uBW+R#0jS`T}^n~6jg zNA!6P1lJy`N2gg>XYDkg>69z*$s~1;9g0?X;=#s_kOt0^LoC@Su^XZYl5blUx=BsB zP59x#GUoq1KzYEzZW`q}SP9~_VzX#{T6B_r=xHTBl4PbtvzaotpLE9&osVemC21ou@40D;u2t>pv^ zuzuum-`f<+{~PAWqX2?O0H&bVL`4N9>V%&Na~31F=;HZTn{|bkgz*Oi70qiT-2ocsj)4NGCR_I#tb0x=p>D@JO;+*{P41Q@nUlQyEweMUAi=% z;3DDfMcgwlU6Nm;DLWn=m}Ht_ase1Co6{|x&$ehbn~z^pzC*%M!b&Jzq0^5SxUg#a z2+?Kv7G9grk}@v7!<9I7jq(6e;-z8D{tX&1x2i-AMCcqLS%EVfw_EHBjkv;vL8IFLP$$jF~ct#B5r=SU79m zo0%UJhyZWql|f09Wn=o~K}C~^NJQ%SU+77a>IQvg6nGURQ?653V=rBwR(rUY;rb*| zE*NoM^;Mk5Wm-~fX{E+f_$|Xv$|2RHE&>gSk`NbRGQN^!qKlvrN3ue_YIW4>a+=1^=<)bfLyBf{q1TnF+)17K!brdT=&M*_%kB34 zs(d`T5WXePV;q5t`Wc!*-ZafS#!H_Y)w{-lyT*|_#tV0iH*Xnl-pd#Nke~f5Kl|av z@ACWpWbOO2)!5FvYaRZrb@(sV8~5`&?pTNJ*JeI!-Ky<-zx+p{e!lq2W52JxO<%h| quHLHcf4}^&SU2k*?)muOXVd#18lWHUJ8PO}&5sZK!_b7b$iD&V`*HCA diff --git a/.env/Lib/site-packages/anyio/abc/__pycache__/_testing.cpython-312.pyc b/.env/Lib/site-packages/anyio/abc/__pycache__/_testing.cpython-312.pyc deleted file mode 100644 index 941f70f0ce493098dac41a9bcbd44919c9d5b66d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2834 zcmd5;&2QX96d!xnySCR!ztWaQl#D89wGg%kBtTJxXrrnsqB*oFLe56ktUXCg{IQI^ zX*TMihzRu$Kq_Z$Rrw=0aN`D-$`VRZPu#X$#GxnN8+$kFG*ms}UA=jJ@6Gd@-+MD} zew~^!2|U02{GETcPRLVy89iFgD8p@F9uSXsN^uz1C;c)F+V!t%Lhkn+znyvlC%QQ^m~l`TBet4?U^2eb`Oo6zb9w3b(!&>H=E zb1E~Jd6M4jMX=-@VN!UmMlS0ny?~`Wp^OU9`O!M%TRci>5XapXy}HFkpSl1J4VlMj zx&h#kG;&qqYA`g|NhnU1_ z!h^L04d{#8iwo`B2^UHGX6(iE26H!f%tPzsB-s<)xJb_v#_sHWTC+;_f zz!JBLbmEcmqwoRg6LAz6ffJimaU6zB=i9E6LcL^lK;G8Uz&G6MKX)bTTrbIRb46;IbkB~RD&XJ$K(%YwU1IrsH8HPlr`3SFS3n z-6iX>!(ubM zUTK$Hlm`761PBdEC&5VgaS0miZ?!xxVj46}U7UttXBLl~l`Aaa zSGQf>#of9}@6x3hj{P^#V5X!Ssm_e&Caq=Yl)M8+D0aJ%!Fe=jbmBNDB+Z&Qi?Z{h z&biWRwdQHLudX=DAd32{t8%$7raj+HSD+W?QX#!sqc#iHtN-D4+>}?qo18pP!8&&< zEr@ zc(Lm^0?%oI_aQNhWDdzWB<~=>#vm(No-{IHk~Bb`{03x~JX0&0J~S0=W>6!wnSrWl zvrmrC4>TA-a{T}m%Wn@2?mRv)#o}9Wg&|rj_$L0ogHC=qT)tdcno4HC#f!NCOOobM zM02f&tp{Mk131&LW|OP}eOh2|lsINb+&Bo}eu1~S7OL>BAh0b99f<)XtHD(04+8I~ z0@0KekON!>{e^xh&*3NH1F*vjMlug%P*D`+4|47)vG&M?J#zhba($1?{i#{^m%m-T Sr~Y00P_cga=pO=uT+_cf{_hq5 diff --git a/.env/Lib/site-packages/anyio/abc/_eventloop.py b/.env/Lib/site-packages/anyio/abc/_eventloop.py deleted file mode 100644 index b1bd085..0000000 --- a/.env/Lib/site-packages/anyio/abc/_eventloop.py +++ /dev/null @@ -1,414 +0,0 @@ -from __future__ import annotations - -import math -import sys -from abc import ABCMeta, abstractmethod -from collections.abc import AsyncIterator, Awaitable, Callable, Sequence -from contextlib import AbstractContextManager -from os import PathLike -from signal import Signals -from socket import AddressFamily, SocketKind, socket -from typing import ( - IO, - TYPE_CHECKING, - Any, - TypeVar, - Union, - overload, -) - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -if sys.version_info >= (3, 10): - from typing import TypeAlias -else: - from typing_extensions import TypeAlias - -if TYPE_CHECKING: - from _typeshed import FileDescriptorLike - - from .._core._synchronization import CapacityLimiter, Event, Lock, Semaphore - from .._core._tasks import CancelScope - from .._core._testing import TaskInfo - from ._sockets import ( - ConnectedUDPSocket, - ConnectedUNIXDatagramSocket, - IPSockAddrType, - SocketListener, - SocketStream, - UDPSocket, - UNIXDatagramSocket, - UNIXSocketStream, - ) - from ._subprocesses import Process - from ._tasks import TaskGroup - from ._testing import TestRunner - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") -StrOrBytesPath: TypeAlias = Union[str, bytes, "PathLike[str]", "PathLike[bytes]"] - - -class AsyncBackend(metaclass=ABCMeta): - @classmethod - @abstractmethod - def run( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - kwargs: dict[str, Any], - options: dict[str, Any], - ) -> T_Retval: - """ - Run the given coroutine function in an asynchronous event loop. - - The current thread must not be already running an event loop. - - :param func: a coroutine function - :param args: positional arguments to ``func`` - :param kwargs: positional arguments to ``func`` - :param options: keyword arguments to call the backend ``run()`` implementation - with - :return: the return value of the coroutine function - """ - - @classmethod - @abstractmethod - def current_token(cls) -> object: - """ - Return an object that allows other threads to run code inside the event loop. - - :return: a token object, specific to the event loop running in the current - thread - """ - - @classmethod - @abstractmethod - def current_time(cls) -> float: - """ - Return the current value of the event loop's internal clock. - - :return: the clock value (seconds) - """ - - @classmethod - @abstractmethod - def cancelled_exception_class(cls) -> type[BaseException]: - """Return the exception class that is raised in a task if it's cancelled.""" - - @classmethod - @abstractmethod - async def checkpoint(cls) -> None: - """ - Check if the task has been cancelled, and allow rescheduling of other tasks. - - This is effectively the same as running :meth:`checkpoint_if_cancelled` and then - :meth:`cancel_shielded_checkpoint`. - """ - - @classmethod - async def checkpoint_if_cancelled(cls) -> None: - """ - Check if the current task group has been cancelled. - - This will check if the task has been cancelled, but will not allow other tasks - to be scheduled if not. - - """ - if cls.current_effective_deadline() == -math.inf: - await cls.checkpoint() - - @classmethod - async def cancel_shielded_checkpoint(cls) -> None: - """ - Allow the rescheduling of other tasks. - - This will give other tasks the opportunity to run, but without checking if the - current task group has been cancelled, unlike with :meth:`checkpoint`. - - """ - with cls.create_cancel_scope(shield=True): - await cls.sleep(0) - - @classmethod - @abstractmethod - async def sleep(cls, delay: float) -> None: - """ - Pause the current task for the specified duration. - - :param delay: the duration, in seconds - """ - - @classmethod - @abstractmethod - def create_cancel_scope( - cls, *, deadline: float = math.inf, shield: bool = False - ) -> CancelScope: - pass - - @classmethod - @abstractmethod - def current_effective_deadline(cls) -> float: - """ - Return the nearest deadline among all the cancel scopes effective for the - current task. - - :return: - - a clock value from the event loop's internal clock - - ``inf`` if there is no deadline in effect - - ``-inf`` if the current scope has been cancelled - :rtype: float - """ - - @classmethod - @abstractmethod - def create_task_group(cls) -> TaskGroup: - pass - - @classmethod - @abstractmethod - def create_event(cls) -> Event: - pass - - @classmethod - @abstractmethod - def create_lock(cls, *, fast_acquire: bool) -> Lock: - pass - - @classmethod - @abstractmethod - def create_semaphore( - cls, - initial_value: int, - *, - max_value: int | None = None, - fast_acquire: bool = False, - ) -> Semaphore: - pass - - @classmethod - @abstractmethod - def create_capacity_limiter(cls, total_tokens: float) -> CapacityLimiter: - pass - - @classmethod - @abstractmethod - async def run_sync_in_worker_thread( - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - abandon_on_cancel: bool = False, - limiter: CapacityLimiter | None = None, - ) -> T_Retval: - pass - - @classmethod - @abstractmethod - def check_cancelled(cls) -> None: - pass - - @classmethod - @abstractmethod - def run_async_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - pass - - @classmethod - @abstractmethod - def run_sync_from_thread( - cls, - func: Callable[[Unpack[PosArgsT]], T_Retval], - args: tuple[Unpack[PosArgsT]], - token: object, - ) -> T_Retval: - pass - - @classmethod - @abstractmethod - async def open_process( - cls, - command: StrOrBytesPath | Sequence[StrOrBytesPath], - *, - stdin: int | IO[Any] | None, - stdout: int | IO[Any] | None, - stderr: int | IO[Any] | None, - **kwargs: Any, - ) -> Process: - pass - - @classmethod - @abstractmethod - def setup_process_pool_exit_at_shutdown(cls, workers: set[Process]) -> None: - pass - - @classmethod - @abstractmethod - async def connect_tcp( - cls, host: str, port: int, local_address: IPSockAddrType | None = None - ) -> SocketStream: - pass - - @classmethod - @abstractmethod - async def connect_unix(cls, path: str | bytes) -> UNIXSocketStream: - pass - - @classmethod - @abstractmethod - def create_tcp_listener(cls, sock: socket) -> SocketListener: - pass - - @classmethod - @abstractmethod - def create_unix_listener(cls, sock: socket) -> SocketListener: - pass - - @classmethod - @abstractmethod - async def create_udp_socket( - cls, - family: AddressFamily, - local_address: IPSockAddrType | None, - remote_address: IPSockAddrType | None, - reuse_port: bool, - ) -> UDPSocket | ConnectedUDPSocket: - pass - - @classmethod - @overload - async def create_unix_datagram_socket( - cls, raw_socket: socket, remote_path: None - ) -> UNIXDatagramSocket: ... - - @classmethod - @overload - async def create_unix_datagram_socket( - cls, raw_socket: socket, remote_path: str | bytes - ) -> ConnectedUNIXDatagramSocket: ... - - @classmethod - @abstractmethod - async def create_unix_datagram_socket( - cls, raw_socket: socket, remote_path: str | bytes | None - ) -> UNIXDatagramSocket | ConnectedUNIXDatagramSocket: - pass - - @classmethod - @abstractmethod - async def getaddrinfo( - cls, - host: bytes | str | None, - port: str | int | None, - *, - family: int | AddressFamily = 0, - type: int | SocketKind = 0, - proto: int = 0, - flags: int = 0, - ) -> Sequence[ - tuple[ - AddressFamily, - SocketKind, - int, - str, - tuple[str, int] | tuple[str, int, int, int] | tuple[int, bytes], - ] - ]: - pass - - @classmethod - @abstractmethod - async def getnameinfo( - cls, sockaddr: IPSockAddrType, flags: int = 0 - ) -> tuple[str, str]: - pass - - @classmethod - @abstractmethod - async def wait_readable(cls, obj: FileDescriptorLike) -> None: - pass - - @classmethod - @abstractmethod - async def wait_writable(cls, obj: FileDescriptorLike) -> None: - pass - - @classmethod - @abstractmethod - def notify_closing(cls, obj: FileDescriptorLike) -> None: - pass - - @classmethod - @abstractmethod - async def wrap_listener_socket(cls, sock: socket) -> SocketListener: - pass - - @classmethod - @abstractmethod - async def wrap_stream_socket(cls, sock: socket) -> SocketStream: - pass - - @classmethod - @abstractmethod - async def wrap_unix_stream_socket(cls, sock: socket) -> UNIXSocketStream: - pass - - @classmethod - @abstractmethod - async def wrap_udp_socket(cls, sock: socket) -> UDPSocket: - pass - - @classmethod - @abstractmethod - async def wrap_connected_udp_socket(cls, sock: socket) -> ConnectedUDPSocket: - pass - - @classmethod - @abstractmethod - async def wrap_unix_datagram_socket(cls, sock: socket) -> UNIXDatagramSocket: - pass - - @classmethod - @abstractmethod - async def wrap_connected_unix_datagram_socket( - cls, sock: socket - ) -> ConnectedUNIXDatagramSocket: - pass - - @classmethod - @abstractmethod - def current_default_thread_limiter(cls) -> CapacityLimiter: - pass - - @classmethod - @abstractmethod - def open_signal_receiver( - cls, *signals: Signals - ) -> AbstractContextManager[AsyncIterator[Signals]]: - pass - - @classmethod - @abstractmethod - def get_current_task(cls) -> TaskInfo: - pass - - @classmethod - @abstractmethod - def get_running_tasks(cls) -> Sequence[TaskInfo]: - pass - - @classmethod - @abstractmethod - async def wait_all_tasks_blocked(cls) -> None: - pass - - @classmethod - @abstractmethod - def create_test_runner(cls, options: dict[str, Any]) -> TestRunner: - pass diff --git a/.env/Lib/site-packages/anyio/abc/_resources.py b/.env/Lib/site-packages/anyio/abc/_resources.py deleted file mode 100644 index 10df115..0000000 --- a/.env/Lib/site-packages/anyio/abc/_resources.py +++ /dev/null @@ -1,33 +0,0 @@ -from __future__ import annotations - -from abc import ABCMeta, abstractmethod -from types import TracebackType -from typing import TypeVar - -T = TypeVar("T") - - -class AsyncResource(metaclass=ABCMeta): - """ - Abstract base class for all closeable asynchronous resources. - - Works as an asynchronous context manager which returns the instance itself on enter, - and calls :meth:`aclose` on exit. - """ - - __slots__ = () - - async def __aenter__(self: T) -> T: - return self - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - await self.aclose() - - @abstractmethod - async def aclose(self) -> None: - """Close the resource.""" diff --git a/.env/Lib/site-packages/anyio/abc/_sockets.py b/.env/Lib/site-packages/anyio/abc/_sockets.py deleted file mode 100644 index 3ff60d4..0000000 --- a/.env/Lib/site-packages/anyio/abc/_sockets.py +++ /dev/null @@ -1,405 +0,0 @@ -from __future__ import annotations - -import errno -import socket -import sys -from abc import abstractmethod -from collections.abc import Callable, Collection, Mapping -from contextlib import AsyncExitStack -from io import IOBase -from ipaddress import IPv4Address, IPv6Address -from socket import AddressFamily -from typing import Any, TypeVar, Union - -from .._core._eventloop import get_async_backend -from .._core._typedattr import ( - TypedAttributeProvider, - TypedAttributeSet, - typed_attribute, -) -from ._streams import ByteStream, Listener, UnreliableObjectStream -from ._tasks import TaskGroup - -if sys.version_info >= (3, 10): - from typing import TypeAlias -else: - from typing_extensions import TypeAlias - -IPAddressType: TypeAlias = Union[str, IPv4Address, IPv6Address] -IPSockAddrType: TypeAlias = tuple[str, int] -SockAddrType: TypeAlias = Union[IPSockAddrType, str] -UDPPacketType: TypeAlias = tuple[bytes, IPSockAddrType] -UNIXDatagramPacketType: TypeAlias = tuple[bytes, str] -T_Retval = TypeVar("T_Retval") - - -def _validate_socket( - sock_or_fd: socket.socket | int, - sock_type: socket.SocketKind, - addr_family: socket.AddressFamily = socket.AF_UNSPEC, - *, - require_connected: bool = False, - require_bound: bool = False, -) -> socket.socket: - if isinstance(sock_or_fd, int): - try: - sock = socket.socket(fileno=sock_or_fd) - except OSError as exc: - if exc.errno == errno.ENOTSOCK: - raise ValueError( - "the file descriptor does not refer to a socket" - ) from exc - elif require_connected: - raise ValueError("the socket must be connected") from exc - elif require_bound: - raise ValueError("the socket must be bound to a local address") from exc - else: - raise - elif isinstance(sock_or_fd, socket.socket): - sock = sock_or_fd - else: - raise TypeError( - f"expected an int or socket, got {type(sock_or_fd).__qualname__} instead" - ) - - try: - if require_connected: - try: - sock.getpeername() - except OSError as exc: - raise ValueError("the socket must be connected") from exc - - if require_bound: - try: - if sock.family in (socket.AF_INET, socket.AF_INET6): - bound_addr = sock.getsockname()[1] - else: - bound_addr = sock.getsockname() - except OSError: - bound_addr = None - - if not bound_addr: - raise ValueError("the socket must be bound to a local address") - - if addr_family != socket.AF_UNSPEC and sock.family != addr_family: - raise ValueError( - f"address family mismatch: expected {addr_family.name}, got " - f"{sock.family.name}" - ) - - if sock.type != sock_type: - raise ValueError( - f"socket type mismatch: expected {sock_type.name}, got {sock.type.name}" - ) - except BaseException: - # Avoid ResourceWarning from the locally constructed socket object - if isinstance(sock_or_fd, int): - sock.detach() - - raise - - sock.setblocking(False) - return sock - - -class SocketAttribute(TypedAttributeSet): - """ - .. attribute:: family - :type: socket.AddressFamily - - the address family of the underlying socket - - .. attribute:: local_address - :type: tuple[str, int] | str - - the local address the underlying socket is connected to - - .. attribute:: local_port - :type: int - - for IP based sockets, the local port the underlying socket is bound to - - .. attribute:: raw_socket - :type: socket.socket - - the underlying stdlib socket object - - .. attribute:: remote_address - :type: tuple[str, int] | str - - the remote address the underlying socket is connected to - - .. attribute:: remote_port - :type: int - - for IP based sockets, the remote port the underlying socket is connected to - """ - - family: AddressFamily = typed_attribute() - local_address: SockAddrType = typed_attribute() - local_port: int = typed_attribute() - raw_socket: socket.socket = typed_attribute() - remote_address: SockAddrType = typed_attribute() - remote_port: int = typed_attribute() - - -class _SocketProvider(TypedAttributeProvider): - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - from .._core._sockets import convert_ipv6_sockaddr as convert - - attributes: dict[Any, Callable[[], Any]] = { - SocketAttribute.family: lambda: self._raw_socket.family, - SocketAttribute.local_address: lambda: convert( - self._raw_socket.getsockname() - ), - SocketAttribute.raw_socket: lambda: self._raw_socket, - } - try: - peername: tuple[str, int] | None = convert(self._raw_socket.getpeername()) - except OSError: - peername = None - - # Provide the remote address for connected sockets - if peername is not None: - attributes[SocketAttribute.remote_address] = lambda: peername - - # Provide local and remote ports for IP based sockets - if self._raw_socket.family in (AddressFamily.AF_INET, AddressFamily.AF_INET6): - attributes[SocketAttribute.local_port] = ( - lambda: self._raw_socket.getsockname()[1] - ) - if peername is not None: - remote_port = peername[1] - attributes[SocketAttribute.remote_port] = lambda: remote_port - - return attributes - - @property - @abstractmethod - def _raw_socket(self) -> socket.socket: - pass - - -class SocketStream(ByteStream, _SocketProvider): - """ - Transports bytes over a socket. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket(cls, sock_or_fd: socket.socket | int) -> SocketStream: - """ - Wrap an existing socket object or file descriptor as a socket stream. - - The newly created socket wrapper takes ownership of the socket being passed in. - The existing socket must already be connected. - - :param sock_or_fd: a socket object or file descriptor - :return: a socket stream - - """ - sock = _validate_socket(sock_or_fd, socket.SOCK_STREAM, require_connected=True) - return await get_async_backend().wrap_stream_socket(sock) - - -class UNIXSocketStream(SocketStream): - @classmethod - async def from_socket(cls, sock_or_fd: socket.socket | int) -> UNIXSocketStream: - """ - Wrap an existing socket object or file descriptor as a UNIX socket stream. - - The newly created socket wrapper takes ownership of the socket being passed in. - The existing socket must already be connected. - - :param sock_or_fd: a socket object or file descriptor - :return: a UNIX socket stream - - """ - sock = _validate_socket( - sock_or_fd, socket.SOCK_STREAM, socket.AF_UNIX, require_connected=True - ) - return await get_async_backend().wrap_unix_stream_socket(sock) - - @abstractmethod - async def send_fds(self, message: bytes, fds: Collection[int | IOBase]) -> None: - """ - Send file descriptors along with a message to the peer. - - :param message: a non-empty bytestring - :param fds: a collection of files (either numeric file descriptors or open file - or socket objects) - """ - - @abstractmethod - async def receive_fds(self, msglen: int, maxfds: int) -> tuple[bytes, list[int]]: - """ - Receive file descriptors along with a message from the peer. - - :param msglen: length of the message to expect from the peer - :param maxfds: maximum number of file descriptors to expect from the peer - :return: a tuple of (message, file descriptors) - """ - - -class SocketListener(Listener[SocketStream], _SocketProvider): - """ - Listens to incoming socket connections. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket( - cls, - sock_or_fd: socket.socket | int, - ) -> SocketListener: - """ - Wrap an existing socket object or file descriptor as a socket listener. - - The newly created listener takes ownership of the socket being passed in. - - :param sock_or_fd: a socket object or file descriptor - :return: a socket listener - - """ - sock = _validate_socket(sock_or_fd, socket.SOCK_STREAM, require_bound=True) - return await get_async_backend().wrap_listener_socket(sock) - - @abstractmethod - async def accept(self) -> SocketStream: - """Accept an incoming connection.""" - - async def serve( - self, - handler: Callable[[SocketStream], Any], - task_group: TaskGroup | None = None, - ) -> None: - from .. import create_task_group - - async with AsyncExitStack() as stack: - if task_group is None: - task_group = await stack.enter_async_context(create_task_group()) - - while True: - stream = await self.accept() - task_group.start_soon(handler, stream) - - -class UDPSocket(UnreliableObjectStream[UDPPacketType], _SocketProvider): - """ - Represents an unconnected UDP socket. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket(cls, sock_or_fd: socket.socket | int) -> UDPSocket: - """ - Wrap an existing socket object or file descriptor as a UDP socket. - - The newly created socket wrapper takes ownership of the socket being passed in. - The existing socket must be bound to a local address. - - :param sock_or_fd: a socket object or file descriptor - :return: a UDP socket - - """ - sock = _validate_socket(sock_or_fd, socket.SOCK_DGRAM, require_bound=True) - return await get_async_backend().wrap_udp_socket(sock) - - async def sendto(self, data: bytes, host: str, port: int) -> None: - """ - Alias for :meth:`~.UnreliableObjectSendStream.send` ((data, (host, port))). - - """ - return await self.send((data, (host, port))) - - -class ConnectedUDPSocket(UnreliableObjectStream[bytes], _SocketProvider): - """ - Represents an connected UDP socket. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket(cls, sock_or_fd: socket.socket | int) -> ConnectedUDPSocket: - """ - Wrap an existing socket object or file descriptor as a connected UDP socket. - - The newly created socket wrapper takes ownership of the socket being passed in. - The existing socket must already be connected. - - :param sock_or_fd: a socket object or file descriptor - :return: a connected UDP socket - - """ - sock = _validate_socket( - sock_or_fd, - socket.SOCK_DGRAM, - require_connected=True, - ) - return await get_async_backend().wrap_connected_udp_socket(sock) - - -class UNIXDatagramSocket( - UnreliableObjectStream[UNIXDatagramPacketType], _SocketProvider -): - """ - Represents an unconnected Unix datagram socket. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket( - cls, - sock_or_fd: socket.socket | int, - ) -> UNIXDatagramSocket: - """ - Wrap an existing socket object or file descriptor as a UNIX datagram - socket. - - The newly created socket wrapper takes ownership of the socket being passed in. - - :param sock_or_fd: a socket object or file descriptor - :return: a UNIX datagram socket - - """ - sock = _validate_socket(sock_or_fd, socket.SOCK_DGRAM, socket.AF_UNIX) - return await get_async_backend().wrap_unix_datagram_socket(sock) - - async def sendto(self, data: bytes, path: str) -> None: - """Alias for :meth:`~.UnreliableObjectSendStream.send` ((data, path)).""" - return await self.send((data, path)) - - -class ConnectedUNIXDatagramSocket(UnreliableObjectStream[bytes], _SocketProvider): - """ - Represents a connected Unix datagram socket. - - Supports all relevant extra attributes from :class:`~SocketAttribute`. - """ - - @classmethod - async def from_socket( - cls, - sock_or_fd: socket.socket | int, - ) -> ConnectedUNIXDatagramSocket: - """ - Wrap an existing socket object or file descriptor as a connected UNIX datagram - socket. - - The newly created socket wrapper takes ownership of the socket being passed in. - The existing socket must already be connected. - - :param sock_or_fd: a socket object or file descriptor - :return: a connected UNIX datagram socket - - """ - sock = _validate_socket( - sock_or_fd, socket.SOCK_DGRAM, socket.AF_UNIX, require_connected=True - ) - return await get_async_backend().wrap_connected_unix_datagram_socket(sock) diff --git a/.env/Lib/site-packages/anyio/abc/_streams.py b/.env/Lib/site-packages/anyio/abc/_streams.py deleted file mode 100644 index 369df3f..0000000 --- a/.env/Lib/site-packages/anyio/abc/_streams.py +++ /dev/null @@ -1,239 +0,0 @@ -from __future__ import annotations - -import sys -from abc import ABCMeta, abstractmethod -from collections.abc import Callable -from typing import Any, Generic, TypeVar, Union - -from .._core._exceptions import EndOfStream -from .._core._typedattr import TypedAttributeProvider -from ._resources import AsyncResource -from ._tasks import TaskGroup - -if sys.version_info >= (3, 10): - from typing import TypeAlias -else: - from typing_extensions import TypeAlias - -T_Item = TypeVar("T_Item") -T_co = TypeVar("T_co", covariant=True) -T_contra = TypeVar("T_contra", contravariant=True) - - -class UnreliableObjectReceiveStream( - Generic[T_co], AsyncResource, TypedAttributeProvider -): - """ - An interface for receiving objects. - - This interface makes no guarantees that the received messages arrive in the order in - which they were sent, or that no messages are missed. - - Asynchronously iterating over objects of this type will yield objects matching the - given type parameter. - """ - - def __aiter__(self) -> UnreliableObjectReceiveStream[T_co]: - return self - - async def __anext__(self) -> T_co: - try: - return await self.receive() - except EndOfStream: - raise StopAsyncIteration from None - - @abstractmethod - async def receive(self) -> T_co: - """ - Receive the next item. - - :raises ~anyio.ClosedResourceError: if the receive stream has been explicitly - closed - :raises ~anyio.EndOfStream: if this stream has been closed from the other end - :raises ~anyio.BrokenResourceError: if this stream has been rendered unusable - due to external causes - """ - - -class UnreliableObjectSendStream( - Generic[T_contra], AsyncResource, TypedAttributeProvider -): - """ - An interface for sending objects. - - This interface makes no guarantees that the messages sent will reach the - recipient(s) in the same order in which they were sent, or at all. - """ - - @abstractmethod - async def send(self, item: T_contra) -> None: - """ - Send an item to the peer(s). - - :param item: the item to send - :raises ~anyio.ClosedResourceError: if the send stream has been explicitly - closed - :raises ~anyio.BrokenResourceError: if this stream has been rendered unusable - due to external causes - """ - - -class UnreliableObjectStream( - UnreliableObjectReceiveStream[T_Item], UnreliableObjectSendStream[T_Item] -): - """ - A bidirectional message stream which does not guarantee the order or reliability of - message delivery. - """ - - -class ObjectReceiveStream(UnreliableObjectReceiveStream[T_co]): - """ - A receive message stream which guarantees that messages are received in the same - order in which they were sent, and that no messages are missed. - """ - - -class ObjectSendStream(UnreliableObjectSendStream[T_contra]): - """ - A send message stream which guarantees that messages are delivered in the same order - in which they were sent, without missing any messages in the middle. - """ - - -class ObjectStream( - ObjectReceiveStream[T_Item], - ObjectSendStream[T_Item], - UnreliableObjectStream[T_Item], -): - """ - A bidirectional message stream which guarantees the order and reliability of message - delivery. - """ - - @abstractmethod - async def send_eof(self) -> None: - """ - Send an end-of-file indication to the peer. - - You should not try to send any further data to this stream after calling this - method. This method is idempotent (does nothing on successive calls). - """ - - -class ByteReceiveStream(AsyncResource, TypedAttributeProvider): - """ - An interface for receiving bytes from a single peer. - - Iterating this byte stream will yield a byte string of arbitrary length, but no more - than 65536 bytes. - """ - - def __aiter__(self) -> ByteReceiveStream: - return self - - async def __anext__(self) -> bytes: - try: - return await self.receive() - except EndOfStream: - raise StopAsyncIteration from None - - @abstractmethod - async def receive(self, max_bytes: int = 65536) -> bytes: - """ - Receive at most ``max_bytes`` bytes from the peer. - - .. note:: Implementers of this interface should not return an empty - :class:`bytes` object, and users should ignore them. - - :param max_bytes: maximum number of bytes to receive - :return: the received bytes - :raises ~anyio.EndOfStream: if this stream has been closed from the other end - """ - - -class ByteSendStream(AsyncResource, TypedAttributeProvider): - """An interface for sending bytes to a single peer.""" - - @abstractmethod - async def send(self, item: bytes) -> None: - """ - Send the given bytes to the peer. - - :param item: the bytes to send - """ - - -class ByteStream(ByteReceiveStream, ByteSendStream): - """A bidirectional byte stream.""" - - @abstractmethod - async def send_eof(self) -> None: - """ - Send an end-of-file indication to the peer. - - You should not try to send any further data to this stream after calling this - method. This method is idempotent (does nothing on successive calls). - """ - - -#: Type alias for all unreliable bytes-oriented receive streams. -AnyUnreliableByteReceiveStream: TypeAlias = Union[ - UnreliableObjectReceiveStream[bytes], ByteReceiveStream -] -#: Type alias for all unreliable bytes-oriented send streams. -AnyUnreliableByteSendStream: TypeAlias = Union[ - UnreliableObjectSendStream[bytes], ByteSendStream -] -#: Type alias for all unreliable bytes-oriented streams. -AnyUnreliableByteStream: TypeAlias = Union[UnreliableObjectStream[bytes], ByteStream] -#: Type alias for all bytes-oriented receive streams. -AnyByteReceiveStream: TypeAlias = Union[ObjectReceiveStream[bytes], ByteReceiveStream] -#: Type alias for all bytes-oriented send streams. -AnyByteSendStream: TypeAlias = Union[ObjectSendStream[bytes], ByteSendStream] -#: Type alias for all bytes-oriented streams. -AnyByteStream: TypeAlias = Union[ObjectStream[bytes], ByteStream] - - -class Listener(Generic[T_co], AsyncResource, TypedAttributeProvider): - """An interface for objects that let you accept incoming connections.""" - - @abstractmethod - async def serve( - self, handler: Callable[[T_co], Any], task_group: TaskGroup | None = None - ) -> None: - """ - Accept incoming connections as they come in and start tasks to handle them. - - :param handler: a callable that will be used to handle each accepted connection - :param task_group: the task group that will be used to start tasks for handling - each accepted connection (if omitted, an ad-hoc task group will be created) - """ - - -class ObjectStreamConnectable(Generic[T_co], metaclass=ABCMeta): - @abstractmethod - async def connect(self) -> ObjectStream[T_co]: - """ - Connect to the remote endpoint. - - :return: an object stream connected to the remote end - :raises ConnectionFailed: if the connection fails - """ - - -class ByteStreamConnectable(metaclass=ABCMeta): - @abstractmethod - async def connect(self) -> ByteStream: - """ - Connect to the remote endpoint. - - :return: a bytestream connected to the remote end - :raises ConnectionFailed: if the connection fails - """ - - -#: Type alias for all connectables returning bytestreams or bytes-oriented object streams -AnyByteStreamConnectable: TypeAlias = Union[ - ObjectStreamConnectable[bytes], ByteStreamConnectable -] diff --git a/.env/Lib/site-packages/anyio/abc/_subprocesses.py b/.env/Lib/site-packages/anyio/abc/_subprocesses.py deleted file mode 100644 index ce0564c..0000000 --- a/.env/Lib/site-packages/anyio/abc/_subprocesses.py +++ /dev/null @@ -1,79 +0,0 @@ -from __future__ import annotations - -from abc import abstractmethod -from signal import Signals - -from ._resources import AsyncResource -from ._streams import ByteReceiveStream, ByteSendStream - - -class Process(AsyncResource): - """An asynchronous version of :class:`subprocess.Popen`.""" - - @abstractmethod - async def wait(self) -> int: - """ - Wait until the process exits. - - :return: the exit code of the process - """ - - @abstractmethod - def terminate(self) -> None: - """ - Terminates the process, gracefully if possible. - - On Windows, this calls ``TerminateProcess()``. - On POSIX systems, this sends ``SIGTERM`` to the process. - - .. seealso:: :meth:`subprocess.Popen.terminate` - """ - - @abstractmethod - def kill(self) -> None: - """ - Kills the process. - - On Windows, this calls ``TerminateProcess()``. - On POSIX systems, this sends ``SIGKILL`` to the process. - - .. seealso:: :meth:`subprocess.Popen.kill` - """ - - @abstractmethod - def send_signal(self, signal: Signals) -> None: - """ - Send a signal to the subprocess. - - .. seealso:: :meth:`subprocess.Popen.send_signal` - - :param signal: the signal number (e.g. :data:`signal.SIGHUP`) - """ - - @property - @abstractmethod - def pid(self) -> int: - """The process ID of the process.""" - - @property - @abstractmethod - def returncode(self) -> int | None: - """ - The return code of the process. If the process has not yet terminated, this will - be ``None``. - """ - - @property - @abstractmethod - def stdin(self) -> ByteSendStream | None: - """The stream for the standard input of the process.""" - - @property - @abstractmethod - def stdout(self) -> ByteReceiveStream | None: - """The stream for the standard output of the process.""" - - @property - @abstractmethod - def stderr(self) -> ByteReceiveStream | None: - """The stream for the standard error output of the process.""" diff --git a/.env/Lib/site-packages/anyio/abc/_tasks.py b/.env/Lib/site-packages/anyio/abc/_tasks.py deleted file mode 100644 index 516b3ec..0000000 --- a/.env/Lib/site-packages/anyio/abc/_tasks.py +++ /dev/null @@ -1,117 +0,0 @@ -from __future__ import annotations - -import sys -from abc import ABCMeta, abstractmethod -from collections.abc import Awaitable, Callable -from types import TracebackType -from typing import TYPE_CHECKING, Any, Protocol, overload - -if sys.version_info >= (3, 13): - from typing import TypeVar -else: - from typing_extensions import TypeVar - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -if TYPE_CHECKING: - from .._core._tasks import CancelScope - -T_Retval = TypeVar("T_Retval") -T_contra = TypeVar("T_contra", contravariant=True, default=None) -PosArgsT = TypeVarTuple("PosArgsT") - - -class TaskStatus(Protocol[T_contra]): - @overload - def started(self: TaskStatus[None]) -> None: ... - - @overload - def started(self, value: T_contra) -> None: ... - - def started(self, value: T_contra | None = None) -> None: - """ - Signal that the task has started. - - :param value: object passed back to the starter of the task - """ - - -class TaskGroup(metaclass=ABCMeta): - """ - Groups several asynchronous tasks together. - - :ivar cancel_scope: the cancel scope inherited by all child tasks - :vartype cancel_scope: CancelScope - - .. note:: On asyncio, support for eager task factories is considered to be - **experimental**. In particular, they don't follow the usual semantics of new - tasks being scheduled on the next iteration of the event loop, and may thus - cause unexpected behavior in code that wasn't written with such semantics in - mind. - """ - - cancel_scope: CancelScope - - @abstractmethod - def start_soon( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[Any]], - *args: Unpack[PosArgsT], - name: object = None, - ) -> None: - """ - Start a new task in this task group. - - :param func: a coroutine function - :param args: positional arguments to call the function with - :param name: name of the task, for the purposes of introspection and debugging - - .. versionadded:: 3.0 - """ - - @abstractmethod - async def start( - self, - func: Callable[..., Awaitable[Any]], - *args: object, - name: object = None, - ) -> Any: - """ - Start a new task and wait until it signals for readiness. - - The target callable must accept a keyword argument ``task_status`` (of type - :class:`TaskStatus`). Awaiting on this method will return whatever was passed to - ``task_status.started()`` (``None`` by default). - - .. note:: The :class:`TaskStatus` class is generic, and the type argument should - indicate the type of the value that will be passed to - ``task_status.started()``. - - :param func: a coroutine function that accepts the ``task_status`` keyword - argument - :param args: positional arguments to call the function with - :param name: an optional name for the task, for introspection and debugging - :return: the value passed to ``task_status.started()`` - :raises RuntimeError: if the task finishes without calling - ``task_status.started()`` - - .. seealso:: :ref:`start_initialize` - - .. versionadded:: 3.0 - """ - - @abstractmethod - async def __aenter__(self) -> TaskGroup: - """Enter the task group context and allow starting new tasks.""" - - @abstractmethod - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - """Exit the task group context waiting for all tasks to finish.""" diff --git a/.env/Lib/site-packages/anyio/abc/_testing.py b/.env/Lib/site-packages/anyio/abc/_testing.py deleted file mode 100644 index 7c50ed7..0000000 --- a/.env/Lib/site-packages/anyio/abc/_testing.py +++ /dev/null @@ -1,65 +0,0 @@ -from __future__ import annotations - -import types -from abc import ABCMeta, abstractmethod -from collections.abc import AsyncGenerator, Callable, Coroutine, Iterable -from typing import Any, TypeVar - -_T = TypeVar("_T") - - -class TestRunner(metaclass=ABCMeta): - """ - Encapsulates a running event loop. Every call made through this object will use the - same event loop. - """ - - def __enter__(self) -> TestRunner: - return self - - @abstractmethod - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: types.TracebackType | None, - ) -> bool | None: ... - - @abstractmethod - def run_asyncgen_fixture( - self, - fixture_func: Callable[..., AsyncGenerator[_T, Any]], - kwargs: dict[str, Any], - ) -> Iterable[_T]: - """ - Run an async generator fixture. - - :param fixture_func: the fixture function - :param kwargs: keyword arguments to call the fixture function with - :return: an iterator yielding the value yielded from the async generator - """ - - @abstractmethod - def run_fixture( - self, - fixture_func: Callable[..., Coroutine[Any, Any, _T]], - kwargs: dict[str, Any], - ) -> _T: - """ - Run an async fixture. - - :param fixture_func: the fixture function - :param kwargs: keyword arguments to call the fixture function with - :return: the return value of the fixture function - """ - - @abstractmethod - def run_test( - self, test_func: Callable[..., Coroutine[Any, Any, Any]], kwargs: dict[str, Any] - ) -> None: - """ - Run an async test function. - - :param test_func: the test function - :param kwargs: keyword arguments to call the test function with - """ diff --git a/.env/Lib/site-packages/anyio/from_thread.py b/.env/Lib/site-packages/anyio/from_thread.py deleted file mode 100644 index 837de5e..0000000 --- a/.env/Lib/site-packages/anyio/from_thread.py +++ /dev/null @@ -1,578 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "BlockingPortal", - "BlockingPortalProvider", - "check_cancelled", - "run", - "run_sync", - "start_blocking_portal", -) - -import sys -from collections.abc import Awaitable, Callable, Generator -from concurrent.futures import Future -from contextlib import ( - AbstractAsyncContextManager, - AbstractContextManager, - contextmanager, -) -from dataclasses import dataclass, field -from functools import partial -from inspect import isawaitable -from threading import Lock, Thread, current_thread, get_ident -from types import TracebackType -from typing import ( - Any, - Generic, - TypeVar, - cast, - overload, -) - -from ._core._eventloop import ( - get_cancelled_exc_class, - threadlocals, -) -from ._core._eventloop import run as run_eventloop -from ._core._exceptions import NoEventLoopError -from ._core._synchronization import Event -from ._core._tasks import CancelScope, create_task_group -from .abc._tasks import TaskStatus -from .lowlevel import EventLoopToken, current_token - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -T_Retval = TypeVar("T_Retval") -T_co = TypeVar("T_co", covariant=True) -PosArgsT = TypeVarTuple("PosArgsT") - - -def _token_or_error(token: EventLoopToken | None) -> EventLoopToken: - if token is not None: - return token - - try: - return threadlocals.current_token - except AttributeError: - raise NoEventLoopError( - "Not running inside an AnyIO worker thread, and no event loop token was " - "provided" - ) from None - - -def run( - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - *args: Unpack[PosArgsT], - token: EventLoopToken | None = None, -) -> T_Retval: - """ - Call a coroutine function from a worker thread. - - :param func: a coroutine function - :param args: positional arguments for the callable - :param token: an event loop token to use to get back to the event loop thread - (required if calling this function from outside an AnyIO worker thread) - :return: the return value of the coroutine function - :raises MissingTokenError: if no token was provided and called from outside an - AnyIO worker thread - :raises RunFinishedError: if the event loop tied to ``token`` is no longer running - - .. versionchanged:: 4.11.0 - Added the ``token`` parameter. - - """ - explicit_token = token is not None - token = _token_or_error(token) - return token.backend_class.run_async_from_thread( - func, args, token=token.native_token if explicit_token else None - ) - - -def run_sync( - func: Callable[[Unpack[PosArgsT]], T_Retval], - *args: Unpack[PosArgsT], - token: EventLoopToken | None = None, -) -> T_Retval: - """ - Call a function in the event loop thread from a worker thread. - - :param func: a callable - :param args: positional arguments for the callable - :param token: an event loop token to use to get back to the event loop thread - (required if calling this function from outside an AnyIO worker thread) - :return: the return value of the callable - :raises MissingTokenError: if no token was provided and called from outside an - AnyIO worker thread - :raises RunFinishedError: if the event loop tied to ``token`` is no longer running - - .. versionchanged:: 4.11.0 - Added the ``token`` parameter. - - """ - explicit_token = token is not None - token = _token_or_error(token) - return token.backend_class.run_sync_from_thread( - func, args, token=token.native_token if explicit_token else None - ) - - -class _BlockingAsyncContextManager(Generic[T_co], AbstractContextManager): - _enter_future: Future[T_co] - _exit_future: Future[bool | None] - _exit_event: Event - _exit_exc_info: tuple[ - type[BaseException] | None, BaseException | None, TracebackType | None - ] = (None, None, None) - - def __init__( - self, async_cm: AbstractAsyncContextManager[T_co], portal: BlockingPortal - ): - self._async_cm = async_cm - self._portal = portal - - async def run_async_cm(self) -> bool | None: - try: - self._exit_event = Event() - value = await self._async_cm.__aenter__() - except BaseException as exc: - self._enter_future.set_exception(exc) - raise - else: - self._enter_future.set_result(value) - - try: - # Wait for the sync context manager to exit. - # This next statement can raise `get_cancelled_exc_class()` if - # something went wrong in a task group in this async context - # manager. - await self._exit_event.wait() - finally: - # In case of cancellation, it could be that we end up here before - # `_BlockingAsyncContextManager.__exit__` is called, and an - # `_exit_exc_info` has been set. - result = await self._async_cm.__aexit__(*self._exit_exc_info) - - return result - - def __enter__(self) -> T_co: - self._enter_future = Future() - self._exit_future = self._portal.start_task_soon(self.run_async_cm) - return self._enter_future.result() - - def __exit__( - self, - __exc_type: type[BaseException] | None, - __exc_value: BaseException | None, - __traceback: TracebackType | None, - ) -> bool | None: - self._exit_exc_info = __exc_type, __exc_value, __traceback - self._portal.call(self._exit_event.set) - return self._exit_future.result() - - -class _BlockingPortalTaskStatus(TaskStatus): - def __init__(self, future: Future): - self._future = future - - def started(self, value: object = None) -> None: - self._future.set_result(value) - - -class BlockingPortal: - """ - An object that lets external threads run code in an asynchronous event loop. - - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - """ - - def __init__(self) -> None: - self._token = current_token() - self._event_loop_thread_id: int | None = get_ident() - self._stop_event = Event() - self._task_group = create_task_group() - - async def __aenter__(self) -> BlockingPortal: - await self._task_group.__aenter__() - return self - - async def __aexit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> bool: - await self.stop() - return await self._task_group.__aexit__(exc_type, exc_val, exc_tb) - - def _check_running(self) -> None: - if self._event_loop_thread_id is None: - raise RuntimeError("This portal is not running") - if self._event_loop_thread_id == get_ident(): - raise RuntimeError( - "This method cannot be called from the event loop thread" - ) - - async def sleep_until_stopped(self) -> None: - """Sleep until :meth:`stop` is called.""" - await self._stop_event.wait() - - async def stop(self, cancel_remaining: bool = False) -> None: - """ - Signal the portal to shut down. - - This marks the portal as no longer accepting new calls and exits from - :meth:`sleep_until_stopped`. - - :param cancel_remaining: ``True`` to cancel all the remaining tasks, ``False`` - to let them finish before returning - - """ - self._event_loop_thread_id = None - self._stop_event.set() - if cancel_remaining: - self._task_group.cancel_scope.cancel("the blocking portal is shutting down") - - async def _call_func( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval] | T_Retval], - args: tuple[Unpack[PosArgsT]], - kwargs: dict[str, Any], - future: Future[T_Retval], - ) -> None: - def callback(f: Future[T_Retval]) -> None: - if f.cancelled(): - if self._event_loop_thread_id == get_ident(): - scope.cancel("the future was cancelled") - elif self._event_loop_thread_id is not None: - self.call(scope.cancel, "the future was cancelled") - - try: - retval_or_awaitable = func(*args, **kwargs) - if isawaitable(retval_or_awaitable): - with CancelScope() as scope: - future.add_done_callback(callback) - retval = await retval_or_awaitable - else: - retval = retval_or_awaitable - except get_cancelled_exc_class(): - future.cancel() - future.set_running_or_notify_cancel() - except BaseException as exc: - if not future.cancelled(): - future.set_exception(exc) - - # Let base exceptions fall through - if not isinstance(exc, Exception): - raise - else: - if not future.cancelled(): - future.set_result(retval) - finally: - scope = None # type: ignore[assignment] - - def _spawn_task_from_thread( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval] | T_Retval], - args: tuple[Unpack[PosArgsT]], - kwargs: dict[str, Any], - name: object, - future: Future[T_Retval], - ) -> None: - """ - Spawn a new task using the given callable. - - :param func: a callable - :param args: positional arguments to be passed to the callable - :param kwargs: keyword arguments to be passed to the callable - :param name: name of the task (will be coerced to a string if not ``None``) - :param future: a future that will resolve to the return value of the callable, - or the exception raised during its execution - - """ - run_sync( - partial(self._task_group.start_soon, name=name), - self._call_func, - func, - args, - kwargs, - future, - token=self._token, - ) - - @overload - def call( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - *args: Unpack[PosArgsT], - ) -> T_Retval: ... - - @overload - def call( - self, func: Callable[[Unpack[PosArgsT]], T_Retval], *args: Unpack[PosArgsT] - ) -> T_Retval: ... - - def call( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval] | T_Retval], - *args: Unpack[PosArgsT], - ) -> T_Retval: - """ - Call the given function in the event loop thread. - - If the callable returns a coroutine object, it is awaited on. - - :param func: any callable - :raises RuntimeError: if the portal is not running or if this method is called - from within the event loop thread - - """ - return cast(T_Retval, self.start_task_soon(func, *args).result()) - - @overload - def start_task_soon( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval]], - *args: Unpack[PosArgsT], - name: object = None, - ) -> Future[T_Retval]: ... - - @overload - def start_task_soon( - self, - func: Callable[[Unpack[PosArgsT]], T_Retval], - *args: Unpack[PosArgsT], - name: object = None, - ) -> Future[T_Retval]: ... - - def start_task_soon( - self, - func: Callable[[Unpack[PosArgsT]], Awaitable[T_Retval] | T_Retval], - *args: Unpack[PosArgsT], - name: object = None, - ) -> Future[T_Retval]: - """ - Start a task in the portal's task group. - - The task will be run inside a cancel scope which can be cancelled by cancelling - the returned future. - - :param func: the target function - :param args: positional arguments passed to ``func`` - :param name: name of the task (will be coerced to a string if not ``None``) - :return: a future that resolves with the return value of the callable if the - task completes successfully, or with the exception raised in the task - :raises RuntimeError: if the portal is not running or if this method is called - from within the event loop thread - :rtype: concurrent.futures.Future[T_Retval] - - .. versionadded:: 3.0 - - """ - self._check_running() - f: Future[T_Retval] = Future() - self._spawn_task_from_thread(func, args, {}, name, f) - return f - - def start_task( - self, - func: Callable[..., Awaitable[T_Retval]], - *args: object, - name: object = None, - ) -> tuple[Future[T_Retval], Any]: - """ - Start a task in the portal's task group and wait until it signals for readiness. - - This method works the same way as :meth:`.abc.TaskGroup.start`. - - :param func: the target function - :param args: positional arguments passed to ``func`` - :param name: name of the task (will be coerced to a string if not ``None``) - :return: a tuple of (future, task_status_value) where the ``task_status_value`` - is the value passed to ``task_status.started()`` from within the target - function - :rtype: tuple[concurrent.futures.Future[T_Retval], Any] - - .. versionadded:: 3.0 - - """ - - def task_done(future: Future[T_Retval]) -> None: - if not task_status_future.done(): - if future.cancelled(): - task_status_future.cancel() - elif future.exception(): - task_status_future.set_exception(future.exception()) - else: - exc = RuntimeError( - "Task exited without calling task_status.started()" - ) - task_status_future.set_exception(exc) - - self._check_running() - task_status_future: Future = Future() - task_status = _BlockingPortalTaskStatus(task_status_future) - f: Future = Future() - f.add_done_callback(task_done) - self._spawn_task_from_thread(func, args, {"task_status": task_status}, name, f) - return f, task_status_future.result() - - def wrap_async_context_manager( - self, cm: AbstractAsyncContextManager[T_co] - ) -> AbstractContextManager[T_co]: - """ - Wrap an async context manager as a synchronous context manager via this portal. - - Spawns a task that will call both ``__aenter__()`` and ``__aexit__()``, stopping - in the middle until the synchronous context manager exits. - - :param cm: an asynchronous context manager - :return: a synchronous context manager - - .. versionadded:: 2.1 - - """ - return _BlockingAsyncContextManager(cm, self) - - -@dataclass -class BlockingPortalProvider: - """ - A manager for a blocking portal. Used as a context manager. The first thread to - enter this context manager causes a blocking portal to be started with the specific - parameters, and the last thread to exit causes the portal to be shut down. Thus, - there will be exactly one blocking portal running in this context as long as at - least one thread has entered this context manager. - - The parameters are the same as for :func:`~anyio.run`. - - :param backend: name of the backend - :param backend_options: backend options - - .. versionadded:: 4.4 - """ - - backend: str = "asyncio" - backend_options: dict[str, Any] | None = None - _lock: Lock = field(init=False, default_factory=Lock) - _leases: int = field(init=False, default=0) - _portal: BlockingPortal = field(init=False) - _portal_cm: AbstractContextManager[BlockingPortal] | None = field( - init=False, default=None - ) - - def __enter__(self) -> BlockingPortal: - with self._lock: - if self._portal_cm is None: - self._portal_cm = start_blocking_portal( - self.backend, self.backend_options - ) - self._portal = self._portal_cm.__enter__() - - self._leases += 1 - return self._portal - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - portal_cm: AbstractContextManager[BlockingPortal] | None = None - with self._lock: - assert self._portal_cm - assert self._leases > 0 - self._leases -= 1 - if not self._leases: - portal_cm = self._portal_cm - self._portal_cm = None - del self._portal - - if portal_cm: - portal_cm.__exit__(None, None, None) - - -@contextmanager -def start_blocking_portal( - backend: str = "asyncio", - backend_options: dict[str, Any] | None = None, - *, - name: str | None = None, -) -> Generator[BlockingPortal, Any, None]: - """ - Start a new event loop in a new thread and run a blocking portal in its main task. - - The parameters are the same as for :func:`~anyio.run`. - - :param backend: name of the backend - :param backend_options: backend options - :param name: name of the thread - :return: a context manager that yields a blocking portal - - .. versionchanged:: 3.0 - Usage as a context manager is now required. - - """ - - async def run_portal() -> None: - async with BlockingPortal() as portal_: - if name is None: - current_thread().name = f"{backend}-portal-{id(portal_):x}" - - future.set_result(portal_) - await portal_.sleep_until_stopped() - - def run_blocking_portal() -> None: - if future.set_running_or_notify_cancel(): - try: - run_eventloop( - run_portal, backend=backend, backend_options=backend_options - ) - except BaseException as exc: - if not future.done(): - future.set_exception(exc) - - future: Future[BlockingPortal] = Future() - thread = Thread(target=run_blocking_portal, daemon=True, name=name) - thread.start() - try: - cancel_remaining_tasks = False - portal = future.result() - try: - yield portal - except BaseException: - cancel_remaining_tasks = True - raise - finally: - try: - portal.call(portal.stop, cancel_remaining_tasks) - except RuntimeError: - pass - finally: - thread.join() - - -def check_cancelled() -> None: - """ - Check if the cancel scope of the host task's running the current worker thread has - been cancelled. - - If the host task's current cancel scope has indeed been cancelled, the - backend-specific cancellation exception will be raised. - - :raises RuntimeError: if the current thread was not spawned by - :func:`.to_thread.run_sync` - - """ - try: - token: EventLoopToken = threadlocals.current_token - except AttributeError: - raise NoEventLoopError( - "This function can only be called inside an AnyIO worker thread" - ) from None - - token.backend_class.check_cancelled() diff --git a/.env/Lib/site-packages/anyio/functools.py b/.env/Lib/site-packages/anyio/functools.py deleted file mode 100644 index b80afe6..0000000 --- a/.env/Lib/site-packages/anyio/functools.py +++ /dev/null @@ -1,375 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "AsyncCacheInfo", - "AsyncCacheParameters", - "AsyncLRUCacheWrapper", - "cache", - "lru_cache", - "reduce", -) - -import functools -import sys -from collections import OrderedDict -from collections.abc import ( - AsyncIterable, - Awaitable, - Callable, - Coroutine, - Hashable, - Iterable, -) -from functools import update_wrapper -from inspect import iscoroutinefunction -from typing import ( - Any, - Generic, - NamedTuple, - TypedDict, - TypeVar, - cast, - final, - overload, -) -from weakref import WeakKeyDictionary - -from ._core._synchronization import Lock -from .lowlevel import RunVar, checkpoint - -if sys.version_info >= (3, 11): - from typing import ParamSpec -else: - from typing_extensions import ParamSpec - -T = TypeVar("T") -S = TypeVar("S") -P = ParamSpec("P") -lru_cache_items: RunVar[ - WeakKeyDictionary[ - AsyncLRUCacheWrapper[Any, Any], - OrderedDict[Hashable, tuple[_InitialMissingType, Lock] | tuple[Any, None]], - ] -] = RunVar("lru_cache_items") - - -class _InitialMissingType: - pass - - -initial_missing: _InitialMissingType = _InitialMissingType() - - -class AsyncCacheInfo(NamedTuple): - hits: int - misses: int - maxsize: int | None - currsize: int - - -class AsyncCacheParameters(TypedDict): - maxsize: int | None - typed: bool - always_checkpoint: bool - - -class _LRUMethodWrapper(Generic[T]): - def __init__(self, wrapper: AsyncLRUCacheWrapper[..., T], instance: object): - self.__wrapper = wrapper - self.__instance = instance - - def cache_info(self) -> AsyncCacheInfo: - return self.__wrapper.cache_info() - - def cache_parameters(self) -> AsyncCacheParameters: - return self.__wrapper.cache_parameters() - - def cache_clear(self) -> None: - self.__wrapper.cache_clear() - - async def __call__(self, *args: Any, **kwargs: Any) -> T: - if self.__instance is None: - return await self.__wrapper(*args, **kwargs) - - return await self.__wrapper(self.__instance, *args, **kwargs) - - -@final -class AsyncLRUCacheWrapper(Generic[P, T]): - def __init__( - self, - func: Callable[P, Awaitable[T]], - maxsize: int | None, - typed: bool, - always_checkpoint: bool, - ): - self.__wrapped__ = func - self._hits: int = 0 - self._misses: int = 0 - self._maxsize = max(maxsize, 0) if maxsize is not None else None - self._currsize: int = 0 - self._typed = typed - self._always_checkpoint = always_checkpoint - update_wrapper(self, func) - - def cache_info(self) -> AsyncCacheInfo: - return AsyncCacheInfo(self._hits, self._misses, self._maxsize, self._currsize) - - def cache_parameters(self) -> AsyncCacheParameters: - return { - "maxsize": self._maxsize, - "typed": self._typed, - "always_checkpoint": self._always_checkpoint, - } - - def cache_clear(self) -> None: - if cache := lru_cache_items.get(None): - cache.pop(self, None) - self._hits = self._misses = self._currsize = 0 - - async def __call__(self, *args: P.args, **kwargs: P.kwargs) -> T: - # Easy case first: if maxsize == 0, no caching is done - if self._maxsize == 0: - value = await self.__wrapped__(*args, **kwargs) - self._misses += 1 - return value - - # The key is constructed as a flat tuple to avoid memory overhead - key: tuple[Any, ...] = args - if kwargs: - # initial_missing is used as a separator - key += (initial_missing,) + sum(kwargs.items(), ()) - - if self._typed: - key += tuple(type(arg) for arg in args) - if kwargs: - key += (initial_missing,) + tuple(type(val) for val in kwargs.values()) - - try: - cache = lru_cache_items.get() - except LookupError: - cache = WeakKeyDictionary() - lru_cache_items.set(cache) - - try: - cache_entry = cache[self] - except KeyError: - cache_entry = cache[self] = OrderedDict() - - cached_value: T | _InitialMissingType - try: - cached_value, lock = cache_entry[key] - except KeyError: - # We're the first task to call this function - cached_value, lock = ( - initial_missing, - Lock(fast_acquire=not self._always_checkpoint), - ) - cache_entry[key] = cached_value, lock - - if lock is None: - # The value was already cached - self._hits += 1 - cache_entry.move_to_end(key) - if self._always_checkpoint: - await checkpoint() - - return cast(T, cached_value) - - async with lock: - # Check if another task filled the cache while we acquired the lock - if (cached_value := cache_entry[key][0]) is initial_missing: - self._misses += 1 - if self._maxsize is not None and self._currsize >= self._maxsize: - cache_entry.popitem(last=False) - else: - self._currsize += 1 - - value = await self.__wrapped__(*args, **kwargs) - cache_entry[key] = value, None - else: - # Another task filled the cache while we were waiting for the lock - self._hits += 1 - cache_entry.move_to_end(key) - value = cast(T, cached_value) - - return value - - def __get__( - self, instance: object, owner: type | None = None - ) -> _LRUMethodWrapper[T]: - wrapper = _LRUMethodWrapper(self, instance) - update_wrapper(wrapper, self.__wrapped__) - return wrapper - - -class _LRUCacheWrapper(Generic[T]): - def __init__(self, maxsize: int | None, typed: bool, always_checkpoint: bool): - self._maxsize = maxsize - self._typed = typed - self._always_checkpoint = always_checkpoint - - @overload - def __call__( # type: ignore[overload-overlap] - self, func: Callable[P, Coroutine[Any, Any, T]], / - ) -> AsyncLRUCacheWrapper[P, T]: ... - - @overload - def __call__( - self, func: Callable[..., T], / - ) -> functools._lru_cache_wrapper[T]: ... - - def __call__( - self, f: Callable[P, Coroutine[Any, Any, T]] | Callable[..., T], / - ) -> AsyncLRUCacheWrapper[P, T] | functools._lru_cache_wrapper[T]: - if iscoroutinefunction(f): - return AsyncLRUCacheWrapper( - f, self._maxsize, self._typed, self._always_checkpoint - ) - - return functools.lru_cache(maxsize=self._maxsize, typed=self._typed)(f) # type: ignore[arg-type] - - -@overload -def cache( # type: ignore[overload-overlap] - func: Callable[P, Coroutine[Any, Any, T]], / -) -> AsyncLRUCacheWrapper[P, T]: ... - - -@overload -def cache(func: Callable[..., T], /) -> functools._lru_cache_wrapper[T]: ... - - -def cache( - func: Callable[..., T] | Callable[P, Coroutine[Any, Any, T]], / -) -> AsyncLRUCacheWrapper[P, T] | functools._lru_cache_wrapper[T]: - """ - A convenient shortcut for :func:`lru_cache` with ``maxsize=None``. - - This is the asynchronous equivalent to :func:`functools.cache`. - - """ - return lru_cache(maxsize=None)(func) - - -@overload -def lru_cache( - *, maxsize: int | None = ..., typed: bool = ..., always_checkpoint: bool = ... -) -> _LRUCacheWrapper[Any]: ... - - -@overload -def lru_cache( # type: ignore[overload-overlap] - func: Callable[P, Coroutine[Any, Any, T]], / -) -> AsyncLRUCacheWrapper[P, T]: ... - - -@overload -def lru_cache(func: Callable[..., T], /) -> functools._lru_cache_wrapper[T]: ... - - -def lru_cache( - func: Callable[P, Coroutine[Any, Any, T]] | Callable[..., T] | None = None, - /, - *, - maxsize: int | None = 128, - typed: bool = False, - always_checkpoint: bool = False, -) -> ( - AsyncLRUCacheWrapper[P, T] | functools._lru_cache_wrapper[T] | _LRUCacheWrapper[Any] -): - """ - An asynchronous version of :func:`functools.lru_cache`. - - If a synchronous function is passed, the standard library - :func:`functools.lru_cache` is applied instead. - - :param always_checkpoint: if ``True``, every call to the cached function will be - guaranteed to yield control to the event loop at least once - - .. note:: Caches and locks are managed on a per-event loop basis. - - """ - if func is None: - return _LRUCacheWrapper[Any](maxsize, typed, always_checkpoint) - - if not callable(func): - raise TypeError("the first argument must be callable") - - return _LRUCacheWrapper[T](maxsize, typed, always_checkpoint)(func) - - -@overload -async def reduce( - function: Callable[[T, S], Awaitable[T]], - iterable: Iterable[S] | AsyncIterable[S], - /, - initial: T, -) -> T: ... - - -@overload -async def reduce( - function: Callable[[T, T], Awaitable[T]], - iterable: Iterable[T] | AsyncIterable[T], - /, -) -> T: ... - - -async def reduce( # type: ignore[misc] - function: Callable[[T, T], Awaitable[T]] | Callable[[T, S], Awaitable[T]], - iterable: Iterable[T] | Iterable[S] | AsyncIterable[T] | AsyncIterable[S], - /, - initial: T | _InitialMissingType = initial_missing, -) -> T: - """ - Asynchronous version of :func:`functools.reduce`. - - :param function: a coroutine function that takes two arguments: the accumulated - value and the next element from the iterable - :param iterable: an iterable or async iterable - :param initial: the initial value (if missing, the first element of the iterable is - used as the initial value) - - """ - element: Any - function_called = False - if isinstance(iterable, AsyncIterable): - async_it = iterable.__aiter__() - if initial is initial_missing: - try: - value = cast(T, await async_it.__anext__()) - except StopAsyncIteration: - raise TypeError( - "reduce() of empty sequence with no initial value" - ) from None - else: - value = cast(T, initial) - - async for element in async_it: - value = await function(value, element) - function_called = True - elif isinstance(iterable, Iterable): - it = iter(iterable) - if initial is initial_missing: - try: - value = cast(T, next(it)) - except StopIteration: - raise TypeError( - "reduce() of empty sequence with no initial value" - ) from None - else: - value = cast(T, initial) - - for element in it: - value = await function(value, element) - function_called = True - else: - raise TypeError("reduce() argument 2 must be an iterable or async iterable") - - # Make sure there is at least one checkpoint, even if an empty iterable and an - # initial value were given - if not function_called: - await checkpoint() - - return value diff --git a/.env/Lib/site-packages/anyio/lowlevel.py b/.env/Lib/site-packages/anyio/lowlevel.py deleted file mode 100644 index ffbb75a..0000000 --- a/.env/Lib/site-packages/anyio/lowlevel.py +++ /dev/null @@ -1,196 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "EventLoopToken", - "RunvarToken", - "RunVar", - "checkpoint", - "checkpoint_if_cancelled", - "cancel_shielded_checkpoint", - "current_token", -) - -import enum -from dataclasses import dataclass -from types import TracebackType -from typing import Any, Generic, Literal, TypeVar, final, overload -from weakref import WeakKeyDictionary - -from ._core._eventloop import get_async_backend -from .abc import AsyncBackend - -T = TypeVar("T") -D = TypeVar("D") - - -async def checkpoint() -> None: - """ - Check for cancellation and allow the scheduler to switch to another task. - - Equivalent to (but more efficient than):: - - await checkpoint_if_cancelled() - await cancel_shielded_checkpoint() - - .. versionadded:: 3.0 - - """ - await get_async_backend().checkpoint() - - -async def checkpoint_if_cancelled() -> None: - """ - Enter a checkpoint if the enclosing cancel scope has been cancelled. - - This does not allow the scheduler to switch to a different task. - - .. versionadded:: 3.0 - - """ - await get_async_backend().checkpoint_if_cancelled() - - -async def cancel_shielded_checkpoint() -> None: - """ - Allow the scheduler to switch to another task but without checking for cancellation. - - Equivalent to (but potentially more efficient than):: - - with CancelScope(shield=True): - await checkpoint() - - .. versionadded:: 3.0 - - """ - await get_async_backend().cancel_shielded_checkpoint() - - -@final -@dataclass(frozen=True, repr=False) -class EventLoopToken: - """ - An opaque object that holds a reference to an event loop. - - .. versionadded:: 4.11.0 - """ - - backend_class: type[AsyncBackend] - native_token: object - - -def current_token() -> EventLoopToken: - """ - Return a token object that can be used to call code in the current event loop from - another thread. - - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - .. versionadded:: 4.11.0 - - """ - backend_class = get_async_backend() - raw_token = backend_class.current_token() - return EventLoopToken(backend_class, raw_token) - - -_run_vars: WeakKeyDictionary[object, dict[RunVar[Any], Any]] = WeakKeyDictionary() - - -class _NoValueSet(enum.Enum): - NO_VALUE_SET = enum.auto() - - -class RunvarToken(Generic[T]): - __slots__ = "_var", "_value", "_redeemed" - - def __init__(self, var: RunVar[T], value: T | Literal[_NoValueSet.NO_VALUE_SET]): - self._var = var - self._value: T | Literal[_NoValueSet.NO_VALUE_SET] = value - self._redeemed = False - - def __enter__(self) -> RunvarToken[T]: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self._var.reset(self) - - -class RunVar(Generic[T]): - """ - Like a :class:`~contextvars.ContextVar`, except scoped to the running event loop. - - Can be used as a context manager, Just like :class:`~contextvars.ContextVar`, that - will reset the variable to its previous value when the context block is exited. - """ - - __slots__ = "_name", "_default" - - NO_VALUE_SET: Literal[_NoValueSet.NO_VALUE_SET] = _NoValueSet.NO_VALUE_SET - - def __init__( - self, name: str, default: T | Literal[_NoValueSet.NO_VALUE_SET] = NO_VALUE_SET - ): - self._name = name - self._default = default - - @property - def _current_vars(self) -> dict[RunVar[T], T]: - native_token = current_token().native_token - try: - return _run_vars[native_token] - except KeyError: - run_vars = _run_vars[native_token] = {} - return run_vars - - @overload - def get(self, default: D) -> T | D: ... - - @overload - def get(self) -> T: ... - - def get( - self, default: D | Literal[_NoValueSet.NO_VALUE_SET] = NO_VALUE_SET - ) -> T | D: - try: - return self._current_vars[self] - except KeyError: - if default is not RunVar.NO_VALUE_SET: - return default - elif self._default is not RunVar.NO_VALUE_SET: - return self._default - - raise LookupError( - f'Run variable "{self._name}" has no value and no default set' - ) - - def set(self, value: T) -> RunvarToken[T]: - current_vars = self._current_vars - token = RunvarToken(self, current_vars.get(self, RunVar.NO_VALUE_SET)) - current_vars[self] = value - return token - - def reset(self, token: RunvarToken[T]) -> None: - if token._var is not self: - raise ValueError("This token does not belong to this RunVar") - - if token._redeemed: - raise ValueError("This token has already been used") - - if token._value is _NoValueSet.NO_VALUE_SET: - try: - del self._current_vars[self] - except KeyError: - pass - else: - self._current_vars[self] = token._value - - token._redeemed = True - - def __repr__(self) -> str: - return f"" diff --git a/.env/Lib/site-packages/anyio/py.typed b/.env/Lib/site-packages/anyio/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/anyio/pytest_plugin.py b/.env/Lib/site-packages/anyio/pytest_plugin.py deleted file mode 100644 index 4222816..0000000 --- a/.env/Lib/site-packages/anyio/pytest_plugin.py +++ /dev/null @@ -1,302 +0,0 @@ -from __future__ import annotations - -import socket -import sys -from collections.abc import Callable, Generator, Iterator -from contextlib import ExitStack, contextmanager -from inspect import isasyncgenfunction, iscoroutinefunction, ismethod -from typing import Any, cast - -import pytest -from _pytest.fixtures import SubRequest -from _pytest.outcomes import Exit - -from . import get_available_backends -from ._core._eventloop import ( - current_async_library, - get_async_backend, - reset_current_async_library, - set_current_async_library, -) -from ._core._exceptions import iterate_exceptions -from .abc import TestRunner - -if sys.version_info < (3, 11): - from exceptiongroup import ExceptionGroup - -_current_runner: TestRunner | None = None -_runner_stack: ExitStack | None = None -_runner_leases = 0 - - -def extract_backend_and_options(backend: object) -> tuple[str, dict[str, Any]]: - if isinstance(backend, str): - return backend, {} - elif isinstance(backend, tuple) and len(backend) == 2: - if isinstance(backend[0], str) and isinstance(backend[1], dict): - return cast(tuple[str, dict[str, Any]], backend) - - raise TypeError("anyio_backend must be either a string or tuple of (string, dict)") - - -@contextmanager -def get_runner( - backend_name: str, backend_options: dict[str, Any] -) -> Iterator[TestRunner]: - global _current_runner, _runner_leases, _runner_stack - if _current_runner is None: - asynclib = get_async_backend(backend_name) - _runner_stack = ExitStack() - if current_async_library() is None: - # Since we're in control of the event loop, we can cache the name of the - # async library - token = set_current_async_library(backend_name) - _runner_stack.callback(reset_current_async_library, token) - - backend_options = backend_options or {} - _current_runner = _runner_stack.enter_context( - asynclib.create_test_runner(backend_options) - ) - - _runner_leases += 1 - try: - yield _current_runner - finally: - _runner_leases -= 1 - if not _runner_leases: - assert _runner_stack is not None - _runner_stack.close() - _runner_stack = _current_runner = None - - -def pytest_addoption(parser: pytest.Parser) -> None: - parser.addini( - "anyio_mode", - default="strict", - help='AnyIO plugin mode (either "strict" or "auto")', - ) - - -def pytest_configure(config: pytest.Config) -> None: - config.addinivalue_line( - "markers", - "anyio: mark the (coroutine function) test to be run asynchronously via anyio.", - ) - if ( - config.getini("anyio_mode") == "auto" - and config.pluginmanager.has_plugin("asyncio") - and config.getini("asyncio_mode") == "auto" - ): - config.issue_config_time_warning( - pytest.PytestConfigWarning( - "AnyIO auto mode has been enabled together with pytest-asyncio auto " - "mode. This may cause unexpected behavior." - ), - 1, - ) - - -@pytest.hookimpl(hookwrapper=True) -def pytest_fixture_setup(fixturedef: Any, request: Any) -> Generator[Any]: - def wrapper(anyio_backend: Any, request: SubRequest, **kwargs: Any) -> Any: - # Rebind any fixture methods to the request instance - if ( - request.instance - and ismethod(func) - and type(func.__self__) is type(request.instance) - ): - local_func = func.__func__.__get__(request.instance) - else: - local_func = func - - backend_name, backend_options = extract_backend_and_options(anyio_backend) - if has_backend_arg: - kwargs["anyio_backend"] = anyio_backend - - if has_request_arg: - kwargs["request"] = request - - with get_runner(backend_name, backend_options) as runner: - if isasyncgenfunction(local_func): - yield from runner.run_asyncgen_fixture(local_func, kwargs) - else: - yield runner.run_fixture(local_func, kwargs) - - # Only apply this to coroutine functions and async generator functions in requests - # that involve the anyio_backend fixture - func = fixturedef.func - if isasyncgenfunction(func) or iscoroutinefunction(func): - if "anyio_backend" in request.fixturenames: - fixturedef.func = wrapper - original_argname = fixturedef.argnames - - if not (has_backend_arg := "anyio_backend" in fixturedef.argnames): - fixturedef.argnames += ("anyio_backend",) - - if not (has_request_arg := "request" in fixturedef.argnames): - fixturedef.argnames += ("request",) - - try: - return (yield) - finally: - fixturedef.func = func - fixturedef.argnames = original_argname - - return (yield) - - -@pytest.hookimpl(tryfirst=True) -def pytest_pycollect_makeitem( - collector: pytest.Module | pytest.Class, name: str, obj: object -) -> None: - if collector.istestfunction(obj, name): - inner_func = obj.hypothesis.inner_test if hasattr(obj, "hypothesis") else obj - if iscoroutinefunction(inner_func): - anyio_auto_mode = collector.config.getini("anyio_mode") == "auto" - marker = collector.get_closest_marker("anyio") - own_markers = getattr(obj, "pytestmark", ()) - if ( - anyio_auto_mode - or marker - or any(marker.name == "anyio" for marker in own_markers) - ): - pytest.mark.usefixtures("anyio_backend")(obj) - - -@pytest.hookimpl(tryfirst=True) -def pytest_pyfunc_call(pyfuncitem: Any) -> bool | None: - def run_with_hypothesis(**kwargs: Any) -> None: - with get_runner(backend_name, backend_options) as runner: - runner.run_test(original_func, kwargs) - - backend = pyfuncitem.funcargs.get("anyio_backend") - if backend: - backend_name, backend_options = extract_backend_and_options(backend) - - if hasattr(pyfuncitem.obj, "hypothesis"): - # Wrap the inner test function unless it's already wrapped - original_func = pyfuncitem.obj.hypothesis.inner_test - if original_func.__qualname__ != run_with_hypothesis.__qualname__: - if iscoroutinefunction(original_func): - pyfuncitem.obj.hypothesis.inner_test = run_with_hypothesis - - return None - - if iscoroutinefunction(pyfuncitem.obj): - funcargs = pyfuncitem.funcargs - testargs = {arg: funcargs[arg] for arg in pyfuncitem._fixtureinfo.argnames} - with get_runner(backend_name, backend_options) as runner: - try: - runner.run_test(pyfuncitem.obj, testargs) - except ExceptionGroup as excgrp: - for exc in iterate_exceptions(excgrp): - if isinstance(exc, (Exit, KeyboardInterrupt, SystemExit)): - raise exc from excgrp - - raise - - return True - - return None - - -@pytest.fixture(scope="module", params=get_available_backends()) -def anyio_backend(request: Any) -> Any: - return request.param - - -@pytest.fixture -def anyio_backend_name(anyio_backend: Any) -> str: - if isinstance(anyio_backend, str): - return anyio_backend - else: - return anyio_backend[0] - - -@pytest.fixture -def anyio_backend_options(anyio_backend: Any) -> dict[str, Any]: - if isinstance(anyio_backend, str): - return {} - else: - return anyio_backend[1] - - -class FreePortFactory: - """ - Manages port generation based on specified socket kind, ensuring no duplicate - ports are generated. - - This class provides functionality for generating available free ports on the - system. It is initialized with a specific socket kind and can generate ports - for given address families while avoiding reuse of previously generated ports. - - Users should not instantiate this class directly, but use the - ``free_tcp_port_factory`` and ``free_udp_port_factory`` fixtures instead. For simple - uses cases, ``free_tcp_port`` and ``free_udp_port`` can be used instead. - """ - - def __init__(self, kind: socket.SocketKind) -> None: - self._kind = kind - self._generated = set[int]() - - @property - def kind(self) -> socket.SocketKind: - """ - The type of socket connection (e.g., :data:`~socket.SOCK_STREAM` or - :data:`~socket.SOCK_DGRAM`) used to bind for checking port availability - - """ - return self._kind - - def __call__(self, family: socket.AddressFamily | None = None) -> int: - """ - Return an unbound port for the given address family. - - :param family: if omitted, both IPv4 and IPv6 addresses will be tried - :return: a port number - - """ - if family is not None: - families = [family] - else: - families = [socket.AF_INET] - if socket.has_ipv6: - families.append(socket.AF_INET6) - - while True: - port = 0 - with ExitStack() as stack: - for family in families: - sock = stack.enter_context(socket.socket(family, self._kind)) - addr = "::1" if family == socket.AF_INET6 else "127.0.0.1" - try: - sock.bind((addr, port)) - except OSError: - break - - if not port: - port = sock.getsockname()[1] - else: - if port not in self._generated: - self._generated.add(port) - return port - - -@pytest.fixture(scope="session") -def free_tcp_port_factory() -> FreePortFactory: - return FreePortFactory(socket.SOCK_STREAM) - - -@pytest.fixture(scope="session") -def free_udp_port_factory() -> FreePortFactory: - return FreePortFactory(socket.SOCK_DGRAM) - - -@pytest.fixture -def free_tcp_port(free_tcp_port_factory: Callable[[], int]) -> int: - return free_tcp_port_factory() - - -@pytest.fixture -def free_udp_port(free_udp_port_factory: Callable[[], int]) -> int: - return free_udp_port_factory() diff --git a/.env/Lib/site-packages/anyio/streams/__init__.py b/.env/Lib/site-packages/anyio/streams/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/anyio/streams/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/anyio/streams/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index f4186ef1af75e64ca16c0c280b0aa88c6b5b5360..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 186 zcmX@j%ge<81V;|EWrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxd73pjh6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdilc?4r6%SU>&M4u=4F<|$LkeT{^GF7%}*)KNwq6t1=_?2#Kj=SM`lJw#v*1Q F3jjNHGUxyR diff --git a/.env/Lib/site-packages/anyio/streams/__pycache__/buffered.cpython-312.pyc b/.env/Lib/site-packages/anyio/streams/__pycache__/buffered.cpython-312.pyc deleted file mode 100644 index d63c0a4868e4a770cfe965838260a0b104638a9e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9065 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 QwcKE#ZO3m2{Ic`-KSD=HPXGV_ diff --git a/.env/Lib/site-packages/anyio/streams/__pycache__/file.cpython-312.pyc b/.env/Lib/site-packages/anyio/streams/__pycache__/file.cpython-312.pyc deleted file mode 100644 index 2436c79bcd4749d077dc5389704586abb0430f1a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7560 zcmcgxYitzP6~1>K`|{%T6F+zj;21W5&BI_Y4(aktOkxOzL|LHg@y-AXyF2ckF=ibT zA>yQt)T*hJmj1{O@}mtdtss$_CQ_xwRiymrA3LTsof0Kc`=jYEgI$r*{^&V(W_H(( zX-OL8g72Jr?)#i`&-w1%-!E7YB9PvH@1@k)azg%s1uyv=f=>S(h)YBx8cmZ7jZ;cA zAud2!Op6&QE@kAn3_PMqXOijFo*N4)znYwsgraoRzi9q%dtz?L3 zVZHjQ=&Uooh;tVJxAdyu@)mPm1bAioV!iIFFlpoIZ>RNsSUAkolEhAx_f<0<>#KPKy>g^9!UcNJDff$l{>a4p*yA)Ynde)Up z(h;(gyV{}CPeb_<(Mgn0SaJYPWFP-Y%!I!Y#(y7MrSl&2NH>_n$Z&pJCsOd46QGXbU2atPM?}~ zYS=;?t{%sTBoakdkVvpaShWPQ5i$)qxp!n&+p#02&dg(n49!rURFkLlEU4{RZ|>NW zsU!pWv$LtBZXRpWvuBSzk?J{Sf_bd#Rg-7blfYK9xs=gta-%YvaiLAUIcG<>EnJ2f zh3rN0X?4Ryb>oHI!>cEx#!t(YiSiXtXqb>zd|I)5qGIKRy~De1NGsV&Xw3(hUlUDG zUC@LkYjjA8i<(ap;46V#{hBx=<#prBNg%dJVy9~AXx)zCB;6whW1-Fn0Y%g43Lq|# zK{`kh$mc0XMrh{ z?4E;pek@!PUy@~*BS%`v_vBxS{gkbS9#M%^!^aM=zHGwGWs~e-EXqkdjU$w;fs%=X zxm)+Cyk=nG+>JZ@RoEQ&{x|aZ1#+vjX0lWnFI6sAy;1*q{p9iu1*A&Qf;*flDpeJ1lx0i=WTlJ==CS6cxD-<#x+9!?<%d$6A2-Rra+3`kCk_U|+v! zc~ANvG#dIKG8(zD{h)K0p(>wn@mvdAfw;dGPNN9sB+pd54GGR(&~<-pd%;zCdwJ*Z zZ{zRACm-n?f28xqBhFr;e#ejVP~tzOnixo(b>!m+L!*rf+$Y$fe55tGZ|rs*MA=+> z8BqS0MWM$rUxWc#?fpU;$#vW7c^2Fsz-RDhJ#19z$f%J;aGMy2}L7BadexKk8=Cs4>cVdE$q9=jnl z_?T|>v8*k18d(SxY&+BqEOTMgQ9wUy6%S1vJ$iht0-WQ=LCQ;UjAfW9n61SO4b2Yp zG6O)Bm1AwlTY=e1%+Q^(MlVB_sX#VFrUfe3Py496Y05`@rFVn~6h)wJBjt3)eo6St zC)fVy^tXUj6kBOr)Fp_g5LrEYjLV32A;QXWpY8|u8qy@716qkDL;MZqe9D^Z>e29DJZyr&LQpXl5!Wn2}Lb1%s-R%0i9q&}PoG^h{HT_ipP|nVR7hZC+zO z1{kB#qw84(#FN!E<%GfbxByRav>uKz6~ITjGf-2Lauz@iU}06$bgivT+0?Y2PcaJ# zVAOm8-R5gRJV^8{M3q76`Jnk?uSUJ-0?}6}j=tvwfeaE0T_sOeJ_-!xBa%0A7sly& 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 bm}==uuHSjLA+`R6EQ5u#!9x>f>8kw?zy#A; diff --git a/.env/Lib/site-packages/anyio/streams/__pycache__/stapled.cpython-312.pyc b/.env/Lib/site-packages/anyio/streams/__pycache__/stapled.cpython-312.pyc deleted file mode 100644 index b7d9d1f786f0aeb613c9777c74bc34c65fce9cbd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7536 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 (3, 12): - from typing import override -else: - from typing_extensions import override - - -@dataclass(eq=False) -class BufferedByteReceiveStream(ByteReceiveStream): - """ - Wraps any bytes-based receive stream and uses a buffer to provide sophisticated - receiving capabilities in the form of a byte stream. - """ - - receive_stream: AnyByteReceiveStream - _buffer: bytearray = field(init=False, default_factory=bytearray) - _closed: bool = field(init=False, default=False) - - async def aclose(self) -> None: - await self.receive_stream.aclose() - self._closed = True - - @property - def buffer(self) -> bytes: - """The bytes currently in the buffer.""" - return bytes(self._buffer) - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return self.receive_stream.extra_attributes - - def feed_data(self, data: Iterable[SupportsIndex], /) -> None: - """ - Append data directly into the buffer. - - Any data in the buffer will be consumed by receive operations before receiving - anything from the wrapped stream. - - :param data: the data to append to the buffer (can be bytes or anything else - that supports ``__index__()``) - - """ - self._buffer.extend(data) - - async def receive(self, max_bytes: int = 65536) -> bytes: - if self._closed: - raise ClosedResourceError - - if self._buffer: - chunk = bytes(self._buffer[:max_bytes]) - del self._buffer[:max_bytes] - return chunk - elif isinstance(self.receive_stream, ByteReceiveStream): - return await self.receive_stream.receive(max_bytes) - else: - # With a bytes-oriented object stream, we need to handle any surplus bytes - # we get from the receive() call - chunk = await self.receive_stream.receive() - if len(chunk) > max_bytes: - # Save the surplus bytes in the buffer - self._buffer.extend(chunk[max_bytes:]) - return chunk[:max_bytes] - else: - return chunk - - async def receive_exactly(self, nbytes: int) -> bytes: - """ - Read exactly the given amount of bytes from the stream. - - :param nbytes: the number of bytes to read - :return: the bytes read - :raises ~anyio.IncompleteRead: if the stream was closed before the requested - amount of bytes could be read from the stream - - """ - while True: - remaining = nbytes - len(self._buffer) - if remaining <= 0: - retval = self._buffer[:nbytes] - del self._buffer[:nbytes] - return bytes(retval) - - try: - if isinstance(self.receive_stream, ByteReceiveStream): - chunk = await self.receive_stream.receive(remaining) - else: - chunk = await self.receive_stream.receive() - except EndOfStream as exc: - raise IncompleteRead from exc - - self._buffer.extend(chunk) - - async def receive_until(self, delimiter: bytes, max_bytes: int) -> bytes: - """ - Read from the stream until the delimiter is found or max_bytes have been read. - - :param delimiter: the marker to look for in the stream - :param max_bytes: maximum number of bytes that will be read before raising - :exc:`~anyio.DelimiterNotFound` - :return: the bytes read (not including the delimiter) - :raises ~anyio.IncompleteRead: if the stream was closed before the delimiter - was found - :raises ~anyio.DelimiterNotFound: if the delimiter is not found within the - bytes read up to the maximum allowed - - """ - delimiter_size = len(delimiter) - offset = 0 - while True: - # Check if the delimiter can be found in the current buffer - index = self._buffer.find(delimiter, offset) - if index >= 0: - found = self._buffer[:index] - del self._buffer[: index + len(delimiter) :] - return bytes(found) - - # Check if the buffer is already at or over the limit - if len(self._buffer) >= max_bytes: - raise DelimiterNotFound(max_bytes) - - # Read more data into the buffer from the socket - try: - data = await self.receive_stream.receive() - except EndOfStream as exc: - raise IncompleteRead from exc - - # Move the offset forward and add the new data to the buffer - offset = max(len(self._buffer) - delimiter_size + 1, 0) - self._buffer.extend(data) - - -class BufferedByteStream(BufferedByteReceiveStream, ByteStream): - """ - A full-duplex variant of :class:`BufferedByteReceiveStream`. All writes are passed - through to the wrapped stream as-is. - """ - - def __init__(self, stream: AnyByteStream): - """ - :param stream: the stream to be wrapped - - """ - super().__init__(stream) - self._stream = stream - - @override - async def send_eof(self) -> None: - await self._stream.send_eof() - - @override - async def send(self, item: bytes) -> None: - await self._stream.send(item) - - -class BufferedConnectable(ByteStreamConnectable): - def __init__(self, connectable: AnyByteStreamConnectable): - """ - :param connectable: the connectable to wrap - - """ - self.connectable = connectable - - @override - async def connect(self) -> BufferedByteStream: - stream = await self.connectable.connect() - return BufferedByteStream(stream) diff --git a/.env/Lib/site-packages/anyio/streams/file.py b/.env/Lib/site-packages/anyio/streams/file.py deleted file mode 100644 index 82d2da8..0000000 --- a/.env/Lib/site-packages/anyio/streams/file.py +++ /dev/null @@ -1,154 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "FileReadStream", - "FileStreamAttribute", - "FileWriteStream", -) - -from collections.abc import Callable, Mapping -from io import SEEK_SET, UnsupportedOperation -from os import PathLike -from pathlib import Path -from typing import Any, BinaryIO, cast - -from .. import ( - BrokenResourceError, - ClosedResourceError, - EndOfStream, - TypedAttributeSet, - to_thread, - typed_attribute, -) -from ..abc import ByteReceiveStream, ByteSendStream - - -class FileStreamAttribute(TypedAttributeSet): - #: the open file descriptor - file: BinaryIO = typed_attribute() - #: the path of the file on the file system, if available (file must be a real file) - path: Path = typed_attribute() - #: the file number, if available (file must be a real file or a TTY) - fileno: int = typed_attribute() - - -class _BaseFileStream: - def __init__(self, file: BinaryIO): - self._file = file - - async def aclose(self) -> None: - await to_thread.run_sync(self._file.close) - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - attributes: dict[Any, Callable[[], Any]] = { - FileStreamAttribute.file: lambda: self._file, - } - - if hasattr(self._file, "name"): - attributes[FileStreamAttribute.path] = lambda: Path(self._file.name) - - try: - self._file.fileno() - except UnsupportedOperation: - pass - else: - attributes[FileStreamAttribute.fileno] = lambda: self._file.fileno() - - return attributes - - -class FileReadStream(_BaseFileStream, ByteReceiveStream): - """ - A byte stream that reads from a file in the file system. - - :param file: a file that has been opened for reading in binary mode - - .. versionadded:: 3.0 - """ - - @classmethod - async def from_path(cls, path: str | PathLike[str]) -> FileReadStream: - """ - Create a file read stream by opening the given file. - - :param path: path of the file to read from - - """ - file = await to_thread.run_sync(Path(path).open, "rb") - return cls(cast(BinaryIO, file)) - - async def receive(self, max_bytes: int = 65536) -> bytes: - try: - data = await to_thread.run_sync(self._file.read, max_bytes) - except ValueError: - raise ClosedResourceError from None - except OSError as exc: - raise BrokenResourceError from exc - - if data: - return data - else: - raise EndOfStream - - async def seek(self, position: int, whence: int = SEEK_SET) -> int: - """ - Seek the file to the given position. - - .. seealso:: :meth:`io.IOBase.seek` - - .. note:: Not all file descriptors are seekable. - - :param position: position to seek the file to - :param whence: controls how ``position`` is interpreted - :return: the new absolute position - :raises OSError: if the file is not seekable - - """ - return await to_thread.run_sync(self._file.seek, position, whence) - - async def tell(self) -> int: - """ - Return the current stream position. - - .. note:: Not all file descriptors are seekable. - - :return: the current absolute position - :raises OSError: if the file is not seekable - - """ - return await to_thread.run_sync(self._file.tell) - - -class FileWriteStream(_BaseFileStream, ByteSendStream): - """ - A byte stream that writes to a file in the file system. - - :param file: a file that has been opened for writing in binary mode - - .. versionadded:: 3.0 - """ - - @classmethod - async def from_path( - cls, path: str | PathLike[str], append: bool = False - ) -> FileWriteStream: - """ - Create a file write stream by opening the given file for writing. - - :param path: path of the file to write to - :param append: if ``True``, open the file for appending; if ``False``, any - existing file at the given path will be truncated - - """ - mode = "ab" if append else "wb" - file = await to_thread.run_sync(Path(path).open, mode) - return cls(cast(BinaryIO, file)) - - async def send(self, item: bytes) -> None: - try: - await to_thread.run_sync(self._file.write, item) - except ValueError: - raise ClosedResourceError from None - except OSError as exc: - raise BrokenResourceError from exc diff --git a/.env/Lib/site-packages/anyio/streams/memory.py b/.env/Lib/site-packages/anyio/streams/memory.py deleted file mode 100644 index a3fa0c3..0000000 --- a/.env/Lib/site-packages/anyio/streams/memory.py +++ /dev/null @@ -1,325 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "MemoryObjectReceiveStream", - "MemoryObjectSendStream", - "MemoryObjectStreamStatistics", -) - -import warnings -from collections import OrderedDict, deque -from dataclasses import dataclass, field -from types import TracebackType -from typing import Generic, NamedTuple, TypeVar - -from .. import ( - BrokenResourceError, - ClosedResourceError, - EndOfStream, - WouldBlock, -) -from .._core._testing import TaskInfo, get_current_task -from ..abc import Event, ObjectReceiveStream, ObjectSendStream -from ..lowlevel import checkpoint - -T_Item = TypeVar("T_Item") -T_co = TypeVar("T_co", covariant=True) -T_contra = TypeVar("T_contra", contravariant=True) - - -class MemoryObjectStreamStatistics(NamedTuple): - current_buffer_used: int #: number of items stored in the buffer - #: maximum number of items that can be stored on this stream (or :data:`math.inf`) - max_buffer_size: float - open_send_streams: int #: number of unclosed clones of the send stream - open_receive_streams: int #: number of unclosed clones of the receive stream - #: number of tasks blocked on :meth:`MemoryObjectSendStream.send` - tasks_waiting_send: int - #: number of tasks blocked on :meth:`MemoryObjectReceiveStream.receive` - tasks_waiting_receive: int - - -@dataclass(eq=False) -class _MemoryObjectItemReceiver(Generic[T_Item]): - task_info: TaskInfo = field(init=False, default_factory=get_current_task) - item: T_Item = field(init=False) - - def __repr__(self) -> str: - # When item is not defined, we get following error with default __repr__: - # AttributeError: 'MemoryObjectItemReceiver' object has no attribute 'item' - item = getattr(self, "item", None) - return f"{self.__class__.__name__}(task_info={self.task_info}, item={item!r})" - - -@dataclass(eq=False) -class _MemoryObjectStreamState(Generic[T_Item]): - max_buffer_size: float = field() - buffer: deque[T_Item] = field(init=False, default_factory=deque) - open_send_channels: int = field(init=False, default=0) - open_receive_channels: int = field(init=False, default=0) - waiting_receivers: OrderedDict[Event, _MemoryObjectItemReceiver[T_Item]] = field( - init=False, default_factory=OrderedDict - ) - waiting_senders: OrderedDict[Event, T_Item] = field( - init=False, default_factory=OrderedDict - ) - - def statistics(self) -> MemoryObjectStreamStatistics: - return MemoryObjectStreamStatistics( - len(self.buffer), - self.max_buffer_size, - self.open_send_channels, - self.open_receive_channels, - len(self.waiting_senders), - len(self.waiting_receivers), - ) - - -@dataclass(eq=False) -class MemoryObjectReceiveStream(Generic[T_co], ObjectReceiveStream[T_co]): - _state: _MemoryObjectStreamState[T_co] - _closed: bool = field(init=False, default=False) - - def __post_init__(self) -> None: - self._state.open_receive_channels += 1 - - def receive_nowait(self) -> T_co: - """ - Receive the next item if it can be done without waiting. - - :return: the received item - :raises ~anyio.ClosedResourceError: if this send stream has been closed - :raises ~anyio.EndOfStream: if the buffer is empty and this stream has been - closed from the sending end - :raises ~anyio.WouldBlock: if there are no items in the buffer and no tasks - waiting to send - - """ - if self._closed: - raise ClosedResourceError - - if self._state.waiting_senders: - # Get the item from the next sender - send_event, item = self._state.waiting_senders.popitem(last=False) - self._state.buffer.append(item) - send_event.set() - - if self._state.buffer: - return self._state.buffer.popleft() - elif not self._state.open_send_channels: - raise EndOfStream - - raise WouldBlock - - async def receive(self) -> T_co: - await checkpoint() - try: - return self.receive_nowait() - except WouldBlock: - # Add ourselves in the queue - receive_event = Event() - receiver = _MemoryObjectItemReceiver[T_co]() - self._state.waiting_receivers[receive_event] = receiver - - try: - await receive_event.wait() - finally: - self._state.waiting_receivers.pop(receive_event, None) - - try: - return receiver.item - except AttributeError: - raise EndOfStream from None - - def clone(self) -> MemoryObjectReceiveStream[T_co]: - """ - Create a clone of this receive stream. - - Each clone can be closed separately. Only when all clones have been closed will - the receiving end of the memory stream be considered closed by the sending ends. - - :return: the cloned stream - - """ - if self._closed: - raise ClosedResourceError - - return MemoryObjectReceiveStream(_state=self._state) - - def close(self) -> None: - """ - Close the stream. - - This works the exact same way as :meth:`aclose`, but is provided as a special - case for the benefit of synchronous callbacks. - - """ - if not self._closed: - self._closed = True - self._state.open_receive_channels -= 1 - if self._state.open_receive_channels == 0: - send_events = list(self._state.waiting_senders.keys()) - for event in send_events: - event.set() - - async def aclose(self) -> None: - self.close() - - def statistics(self) -> MemoryObjectStreamStatistics: - """ - Return statistics about the current state of this stream. - - .. versionadded:: 3.0 - """ - return self._state.statistics() - - def __enter__(self) -> MemoryObjectReceiveStream[T_co]: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.close() - - def __del__(self) -> None: - if not self._closed: - warnings.warn( - f"Unclosed <{self.__class__.__name__} at {id(self):x}>", - ResourceWarning, - stacklevel=1, - source=self, - ) - - -@dataclass(eq=False) -class MemoryObjectSendStream(Generic[T_contra], ObjectSendStream[T_contra]): - _state: _MemoryObjectStreamState[T_contra] - _closed: bool = field(init=False, default=False) - - def __post_init__(self) -> None: - self._state.open_send_channels += 1 - - def send_nowait(self, item: T_contra) -> None: - """ - Send an item immediately if it can be done without waiting. - - :param item: the item to send - :raises ~anyio.ClosedResourceError: if this send stream has been closed - :raises ~anyio.BrokenResourceError: if the stream has been closed from the - receiving end - :raises ~anyio.WouldBlock: if the buffer is full and there are no tasks waiting - to receive - - """ - if self._closed: - raise ClosedResourceError - if not self._state.open_receive_channels: - raise BrokenResourceError - - while self._state.waiting_receivers: - receive_event, receiver = self._state.waiting_receivers.popitem(last=False) - if not receiver.task_info.has_pending_cancellation(): - receiver.item = item - receive_event.set() - return - - if len(self._state.buffer) < self._state.max_buffer_size: - self._state.buffer.append(item) - else: - raise WouldBlock - - async def send(self, item: T_contra) -> None: - """ - Send an item to the stream. - - If the buffer is full, this method blocks until there is again room in the - buffer or the item can be sent directly to a receiver. - - :param item: the item to send - :raises ~anyio.ClosedResourceError: if this send stream has been closed - :raises ~anyio.BrokenResourceError: if the stream has been closed from the - receiving end - - """ - await checkpoint() - try: - self.send_nowait(item) - except WouldBlock: - # Wait until there's someone on the receiving end - send_event = Event() - self._state.waiting_senders[send_event] = item - try: - await send_event.wait() - except BaseException: - self._state.waiting_senders.pop(send_event, None) - raise - - if send_event in self._state.waiting_senders: - del self._state.waiting_senders[send_event] - raise BrokenResourceError from None - - def clone(self) -> MemoryObjectSendStream[T_contra]: - """ - Create a clone of this send stream. - - Each clone can be closed separately. Only when all clones have been closed will - the sending end of the memory stream be considered closed by the receiving ends. - - :return: the cloned stream - - """ - if self._closed: - raise ClosedResourceError - - return MemoryObjectSendStream(_state=self._state) - - def close(self) -> None: - """ - Close the stream. - - This works the exact same way as :meth:`aclose`, but is provided as a special - case for the benefit of synchronous callbacks. - - """ - if not self._closed: - self._closed = True - self._state.open_send_channels -= 1 - if self._state.open_send_channels == 0: - receive_events = list(self._state.waiting_receivers.keys()) - self._state.waiting_receivers.clear() - for event in receive_events: - event.set() - - async def aclose(self) -> None: - self.close() - - def statistics(self) -> MemoryObjectStreamStatistics: - """ - Return statistics about the current state of this stream. - - .. versionadded:: 3.0 - """ - return self._state.statistics() - - def __enter__(self) -> MemoryObjectSendStream[T_contra]: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_val: BaseException | None, - exc_tb: TracebackType | None, - ) -> None: - self.close() - - def __del__(self) -> None: - if not self._closed: - warnings.warn( - f"Unclosed <{self.__class__.__name__} at {id(self):x}>", - ResourceWarning, - stacklevel=1, - source=self, - ) diff --git a/.env/Lib/site-packages/anyio/streams/stapled.py b/.env/Lib/site-packages/anyio/streams/stapled.py deleted file mode 100644 index 9248b68..0000000 --- a/.env/Lib/site-packages/anyio/streams/stapled.py +++ /dev/null @@ -1,147 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "MultiListener", - "StapledByteStream", - "StapledObjectStream", -) - -from collections.abc import Callable, Mapping, Sequence -from dataclasses import dataclass -from typing import Any, Generic, TypeVar - -from ..abc import ( - ByteReceiveStream, - ByteSendStream, - ByteStream, - Listener, - ObjectReceiveStream, - ObjectSendStream, - ObjectStream, - TaskGroup, -) - -T_Item = TypeVar("T_Item") -T_Stream = TypeVar("T_Stream") - - -@dataclass(eq=False) -class StapledByteStream(ByteStream): - """ - Combines two byte streams into a single, bidirectional byte stream. - - Extra attributes will be provided from both streams, with the receive stream - providing the values in case of a conflict. - - :param ByteSendStream send_stream: the sending byte stream - :param ByteReceiveStream receive_stream: the receiving byte stream - """ - - send_stream: ByteSendStream - receive_stream: ByteReceiveStream - - async def receive(self, max_bytes: int = 65536) -> bytes: - return await self.receive_stream.receive(max_bytes) - - async def send(self, item: bytes) -> None: - await self.send_stream.send(item) - - async def send_eof(self) -> None: - await self.send_stream.aclose() - - async def aclose(self) -> None: - await self.send_stream.aclose() - await self.receive_stream.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return { - **self.send_stream.extra_attributes, - **self.receive_stream.extra_attributes, - } - - -@dataclass(eq=False) -class StapledObjectStream(Generic[T_Item], ObjectStream[T_Item]): - """ - Combines two object streams into a single, bidirectional object stream. - - Extra attributes will be provided from both streams, with the receive stream - providing the values in case of a conflict. - - :param ObjectSendStream send_stream: the sending object stream - :param ObjectReceiveStream receive_stream: the receiving object stream - """ - - send_stream: ObjectSendStream[T_Item] - receive_stream: ObjectReceiveStream[T_Item] - - async def receive(self) -> T_Item: - return await self.receive_stream.receive() - - async def send(self, item: T_Item) -> None: - await self.send_stream.send(item) - - async def send_eof(self) -> None: - await self.send_stream.aclose() - - async def aclose(self) -> None: - await self.send_stream.aclose() - await self.receive_stream.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return { - **self.send_stream.extra_attributes, - **self.receive_stream.extra_attributes, - } - - -@dataclass(eq=False) -class MultiListener(Generic[T_Stream], Listener[T_Stream]): - """ - Combines multiple listeners into one, serving connections from all of them at once. - - Any MultiListeners in the given collection of listeners will have their listeners - moved into this one. - - Extra attributes are provided from each listener, with each successive listener - overriding any conflicting attributes from the previous one. - - :param listeners: listeners to serve - :type listeners: Sequence[Listener[T_Stream]] - """ - - listeners: Sequence[Listener[T_Stream]] - - def __post_init__(self) -> None: - listeners: list[Listener[T_Stream]] = [] - for listener in self.listeners: - if isinstance(listener, MultiListener): - listeners.extend(listener.listeners) - del listener.listeners[:] # type: ignore[attr-defined] - else: - listeners.append(listener) - - self.listeners = listeners - - async def serve( - self, handler: Callable[[T_Stream], Any], task_group: TaskGroup | None = None - ) -> None: - from .. import create_task_group - - async with create_task_group() as tg: - for listener in self.listeners: - tg.start_soon(listener.serve, handler, task_group) - - async def aclose(self) -> None: - for listener in self.listeners: - await listener.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - attributes: dict = {} - for listener in self.listeners: - attributes.update(listener.extra_attributes) - - return attributes diff --git a/.env/Lib/site-packages/anyio/streams/text.py b/.env/Lib/site-packages/anyio/streams/text.py deleted file mode 100644 index 296cd25..0000000 --- a/.env/Lib/site-packages/anyio/streams/text.py +++ /dev/null @@ -1,176 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "TextConnectable", - "TextReceiveStream", - "TextSendStream", - "TextStream", -) - -import codecs -import sys -from collections.abc import Callable, Mapping -from dataclasses import InitVar, dataclass, field -from typing import Any - -from ..abc import ( - AnyByteReceiveStream, - AnyByteSendStream, - AnyByteStream, - AnyByteStreamConnectable, - ObjectReceiveStream, - ObjectSendStream, - ObjectStream, - ObjectStreamConnectable, -) - -if sys.version_info >= (3, 12): - from typing import override -else: - from typing_extensions import override - - -@dataclass(eq=False) -class TextReceiveStream(ObjectReceiveStream[str]): - """ - Stream wrapper that decodes bytes to strings using the given encoding. - - Decoding is done using :class:`~codecs.IncrementalDecoder` which returns any - completely received unicode characters as soon as they come in. - - :param transport_stream: any bytes-based receive stream - :param encoding: character encoding to use for decoding bytes to strings (defaults - to ``utf-8``) - :param errors: handling scheme for decoding errors (defaults to ``strict``; see the - `codecs module documentation`_ for a comprehensive list of options) - - .. _codecs module documentation: - https://docs.python.org/3/library/codecs.html#codec-objects - """ - - transport_stream: AnyByteReceiveStream - encoding: InitVar[str] = "utf-8" - errors: InitVar[str] = "strict" - _decoder: codecs.IncrementalDecoder = field(init=False) - - def __post_init__(self, encoding: str, errors: str) -> None: - decoder_class = codecs.getincrementaldecoder(encoding) - self._decoder = decoder_class(errors=errors) - - async def receive(self) -> str: - while True: - chunk = await self.transport_stream.receive() - decoded = self._decoder.decode(chunk) - if decoded: - return decoded - - async def aclose(self) -> None: - await self.transport_stream.aclose() - self._decoder.reset() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return self.transport_stream.extra_attributes - - -@dataclass(eq=False) -class TextSendStream(ObjectSendStream[str]): - """ - Sends strings to the wrapped stream as bytes using the given encoding. - - :param AnyByteSendStream transport_stream: any bytes-based send stream - :param str encoding: character encoding to use for encoding strings to bytes - (defaults to ``utf-8``) - :param str errors: handling scheme for encoding errors (defaults to ``strict``; see - the `codecs module documentation`_ for a comprehensive list of options) - - .. _codecs module documentation: - https://docs.python.org/3/library/codecs.html#codec-objects - """ - - transport_stream: AnyByteSendStream - encoding: InitVar[str] = "utf-8" - errors: str = "strict" - _encoder: Callable[..., tuple[bytes, int]] = field(init=False) - - def __post_init__(self, encoding: str) -> None: - self._encoder = codecs.getencoder(encoding) - - async def send(self, item: str) -> None: - encoded = self._encoder(item, self.errors)[0] - await self.transport_stream.send(encoded) - - async def aclose(self) -> None: - await self.transport_stream.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return self.transport_stream.extra_attributes - - -@dataclass(eq=False) -class TextStream(ObjectStream[str]): - """ - A bidirectional stream that decodes bytes to strings on receive and encodes strings - to bytes on send. - - Extra attributes will be provided from both streams, with the receive stream - providing the values in case of a conflict. - - :param AnyByteStream transport_stream: any bytes-based stream - :param str encoding: character encoding to use for encoding/decoding strings to/from - bytes (defaults to ``utf-8``) - :param str errors: handling scheme for encoding errors (defaults to ``strict``; see - the `codecs module documentation`_ for a comprehensive list of options) - - .. _codecs module documentation: - https://docs.python.org/3/library/codecs.html#codec-objects - """ - - transport_stream: AnyByteStream - encoding: InitVar[str] = "utf-8" - errors: InitVar[str] = "strict" - _receive_stream: TextReceiveStream = field(init=False) - _send_stream: TextSendStream = field(init=False) - - def __post_init__(self, encoding: str, errors: str) -> None: - self._receive_stream = TextReceiveStream( - self.transport_stream, encoding=encoding, errors=errors - ) - self._send_stream = TextSendStream( - self.transport_stream, encoding=encoding, errors=errors - ) - - async def receive(self) -> str: - return await self._receive_stream.receive() - - async def send(self, item: str) -> None: - await self._send_stream.send(item) - - async def send_eof(self) -> None: - await self.transport_stream.send_eof() - - async def aclose(self) -> None: - await self._send_stream.aclose() - await self._receive_stream.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return { - **self._send_stream.extra_attributes, - **self._receive_stream.extra_attributes, - } - - -class TextConnectable(ObjectStreamConnectable[str]): - def __init__(self, connectable: AnyByteStreamConnectable): - """ - :param connectable: the bytestream endpoint to wrap - - """ - self.connectable = connectable - - @override - async def connect(self) -> TextStream: - stream = await self.connectable.connect() - return TextStream(stream) diff --git a/.env/Lib/site-packages/anyio/streams/tls.py b/.env/Lib/site-packages/anyio/streams/tls.py deleted file mode 100644 index b507488..0000000 --- a/.env/Lib/site-packages/anyio/streams/tls.py +++ /dev/null @@ -1,424 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "TLSAttribute", - "TLSConnectable", - "TLSListener", - "TLSStream", -) - -import logging -import re -import ssl -import sys -from collections.abc import Callable, Mapping -from dataclasses import dataclass -from functools import wraps -from ssl import SSLContext -from typing import Any, TypeVar - -from .. import ( - BrokenResourceError, - EndOfStream, - aclose_forcefully, - get_cancelled_exc_class, - to_thread, -) -from .._core._typedattr import TypedAttributeSet, typed_attribute -from ..abc import ( - AnyByteStream, - AnyByteStreamConnectable, - ByteStream, - ByteStreamConnectable, - Listener, - TaskGroup, -) - -if sys.version_info >= (3, 10): - from typing import TypeAlias -else: - from typing_extensions import TypeAlias - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -if sys.version_info >= (3, 12): - from typing import override -else: - from typing_extensions import override - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") -_PCTRTT: TypeAlias = tuple[tuple[str, str], ...] -_PCTRTTT: TypeAlias = tuple[_PCTRTT, ...] - - -class TLSAttribute(TypedAttributeSet): - """Contains Transport Layer Security related attributes.""" - - #: the selected ALPN protocol - alpn_protocol: str | None = typed_attribute() - #: the channel binding for type ``tls-unique`` - channel_binding_tls_unique: bytes = typed_attribute() - #: the selected cipher - cipher: tuple[str, str, int] = typed_attribute() - #: the peer certificate in dictionary form (see :meth:`ssl.SSLSocket.getpeercert` - # for more information) - peer_certificate: None | (dict[str, str | _PCTRTTT | _PCTRTT]) = typed_attribute() - #: the peer certificate in binary form - peer_certificate_binary: bytes | None = typed_attribute() - #: ``True`` if this is the server side of the connection - server_side: bool = typed_attribute() - #: ciphers shared by the client during the TLS handshake (``None`` if this is the - #: client side) - shared_ciphers: list[tuple[str, str, int]] | None = typed_attribute() - #: the :class:`~ssl.SSLObject` used for encryption - ssl_object: ssl.SSLObject = typed_attribute() - #: ``True`` if this stream does (and expects) a closing TLS handshake when the - #: stream is being closed - standard_compatible: bool = typed_attribute() - #: the TLS protocol version (e.g. ``TLSv1.2``) - tls_version: str = typed_attribute() - - -@dataclass(eq=False) -class TLSStream(ByteStream): - """ - A stream wrapper that encrypts all sent data and decrypts received data. - - This class has no public initializer; use :meth:`wrap` instead. - All extra attributes from :class:`~TLSAttribute` are supported. - - :var AnyByteStream transport_stream: the wrapped stream - - """ - - transport_stream: AnyByteStream - standard_compatible: bool - _ssl_object: ssl.SSLObject - _read_bio: ssl.MemoryBIO - _write_bio: ssl.MemoryBIO - - @classmethod - async def wrap( - cls, - transport_stream: AnyByteStream, - *, - server_side: bool | None = None, - hostname: str | None = None, - ssl_context: ssl.SSLContext | None = None, - standard_compatible: bool = True, - ) -> TLSStream: - """ - Wrap an existing stream with Transport Layer Security. - - This performs a TLS handshake with the peer. - - :param transport_stream: a bytes-transporting stream to wrap - :param server_side: ``True`` if this is the server side of the connection, - ``False`` if this is the client side (if omitted, will be set to ``False`` - if ``hostname`` has been provided, ``False`` otherwise). Used only to create - a default context when an explicit context has not been provided. - :param hostname: host name of the peer (if host name checking is desired) - :param ssl_context: the SSLContext object to use (if not provided, a secure - default will be created) - :param standard_compatible: if ``False``, skip the closing handshake when - closing the connection, and don't raise an exception if the peer does the - same - :raises ~ssl.SSLError: if the TLS handshake fails - - """ - if server_side is None: - server_side = not hostname - - if not ssl_context: - purpose = ( - ssl.Purpose.CLIENT_AUTH if server_side else ssl.Purpose.SERVER_AUTH - ) - ssl_context = ssl.create_default_context(purpose) - - # Re-enable detection of unexpected EOFs if it was disabled by Python - if hasattr(ssl, "OP_IGNORE_UNEXPECTED_EOF"): - ssl_context.options &= ~ssl.OP_IGNORE_UNEXPECTED_EOF - - bio_in = ssl.MemoryBIO() - bio_out = ssl.MemoryBIO() - - # External SSLContext implementations may do blocking I/O in wrap_bio(), - # but the standard library implementation won't - if type(ssl_context) is ssl.SSLContext: - ssl_object = ssl_context.wrap_bio( - bio_in, bio_out, server_side=server_side, server_hostname=hostname - ) - else: - ssl_object = await to_thread.run_sync( - ssl_context.wrap_bio, - bio_in, - bio_out, - server_side, - hostname, - None, - ) - - wrapper = cls( - transport_stream=transport_stream, - standard_compatible=standard_compatible, - _ssl_object=ssl_object, - _read_bio=bio_in, - _write_bio=bio_out, - ) - await wrapper._call_sslobject_method(ssl_object.do_handshake) - return wrapper - - async def _call_sslobject_method( - self, func: Callable[[Unpack[PosArgsT]], T_Retval], *args: Unpack[PosArgsT] - ) -> T_Retval: - while True: - try: - result = func(*args) - except ssl.SSLWantReadError: - try: - # Flush any pending writes first - if self._write_bio.pending: - await self.transport_stream.send(self._write_bio.read()) - - data = await self.transport_stream.receive() - except EndOfStream: - self._read_bio.write_eof() - except OSError as exc: - self._read_bio.write_eof() - self._write_bio.write_eof() - raise BrokenResourceError from exc - else: - self._read_bio.write(data) - except ssl.SSLWantWriteError: - await self.transport_stream.send(self._write_bio.read()) - except ssl.SSLSyscallError as exc: - self._read_bio.write_eof() - self._write_bio.write_eof() - raise BrokenResourceError from exc - except ssl.SSLError as exc: - self._read_bio.write_eof() - self._write_bio.write_eof() - if isinstance(exc, ssl.SSLEOFError) or ( - exc.strerror and "UNEXPECTED_EOF_WHILE_READING" in exc.strerror - ): - if self.standard_compatible: - raise BrokenResourceError from exc - else: - raise EndOfStream from None - - raise - else: - # Flush any pending writes first - if self._write_bio.pending: - await self.transport_stream.send(self._write_bio.read()) - - return result - - async def unwrap(self) -> tuple[AnyByteStream, bytes]: - """ - Does the TLS closing handshake. - - :return: a tuple of (wrapped byte stream, bytes left in the read buffer) - - """ - await self._call_sslobject_method(self._ssl_object.unwrap) - self._read_bio.write_eof() - self._write_bio.write_eof() - return self.transport_stream, self._read_bio.read() - - async def aclose(self) -> None: - if self.standard_compatible: - try: - await self.unwrap() - except BaseException: - await aclose_forcefully(self.transport_stream) - raise - - await self.transport_stream.aclose() - - async def receive(self, max_bytes: int = 65536) -> bytes: - data = await self._call_sslobject_method(self._ssl_object.read, max_bytes) - if not data: - raise EndOfStream - - return data - - async def send(self, item: bytes) -> None: - await self._call_sslobject_method(self._ssl_object.write, item) - - async def send_eof(self) -> None: - tls_version = self.extra(TLSAttribute.tls_version) - match = re.match(r"TLSv(\d+)(?:\.(\d+))?", tls_version) - if match: - major, minor = int(match.group(1)), int(match.group(2) or 0) - if (major, minor) < (1, 3): - raise NotImplementedError( - f"send_eof() requires at least TLSv1.3; current " - f"session uses {tls_version}" - ) - - raise NotImplementedError( - "send_eof() has not yet been implemented for TLS streams" - ) - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return { - **self.transport_stream.extra_attributes, - TLSAttribute.alpn_protocol: self._ssl_object.selected_alpn_protocol, - TLSAttribute.channel_binding_tls_unique: ( - self._ssl_object.get_channel_binding - ), - TLSAttribute.cipher: self._ssl_object.cipher, - TLSAttribute.peer_certificate: lambda: self._ssl_object.getpeercert(False), - TLSAttribute.peer_certificate_binary: lambda: self._ssl_object.getpeercert( - True - ), - TLSAttribute.server_side: lambda: self._ssl_object.server_side, - TLSAttribute.shared_ciphers: lambda: self._ssl_object.shared_ciphers() - if self._ssl_object.server_side - else None, - TLSAttribute.standard_compatible: lambda: self.standard_compatible, - TLSAttribute.ssl_object: lambda: self._ssl_object, - TLSAttribute.tls_version: self._ssl_object.version, - } - - -@dataclass(eq=False) -class TLSListener(Listener[TLSStream]): - """ - A convenience listener that wraps another listener and auto-negotiates a TLS session - on every accepted connection. - - If the TLS handshake times out or raises an exception, - :meth:`handle_handshake_error` is called to do whatever post-mortem processing is - deemed necessary. - - Supports only the :attr:`~TLSAttribute.standard_compatible` extra attribute. - - :param Listener listener: the listener to wrap - :param ssl_context: the SSL context object - :param standard_compatible: a flag passed through to :meth:`TLSStream.wrap` - :param handshake_timeout: time limit for the TLS handshake - (passed to :func:`~anyio.fail_after`) - """ - - listener: Listener[Any] - ssl_context: ssl.SSLContext - standard_compatible: bool = True - handshake_timeout: float = 30 - - @staticmethod - async def handle_handshake_error(exc: BaseException, stream: AnyByteStream) -> None: - """ - Handle an exception raised during the TLS handshake. - - This method does 3 things: - - #. Forcefully closes the original stream - #. Logs the exception (unless it was a cancellation exception) using the - ``anyio.streams.tls`` logger - #. Reraises the exception if it was a base exception or a cancellation exception - - :param exc: the exception - :param stream: the original stream - - """ - await aclose_forcefully(stream) - - # Log all except cancellation exceptions - if not isinstance(exc, get_cancelled_exc_class()): - # CPython (as of 3.11.5) returns incorrect `sys.exc_info()` here when using - # any asyncio implementation, so we explicitly pass the exception to log - # (https://github.com/python/cpython/issues/108668). Trio does not have this - # issue because it works around the CPython bug. - logging.getLogger(__name__).exception( - "Error during TLS handshake", exc_info=exc - ) - - # Only reraise base exceptions and cancellation exceptions - if not isinstance(exc, Exception) or isinstance(exc, get_cancelled_exc_class()): - raise - - async def serve( - self, - handler: Callable[[TLSStream], Any], - task_group: TaskGroup | None = None, - ) -> None: - @wraps(handler) - async def handler_wrapper(stream: AnyByteStream) -> None: - from .. import fail_after - - try: - with fail_after(self.handshake_timeout): - wrapped_stream = await TLSStream.wrap( - stream, - ssl_context=self.ssl_context, - standard_compatible=self.standard_compatible, - ) - except BaseException as exc: - await self.handle_handshake_error(exc, stream) - else: - await handler(wrapped_stream) - - await self.listener.serve(handler_wrapper, task_group) - - async def aclose(self) -> None: - await self.listener.aclose() - - @property - def extra_attributes(self) -> Mapping[Any, Callable[[], Any]]: - return { - TLSAttribute.standard_compatible: lambda: self.standard_compatible, - } - - -class TLSConnectable(ByteStreamConnectable): - """ - Wraps another connectable and does TLS negotiation after a successful connection. - - :param connectable: the connectable to wrap - :param hostname: host name of the server (if host name checking is desired) - :param ssl_context: the SSLContext object to use (if not provided, a secure default - will be created) - :param standard_compatible: if ``False``, skip the closing handshake when closing - the connection, and don't raise an exception if the server does the same - """ - - def __init__( - self, - connectable: AnyByteStreamConnectable, - *, - hostname: str | None = None, - ssl_context: ssl.SSLContext | None = None, - standard_compatible: bool = True, - ) -> None: - self.connectable = connectable - self.ssl_context: SSLContext = ssl_context or ssl.create_default_context( - ssl.Purpose.SERVER_AUTH - ) - if not isinstance(self.ssl_context, ssl.SSLContext): - raise TypeError( - "ssl_context must be an instance of ssl.SSLContext, not " - f"{type(self.ssl_context).__name__}" - ) - self.hostname = hostname - self.standard_compatible = standard_compatible - - @override - async def connect(self) -> TLSStream: - stream = await self.connectable.connect() - try: - return await TLSStream.wrap( - stream, - hostname=self.hostname, - ssl_context=self.ssl_context, - standard_compatible=self.standard_compatible, - ) - except BaseException: - await aclose_forcefully(stream) - raise diff --git a/.env/Lib/site-packages/anyio/to_interpreter.py b/.env/Lib/site-packages/anyio/to_interpreter.py deleted file mode 100644 index 694dbe7..0000000 --- a/.env/Lib/site-packages/anyio/to_interpreter.py +++ /dev/null @@ -1,246 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "run_sync", - "current_default_interpreter_limiter", -) - -import atexit -import os -import sys -from collections import deque -from collections.abc import Callable -from typing import Any, Final, TypeVar - -from . import current_time, to_thread -from ._core._exceptions import BrokenWorkerInterpreter -from ._core._synchronization import CapacityLimiter -from .lowlevel import RunVar - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -if sys.version_info >= (3, 14): - from concurrent.interpreters import ExecutionFailed, create - - def _interp_call( - func: Callable[..., Any], args: tuple[Any, ...] - ) -> tuple[Any, bool]: - try: - retval = func(*args) - except BaseException as exc: - return exc, True - else: - return retval, False - - class _Worker: - last_used: float = 0 - - def __init__(self) -> None: - self._interpreter = create() - - def destroy(self) -> None: - self._interpreter.close() - - def call( - self, - func: Callable[..., T_Retval], - args: tuple[Any, ...], - ) -> T_Retval: - try: - res, is_exception = self._interpreter.call(_interp_call, func, args) - except ExecutionFailed as exc: - raise BrokenWorkerInterpreter(exc.excinfo) from exc - - if is_exception: - raise res - - return res -elif sys.version_info >= (3, 13): - import _interpqueues - import _interpreters - - UNBOUND: Final = 2 # I have no clue how this works, but it was used in the stdlib - FMT_UNPICKLED: Final = 0 - FMT_PICKLED: Final = 1 - QUEUE_PICKLE_ARGS: Final = (FMT_PICKLED, UNBOUND) - QUEUE_UNPICKLE_ARGS: Final = (FMT_UNPICKLED, UNBOUND) - - _run_func = compile( - """ -import _interpqueues -from _interpreters import NotShareableError -from pickle import loads, dumps, HIGHEST_PROTOCOL - -QUEUE_PICKLE_ARGS = (1, 2) -QUEUE_UNPICKLE_ARGS = (0, 2) - -item = _interpqueues.get(queue_id)[0] -try: - func, args = loads(item) - retval = func(*args) -except BaseException as exc: - is_exception = True - retval = exc -else: - is_exception = False - -try: - _interpqueues.put(queue_id, (retval, is_exception), *QUEUE_UNPICKLE_ARGS) -except NotShareableError: - retval = dumps(retval, HIGHEST_PROTOCOL) - _interpqueues.put(queue_id, (retval, is_exception), *QUEUE_PICKLE_ARGS) - """, - "", - "exec", - ) - - class _Worker: - last_used: float = 0 - - def __init__(self) -> None: - self._interpreter_id = _interpreters.create() - self._queue_id = _interpqueues.create(1, *QUEUE_UNPICKLE_ARGS) - _interpreters.set___main___attrs( - self._interpreter_id, {"queue_id": self._queue_id} - ) - - def destroy(self) -> None: - _interpqueues.destroy(self._queue_id) - _interpreters.destroy(self._interpreter_id) - - def call( - self, - func: Callable[..., T_Retval], - args: tuple[Any, ...], - ) -> T_Retval: - import pickle - - item = pickle.dumps((func, args), pickle.HIGHEST_PROTOCOL) - _interpqueues.put(self._queue_id, item, *QUEUE_PICKLE_ARGS) - exc_info = _interpreters.exec(self._interpreter_id, _run_func) - if exc_info: - raise BrokenWorkerInterpreter(exc_info) - - res = _interpqueues.get(self._queue_id) - (res, is_exception), fmt = res[:2] - if fmt == FMT_PICKLED: - res = pickle.loads(res) - - if is_exception: - raise res - - return res -else: - - class _Worker: - last_used: float = 0 - - def __init__(self) -> None: - raise RuntimeError("subinterpreters require at least Python 3.13") - - def call( - self, - func: Callable[..., T_Retval], - args: tuple[Any, ...], - ) -> T_Retval: - raise NotImplementedError - - def destroy(self) -> None: - pass - - -DEFAULT_CPU_COUNT: Final = 8 # this is just an arbitrarily selected value -MAX_WORKER_IDLE_TIME = ( - 30 # seconds a subinterpreter can be idle before becoming eligible for pruning -) - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") - -_idle_workers = RunVar[deque[_Worker]]("_available_workers") -_default_interpreter_limiter = RunVar[CapacityLimiter]("_default_interpreter_limiter") - - -def _stop_workers(workers: deque[_Worker]) -> None: - for worker in workers: - worker.destroy() - - workers.clear() - - -async def run_sync( - func: Callable[[Unpack[PosArgsT]], T_Retval], - *args: Unpack[PosArgsT], - limiter: CapacityLimiter | None = None, -) -> T_Retval: - """ - Call the given function with the given arguments in a subinterpreter. - - .. warning:: On Python 3.13, the :mod:`concurrent.interpreters` module was not yet - available, so the code path for that Python version relies on an undocumented, - private API. As such, it is recommended to not rely on this function for anything - mission-critical on Python 3.13. - - :param func: a callable - :param args: the positional arguments for the callable - :param limiter: capacity limiter to use to limit the total number of subinterpreters - running (if omitted, the default limiter is used) - :return: the result of the call - :raises BrokenWorkerInterpreter: if there's an internal error in a subinterpreter - - """ - if limiter is None: - limiter = current_default_interpreter_limiter() - - try: - idle_workers = _idle_workers.get() - except LookupError: - idle_workers = deque() - _idle_workers.set(idle_workers) - atexit.register(_stop_workers, idle_workers) - - async with limiter: - try: - worker = idle_workers.pop() - except IndexError: - worker = _Worker() - - try: - return await to_thread.run_sync( - worker.call, - func, - args, - limiter=limiter, - ) - finally: - # Prune workers that have been idle for too long - now = current_time() - while idle_workers: - if now - idle_workers[0].last_used <= MAX_WORKER_IDLE_TIME: - break - - await to_thread.run_sync(idle_workers.popleft().destroy, limiter=limiter) - - worker.last_used = current_time() - idle_workers.append(worker) - - -def current_default_interpreter_limiter() -> CapacityLimiter: - """ - Return the capacity limiter used by default to limit the number of concurrently - running subinterpreters. - - Defaults to the number of CPU cores. - - :return: a capacity limiter object - - """ - try: - return _default_interpreter_limiter.get() - except LookupError: - limiter = CapacityLimiter(os.cpu_count() or DEFAULT_CPU_COUNT) - _default_interpreter_limiter.set(limiter) - return limiter diff --git a/.env/Lib/site-packages/anyio/to_process.py b/.env/Lib/site-packages/anyio/to_process.py deleted file mode 100644 index b289234..0000000 --- a/.env/Lib/site-packages/anyio/to_process.py +++ /dev/null @@ -1,266 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "current_default_process_limiter", - "process_worker", - "run_sync", -) - -import os -import pickle -import subprocess -import sys -from collections import deque -from collections.abc import Callable -from importlib.util import module_from_spec, spec_from_file_location -from typing import TypeVar, cast - -from ._core._eventloop import current_time, get_async_backend, get_cancelled_exc_class -from ._core._exceptions import BrokenWorkerProcess -from ._core._subprocesses import open_process -from ._core._synchronization import CapacityLimiter -from ._core._tasks import CancelScope, fail_after -from .abc import ByteReceiveStream, ByteSendStream, Process -from .lowlevel import RunVar, checkpoint_if_cancelled -from .streams.buffered import BufferedByteReceiveStream - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -WORKER_MAX_IDLE_TIME = 300 # 5 minutes - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") - -_process_pool_workers: RunVar[set[Process]] = RunVar("_process_pool_workers") -_process_pool_idle_workers: RunVar[deque[tuple[Process, float]]] = RunVar( - "_process_pool_idle_workers" -) -_default_process_limiter: RunVar[CapacityLimiter] = RunVar("_default_process_limiter") - - -async def run_sync( # type: ignore[return] - func: Callable[[Unpack[PosArgsT]], T_Retval], - *args: Unpack[PosArgsT], - cancellable: bool = False, - limiter: CapacityLimiter | None = None, -) -> T_Retval: - """ - Call the given function with the given arguments in a worker process. - - If the ``cancellable`` option is enabled and the task waiting for its completion is - cancelled, the worker process running it will be abruptly terminated using SIGKILL - (or ``terminateProcess()`` on Windows). - - :param func: a callable - :param args: positional arguments for the callable - :param cancellable: ``True`` to allow cancellation of the operation while it's - running - :param limiter: capacity limiter to use to limit the total amount of processes - running (if omitted, the default limiter is used) - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - :return: an awaitable that yields the return value of the function. - - """ - - async def send_raw_command(pickled_cmd: bytes) -> object: - try: - await stdin.send(pickled_cmd) - response = await buffered.receive_until(b"\n", 50) - status, length = response.split(b" ") - if status not in (b"RETURN", b"EXCEPTION"): - raise RuntimeError( - f"Worker process returned unexpected response: {response!r}" - ) - - pickled_response = await buffered.receive_exactly(int(length)) - except BaseException as exc: - workers.discard(process) - try: - process.kill() - with CancelScope(shield=True): - await process.aclose() - except ProcessLookupError: - pass - - if isinstance(exc, get_cancelled_exc_class()): - raise - else: - raise BrokenWorkerProcess from exc - - retval = pickle.loads(pickled_response) - if status == b"EXCEPTION": - assert isinstance(retval, BaseException) - raise retval - else: - return retval - - # First pickle the request before trying to reserve a worker process - await checkpoint_if_cancelled() - request = pickle.dumps(("run", func, args), protocol=pickle.HIGHEST_PROTOCOL) - - # If this is the first run in this event loop thread, set up the necessary variables - try: - workers = _process_pool_workers.get() - idle_workers = _process_pool_idle_workers.get() - except LookupError: - workers = set() - idle_workers = deque() - _process_pool_workers.set(workers) - _process_pool_idle_workers.set(idle_workers) - get_async_backend().setup_process_pool_exit_at_shutdown(workers) - - async with limiter or current_default_process_limiter(): - # Pop processes from the pool (starting from the most recently used) until we - # find one that hasn't exited yet - process: Process - while idle_workers: - process, idle_since = idle_workers.pop() - if process.returncode is None: - stdin = cast(ByteSendStream, process.stdin) - buffered = BufferedByteReceiveStream( - cast(ByteReceiveStream, process.stdout) - ) - - # Prune any other workers that have been idle for WORKER_MAX_IDLE_TIME - # seconds or longer - now = current_time() - killed_processes: list[Process] = [] - while idle_workers: - if now - idle_workers[0][1] < WORKER_MAX_IDLE_TIME: - break - - process_to_kill, idle_since = idle_workers.popleft() - process_to_kill.kill() - workers.remove(process_to_kill) - killed_processes.append(process_to_kill) - - with CancelScope(shield=True): - for killed_process in killed_processes: - await killed_process.aclose() - - break - - workers.remove(process) - else: - command = [sys.executable, "-u", "-m", __name__] - process = await open_process( - command, stdin=subprocess.PIPE, stdout=subprocess.PIPE - ) - try: - stdin = cast(ByteSendStream, process.stdin) - buffered = BufferedByteReceiveStream( - cast(ByteReceiveStream, process.stdout) - ) - with fail_after(20): - message = await buffered.receive(6) - - if message != b"READY\n": - raise BrokenWorkerProcess( - f"Worker process returned unexpected response: {message!r}" - ) - - main_module_path = getattr(sys.modules["__main__"], "__file__", None) - pickled = pickle.dumps( - ("init", sys.path, main_module_path), - protocol=pickle.HIGHEST_PROTOCOL, - ) - await send_raw_command(pickled) - except (BrokenWorkerProcess, get_cancelled_exc_class()): - raise - except BaseException as exc: - process.kill() - raise BrokenWorkerProcess( - "Error during worker process initialization" - ) from exc - - workers.add(process) - - with CancelScope(shield=not cancellable): - try: - return cast(T_Retval, await send_raw_command(request)) - finally: - if process in workers: - idle_workers.append((process, current_time())) - - -def current_default_process_limiter() -> CapacityLimiter: - """ - Return the capacity limiter that is used by default to limit the number of worker - processes. - - :return: a capacity limiter object - - """ - try: - return _default_process_limiter.get() - except LookupError: - limiter = CapacityLimiter(os.cpu_count() or 2) - _default_process_limiter.set(limiter) - return limiter - - -def process_worker() -> None: - # Redirect standard streams to os.devnull so that user code won't interfere with the - # parent-worker communication - stdin = sys.stdin - stdout = sys.stdout - sys.stdin = open(os.devnull) - sys.stdout = open(os.devnull, "w") - - stdout.buffer.write(b"READY\n") - while True: - retval = exception = None - try: - command, *args = pickle.load(stdin.buffer) - except EOFError: - return - except BaseException as exc: - exception = exc - else: - if command == "run": - func, args = args - try: - retval = func(*args) - except BaseException as exc: - exception = exc - elif command == "init": - main_module_path: str | None - sys.path, main_module_path = args - del sys.modules["__main__"] - if main_module_path and os.path.isfile(main_module_path): - # Load the parent's main module but as __mp_main__ instead of - # __main__ (like multiprocessing does) to avoid infinite recursion - try: - spec = spec_from_file_location("__mp_main__", main_module_path) - if spec and spec.loader: - main = module_from_spec(spec) - spec.loader.exec_module(main) - sys.modules["__main__"] = main - except BaseException as exc: - exception = exc - try: - if exception is not None: - status = b"EXCEPTION" - pickled = pickle.dumps(exception, pickle.HIGHEST_PROTOCOL) - else: - status = b"RETURN" - pickled = pickle.dumps(retval, pickle.HIGHEST_PROTOCOL) - except BaseException as exc: - exception = exc - status = b"EXCEPTION" - pickled = pickle.dumps(exc, pickle.HIGHEST_PROTOCOL) - - stdout.buffer.write(b"%s %d\n" % (status, len(pickled))) - stdout.buffer.write(pickled) - - # Respect SIGTERM - if isinstance(exception, SystemExit): - raise exception - - -if __name__ == "__main__": - process_worker() diff --git a/.env/Lib/site-packages/anyio/to_thread.py b/.env/Lib/site-packages/anyio/to_thread.py deleted file mode 100644 index 4be5b71..0000000 --- a/.env/Lib/site-packages/anyio/to_thread.py +++ /dev/null @@ -1,78 +0,0 @@ -from __future__ import annotations - -__all__ = ( - "run_sync", - "current_default_thread_limiter", -) - -import sys -from collections.abc import Callable -from typing import TypeVar -from warnings import warn - -from ._core._eventloop import get_async_backend -from .abc import CapacityLimiter - -if sys.version_info >= (3, 11): - from typing import TypeVarTuple, Unpack -else: - from typing_extensions import TypeVarTuple, Unpack - -T_Retval = TypeVar("T_Retval") -PosArgsT = TypeVarTuple("PosArgsT") - - -async def run_sync( - func: Callable[[Unpack[PosArgsT]], T_Retval], - *args: Unpack[PosArgsT], - abandon_on_cancel: bool = False, - cancellable: bool | None = None, - limiter: CapacityLimiter | None = None, -) -> T_Retval: - """ - Call the given function with the given arguments in a worker thread. - - If the ``cancellable`` option is enabled and the task waiting for its completion is - cancelled, the thread will still run its course but its return value (or any raised - exception) will be ignored. - - :param func: a callable - :param args: positional arguments for the callable - :param abandon_on_cancel: ``True`` to abandon the thread (leaving it to run - unchecked on own) if the host task is cancelled, ``False`` to ignore - cancellations in the host task until the operation has completed in the worker - thread - :param cancellable: deprecated alias of ``abandon_on_cancel``; will override - ``abandon_on_cancel`` if both parameters are passed - :param limiter: capacity limiter to use to limit the total amount of threads running - (if omitted, the default limiter is used) - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - :return: an awaitable that yields the return value of the function. - - """ - if cancellable is not None: - abandon_on_cancel = cancellable - warn( - "The `cancellable=` keyword argument to `anyio.to_thread.run_sync` is " - "deprecated since AnyIO 4.1.0; use `abandon_on_cancel=` instead", - DeprecationWarning, - stacklevel=2, - ) - - return await get_async_backend().run_sync_in_worker_thread( - func, args, abandon_on_cancel=abandon_on_cancel, limiter=limiter - ) - - -def current_default_thread_limiter() -> CapacityLimiter: - """ - Return the capacity limiter that is used by default to limit the number of - concurrent threads. - - :return: a capacity limiter object - :raises NoEventLoopError: if no supported asynchronous event loop is running in the - current thread - - """ - return get_async_backend().current_default_thread_limiter() diff --git a/.env/Lib/site-packages/click-8.3.1.dist-info/INSTALLER b/.env/Lib/site-packages/click-8.3.1.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/click-8.3.1.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/click-8.3.1.dist-info/METADATA b/.env/Lib/site-packages/click-8.3.1.dist-info/METADATA deleted file mode 100644 index 3f433af..0000000 --- a/.env/Lib/site-packages/click-8.3.1.dist-info/METADATA +++ /dev/null @@ -1,84 +0,0 @@ -Metadata-Version: 2.4 -Name: click -Version: 8.3.1 -Summary: Composable command line interface toolkit -Maintainer-email: Pallets -Requires-Python: >=3.10 -Description-Content-Type: text/markdown -License-Expression: BSD-3-Clause -Classifier: Development Status :: 5 - Production/Stable -Classifier: Intended Audience :: Developers -Classifier: Operating System :: OS Independent -Classifier: Programming Language :: Python -Classifier: Typing :: Typed -License-File: LICENSE.txt -Requires-Dist: colorama; platform_system == 'Windows' -Project-URL: Changes, https://click.palletsprojects.com/page/changes/ -Project-URL: Chat, https://discord.gg/pallets -Project-URL: Documentation, https://click.palletsprojects.com/ -Project-URL: Donate, https://palletsprojects.com/donate -Project-URL: Source, https://github.com/pallets/click/ - -

- -# Click - -Click is a Python package for creating beautiful command line interfaces -in a composable way with as little code as necessary. It's the "Command -Line Interface Creation Kit". It's highly configurable but comes with -sensible defaults out of the box. - -It aims to make the process of writing command line tools quick and fun -while also preventing any frustration caused by the inability to -implement an intended CLI API. - -Click in three points: - -- Arbitrary nesting of commands -- Automatic help page generation -- Supports lazy loading of subcommands at runtime - - -## A Simple Example - -```python -import click - -@click.command() -@click.option("--count", default=1, help="Number of greetings.") -@click.option("--name", prompt="Your name", help="The person to greet.") -def hello(count, name): - """Simple program that greets NAME for a total of COUNT times.""" - for _ in range(count): - click.echo(f"Hello, {name}!") - -if __name__ == '__main__': - hello() -``` - -``` -$ python hello.py --count=3 -Your name: Click -Hello, Click! -Hello, Click! -Hello, Click! -``` - - -## Donate - -The Pallets organization develops and supports Click and other popular -packages. In order to grow the community of contributors and users, and -allow the maintainers to devote more time to the projects, [please -donate today][]. - -[please donate today]: https://palletsprojects.com/donate - -## Contributing - -See our [detailed contributing documentation][contrib] for many ways to -contribute, including reporting issues, requesting features, asking or answering -questions, and making PRs. - -[contrib]: https://palletsprojects.com/contributing/ - diff --git a/.env/Lib/site-packages/click-8.3.1.dist-info/RECORD b/.env/Lib/site-packages/click-8.3.1.dist-info/RECORD deleted file mode 100644 index 008d697..0000000 --- a/.env/Lib/site-packages/click-8.3.1.dist-info/RECORD +++ /dev/null @@ -1,40 +0,0 @@ -click-8.3.1.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -click-8.3.1.dist-info/METADATA,sha256=XZeBrMAE0ghTE88SjfrSDuSyNCpBPplxJR1tbwD9oZg,2621 -click-8.3.1.dist-info/RECORD,, -click-8.3.1.dist-info/WHEEL,sha256=G2gURzTEtmeR8nrdXUJfNiB3VYVxigPQ-bEQujpNiNs,82 -click-8.3.1.dist-info/licenses/LICENSE.txt,sha256=morRBqOU6FO_4h9C9OctWSgZoigF2ZG18ydQKSkrZY0,1475 -click/__init__.py,sha256=6YyS1aeyknZ0LYweWozNZy0A9nZ_11wmYIhv3cbQrYo,4473 -click/__pycache__/__init__.cpython-312.pyc,, -click/__pycache__/_compat.cpython-312.pyc,, -click/__pycache__/_termui_impl.cpython-312.pyc,, -click/__pycache__/_textwrap.cpython-312.pyc,, -click/__pycache__/_utils.cpython-312.pyc,, -click/__pycache__/_winconsole.cpython-312.pyc,, -click/__pycache__/core.cpython-312.pyc,, -click/__pycache__/decorators.cpython-312.pyc,, -click/__pycache__/exceptions.cpython-312.pyc,, -click/__pycache__/formatting.cpython-312.pyc,, -click/__pycache__/globals.cpython-312.pyc,, -click/__pycache__/parser.cpython-312.pyc,, -click/__pycache__/shell_completion.cpython-312.pyc,, -click/__pycache__/termui.cpython-312.pyc,, -click/__pycache__/testing.cpython-312.pyc,, -click/__pycache__/types.cpython-312.pyc,, -click/__pycache__/utils.cpython-312.pyc,, -click/_compat.py,sha256=v3xBZkFbvA1BXPRkFfBJc6-pIwPI7345m-kQEnpVAs4,18693 -click/_termui_impl.py,sha256=rgCb3On8X5A4200rA5L6i13u5iapmFer7sru57Jy6zA,27093 -click/_textwrap.py,sha256=BOae0RQ6vg3FkNgSJyOoGzG1meGMxJ_ukWVZKx_v-0o,1400 -click/_utils.py,sha256=kZwtTf5gMuCilJJceS2iTCvRvCY-0aN5rJq8gKw7p8g,943 -click/_winconsole.py,sha256=_vxUuUaxwBhoR0vUWCNuHY8VUefiMdCIyU2SXPqoF-A,8465 -click/core.py,sha256=U6Bfxt8GkjNDqyJ0HqXvluJHtyZ4sY5USAvM1Cdq7mQ,132105 -click/decorators.py,sha256=5P7abhJtAQYp_KHgjUvhMv464ERwOzrv2enNknlwHyQ,18461 -click/exceptions.py,sha256=8utf8w6V5hJXMnO_ic1FNrtbwuEn1NUu1aDwV8UqnG4,9954 -click/formatting.py,sha256=RVfwwr0rwWNpgGr8NaHodPzkIr7_tUyVh_nDdanLMNc,9730 -click/globals.py,sha256=gM-Nh6A4M0HB_SgkaF5M4ncGGMDHc_flHXu9_oh4GEU,1923 -click/parser.py,sha256=Q31pH0FlQZEq-UXE_ABRzlygEfvxPTuZbWNh4xfXmzw,19010 -click/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -click/shell_completion.py,sha256=Cc4GQUFuWpfQBa9sF5qXeeYI7n3tI_1k6ZdSn4BZbT0,20994 -click/termui.py,sha256=hqCEjNndU-nzW08nRAkBaVgfZp_FdCA9KxfIWlKYaMc,31037 -click/testing.py,sha256=EERbzcl1br0mW0qBS9EqkknfNfXB9WQEW0ELIpkvuSs,19102 -click/types.py,sha256=ek54BNSFwPKsqtfT7jsqcc4WHui8AIFVMKM4oVZIXhc,39927 -click/utils.py,sha256=gCUoewdAhA-QLBUUHxrLh4uj6m7T1WjZZMNPvR0I7YA,20257 diff --git a/.env/Lib/site-packages/click-8.3.1.dist-info/WHEEL b/.env/Lib/site-packages/click-8.3.1.dist-info/WHEEL deleted file mode 100644 index d8b9936..0000000 --- a/.env/Lib/site-packages/click-8.3.1.dist-info/WHEEL +++ /dev/null @@ -1,4 +0,0 @@ -Wheel-Version: 1.0 -Generator: flit 3.12.0 -Root-Is-Purelib: true -Tag: py3-none-any diff --git a/.env/Lib/site-packages/click-8.3.1.dist-info/licenses/LICENSE.txt b/.env/Lib/site-packages/click-8.3.1.dist-info/licenses/LICENSE.txt deleted file mode 100644 index d12a849..0000000 --- a/.env/Lib/site-packages/click-8.3.1.dist-info/licenses/LICENSE.txt +++ /dev/null @@ -1,28 +0,0 @@ -Copyright 2014 Pallets - -Redistribution and use in source and binary forms, with or without -modification, are permitted provided that the following conditions are -met: - -1. Redistributions of source code must retain the above copyright - notice, this list of conditions and the following disclaimer. - -2. Redistributions in binary form must reproduce the above copyright - notice, this list of conditions and the following disclaimer in the - documentation and/or other materials provided with the distribution. - -3. Neither the name of the copyright holder nor the names of its - contributors may be used to endorse or promote products derived from - this software without specific prior written permission. - -THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A -PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED -TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR -PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF -LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING -NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS -SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/.env/Lib/site-packages/click/__init__.py b/.env/Lib/site-packages/click/__init__.py deleted file mode 100644 index 1aa547c..0000000 --- a/.env/Lib/site-packages/click/__init__.py +++ /dev/null @@ -1,123 +0,0 @@ -""" -Click is a simple Python module inspired by the stdlib optparse to make -writing command line scripts fun. Unlike other modules, it's based -around a simple API that does not come with too much magic and is -composable. -""" - -from __future__ import annotations - -from .core import Argument as Argument -from .core import Command as Command -from .core import CommandCollection as CommandCollection -from .core import Context as Context -from .core import Group as Group -from .core import Option as Option -from .core import Parameter as Parameter -from .decorators import argument as argument -from .decorators import command as command -from .decorators import confirmation_option as confirmation_option -from .decorators import group as group -from .decorators import help_option as help_option -from .decorators import make_pass_decorator as make_pass_decorator -from .decorators import option as option -from .decorators import pass_context as pass_context -from .decorators import pass_obj as pass_obj -from .decorators import password_option as password_option -from .decorators import version_option as version_option -from .exceptions import Abort as Abort -from .exceptions import BadArgumentUsage as BadArgumentUsage -from .exceptions import BadOptionUsage as BadOptionUsage -from .exceptions import BadParameter as BadParameter -from .exceptions import ClickException as ClickException -from .exceptions import FileError as FileError -from .exceptions import MissingParameter as MissingParameter -from .exceptions import NoSuchOption as NoSuchOption -from .exceptions import UsageError as UsageError -from .formatting import HelpFormatter as HelpFormatter -from .formatting import wrap_text as wrap_text -from .globals import get_current_context as get_current_context -from .termui import clear as clear -from .termui import confirm as confirm -from .termui import echo_via_pager as echo_via_pager -from .termui import edit as edit -from .termui import getchar as getchar -from .termui import launch as launch -from .termui import pause as pause -from .termui import progressbar as progressbar -from .termui import prompt as prompt -from .termui import secho as secho -from .termui import style as style -from .termui import unstyle as unstyle -from .types import BOOL as BOOL -from .types import Choice as Choice -from .types import DateTime as DateTime -from .types import File as File -from .types import FLOAT as FLOAT -from .types import FloatRange as FloatRange -from .types import INT as INT -from .types import IntRange as IntRange -from .types import ParamType as ParamType -from .types import Path as Path -from .types import STRING as STRING -from .types import Tuple as Tuple -from .types import UNPROCESSED as UNPROCESSED -from .types import UUID as UUID -from .utils import echo as echo -from .utils import format_filename as format_filename -from .utils import get_app_dir as get_app_dir -from .utils import get_binary_stream as get_binary_stream -from .utils import get_text_stream as get_text_stream -from .utils import open_file as open_file - - -def __getattr__(name: str) -> object: - import warnings - - if name == "BaseCommand": - from .core import _BaseCommand - - warnings.warn( - "'BaseCommand' is deprecated and will be removed in Click 9.0. Use" - " 'Command' instead.", - DeprecationWarning, - stacklevel=2, - ) - return _BaseCommand - - if name == "MultiCommand": - from .core import _MultiCommand - - warnings.warn( - "'MultiCommand' is deprecated and will be removed in Click 9.0. Use" - " 'Group' instead.", - DeprecationWarning, - stacklevel=2, - ) - return _MultiCommand - - if name == "OptionParser": - from .parser import _OptionParser - - warnings.warn( - "'OptionParser' is deprecated and will be removed in Click 9.0. The" - " old parser is available in 'optparse'.", - DeprecationWarning, - stacklevel=2, - ) - return _OptionParser - - if name == "__version__": - import importlib.metadata - import warnings - - warnings.warn( - "The '__version__' attribute is deprecated and will be removed in" - " Click 9.1. Use feature detection or" - " 'importlib.metadata.version(\"click\")' instead.", - DeprecationWarning, - stacklevel=2, - ) - return importlib.metadata.version("click") - - raise AttributeError(name) diff --git a/.env/Lib/site-packages/click/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index f259bb12f1aff344db20397d78827ae2513b5806..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4064 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 diff --git a/.env/Lib/site-packages/click/__pycache__/_utils.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/_utils.cpython-312.pyc deleted file mode 100644 index e0c1fd53ee0f0752b6631f6dd33e3ee87a92af2d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1192 zcmZ`&&rcIU6rR~1Z7Btz6^I&j<+cX4m?#i5Mxzx;N;Lu|Y?_SA?f_f1yLD$vEb+jB z7>Nfo9yl295D)$X9!xxXDJOA_i3ulf7&vkAO_!P)<0Sj$d+&Si?R#%#K6i8^5v)%i zpV{w4g!Z`7Tyg~P!3O}F$U+u&P#NnOb1XQ5E;M#gmjpD9EO8B45*12~jLjs`{tj~U zrN?Nj)EVJcsmI$8FkP*-2y#@+{|8 z%Sx3|O9?#1qVr%?*fFat8%~unEtyncX&MmJaF{nX7FoFU(f!@GMW6G|WZn+Do)psTC(})AxW|woS^48g-Y8d3&zN?0^nd40Fj?0JG`X z=2Dha1KZ&v;J@=}#V2H?yE!hcY2sIh^@CVNyXoFd?UnW=t?s1NdiTYx3tP9}Pi#+Y zU9F!OsV7Dbcv}Tk2xC4Cs=gbFEb!HY-v;I|Nl4kVst#`|Mabi-;T*RZC&cnhFvq;P z5;cQRo+(V_ro*l~`J3aUkjvekA`froXL5cAB-%zke*u1q3;xC$A`XC^1+j*H39_8{ z)rmSXFYd19zV@Ac{iH6PJe06J8b$ITHN^{>uR@tnkk1b{lqlS~Fc#D*w!0AGAe8tu zjQ-IB4t_89isxaJzTt%NX%f|$o+kfP+Q)OnP+7WIDJSAgKB$oTWw7cs`)QU4xl z|Ax|su@oMC9&FUsYljHn{yEgvz0tYexhn6X(I1&JUorzbnSna$+7~1&?Wa(CYC~OD jU*>j_r&r^<=*o|t-p$!pv-R`Yot~jOO7Ef@jUs*nCMFm} diff --git a/.env/Lib/site-packages/click/__pycache__/_winconsole.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/_winconsole.cpython-312.pyc deleted file mode 100644 index e98dec6cd58851a8324b474b1a223e6e84a54b0f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11757 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 diff --git a/.env/Lib/site-packages/click/__pycache__/core.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/core.cpython-312.pyc deleted file mode 100644 index 1ecd550140cff21f3c9196ab2b42629487514828..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 134412 zcmd443v?XUc_!FTpn-0n8$jbt5RC@`Gzm7}6bVw4AONH&ijXW)R0K%@n&<}EAkcuT z8>9$>HYqD(KxQJ+mP2SPM^xgorpB6>-jiJ`J7;&KadzXA%%fYN+^r6G0&lc4NzTp& zGL^N>WajMm|F>>cb)!JaadLKxVAbuaTle0&_rL%9{`XJI%1SLbe)!%$iz)5(w^*)OB9@4C!ZK+cvRWx^o3KsVha76!IppMfMMFh=EgmZ7t82)`*OH+U zzLpNTakWo)@UETTbs$|fRK{uNMERt5$jj*>qxb5J zx?-YvGB6b2bS2U)LoJ-HLb`RRmD7Ht+lJaWU5#}6P&=ngM#iiOrLtp%AB=klr%1h12UNwoYyv+Q#W- zq_+=k=X3z+9YZ@f-7>Lra@Wu=mAc*BfV#652rUEy?1CYr#q1D z8S1fGqVDmt5)AaKQiMM2oY+74;Lw9si;Z&4=}%=U(iI85v+=t&@q6f@HS=xyu2s$R zaEV3#7kMnQ`DMpYZ;2&xM87Ljk;fxjUv>^X5;+>#hFpD7AMy>#r$|2$+5WQgyJ};G z`t>Jx_GDxS-Wjl24p|~QU$#VcMV;f9WxMt3cLDEo3?1T}yOFb7`<_Sn-UGO|MZ0&H z@9n|8z0t?Cw~z4M9^Bm*Jsdsq&i?N@)N&u^ClBIDD0*1Sf0XYYz}<(UNAvOz@{@=0 zq&Iq0%l|~=bCE~zJx}t}K0NK$TJ$--H-LKwqo32>c#7{H!rey=caKG$iX28c$Ish> z$Nq|%7_{cx;dp#18BWHg;)y~0wueuS;!@l5AQ_&<9frqZRo-R_N)6p>9 zEjx~iq9^hG@$=KszaqH7U1t(M5^q~`zJrhV9vU7T80bGXe4_X9rw8ca(}Twbj^m-T z_vDm9g?JtbN1hBT;mN4@fbzbH*yx#o7e}Mh^qI)vpl0IgI~q$QV)0YP+!cdUy~?S? z;l!iSiRl4FnNm=8>C*{(GylSCW%TIalrkB{*H9OE#;0QO;VCXYf#2@wsp(<0bLjE( zOyacRciHe0oE6n3`2Eu1#PmcgNhKhAG2R@DDU(PRO)FEAeAAgoo}Y-~mVY>c?;1@G z8a|EoI zSmx|neU3TDMeF5_bB^)y)j1XG_nG=LWaVI>;xOa@NJB9Ul3RLU89rV083qsbS00*B*gr$%{t2ULIzj7=$AJ=Yd8 zzMt+mGWmL=(BQeAOQI!S0p|l=*-#C>|CkdlALoAfE&v>2+oz==+2+skgLu zkM;dBKb>1Wbly^ev$6{hX`*q@j%rf{cd2_o*{pZoMdBJw9$X6##g|m6AEuX0pK21w z6V~$pd1ZaXI%oZ+En+jDf_P&8mh*xmsZGB*hn!3=StAa9*SY4qMQgtsEFPR~Kb#E2 z5&_Vs!1NRlHO3>5oC<^^5gmR5vz2@gnNCL)bq3AWib;f*WAVV~=`hZdQ;}$B)_0I_ zl=1bJKsaHTiK`0>%!bw|EpYra)gW4!n390SrUV z^At#HY*HmB&Yb6DIH3Vr&NVt2VVsVcFd50Yl;|l;p{SB8o{T1l3hCglJVeD+4Fl*$ z&kV<-$QHp5e64Z>PY&aM;$@sJT2_iIp1O>?=>vDuw})=p|KRCc8}?;g`f8XiAp3EBs9MWAf4(d5CPBj+GAl(WyqrWFqg$~k}pl%Or=qAQWI zTMDIHJzH%J5UcmoED^3?Jx82^??5~}8qyKu{HwSM#wp9H< z%6ITRXWDmg$$cp0IHWv|Yu=kLMx3{{S(3Ofj&_R@Vwz?` z6E7CeVY&`$baT#nsR|T}N-sE&(~6v0884x>3$`!X&RLThEk#ZBpS33a{d1NV?ax}y zNt$**CH4W84HT0FtPvsOL;&ob62>$H)d9Z*MuKKoR5D|U&|uJ^RHN<6KAdt+CS6J+ z(m^}lQclo)YcxUV3OHX3i$6SbYBCy64$sh(7>*L3mQa2TS;F|ApcCWhl~T8}{I;v? z=eOOpKeafWa1^3zil?RFUi~YTg_V*i|ln&VO zd=*IW1lWjjzCx0>ecm!=jaXmyT(Ie^5e?%zHVNtgZAaLHQGF_!oKfNlW+o!h(FrgD zkpTA|vAvA=h{u>xB0!5soezNbo0*6Z^#E-G?0W8WZ1gm<5?s}}DBg^L^*9rabQ6L-7X{CQE4>uOh$+BU%=AP7 zC5C1j8bq(FoQwJqU}O_$dvr7kUS|~Fc2<0v+I9h5oHrCW9EuX!7MzoKUpA8)F$2zk z`r&LWf)7K+NrFpiUf<(~38yR3#5CA!Lm%Ph$>{0u*%emRXUi7mJVMiWE1FE8A>$|~5d#0QB^uuXngOJ= zB^)_Rv?H>mO_sh{FMTtA6_+`x;R8St;esl{G_y#2!+H%O_kwNQm;?9e&@t%+t2v4G zSmvx_wqrnCNezzxXOvqlI%bY&SoebUS21)Gta-CGuj0x* zf^2|jX#WJjXCiSRBy+!vZ%!P<3B22i&0f-+tq*4Ex1{U0+;l9}@0#yhaasIz*B-q3 zV9Fb~-j?!iNVzur{L^BKui>W_Yf1C%%GyQe*SlZo1C1+bzMCLQ^5u#)%kTMn?0;zO zu|r<6qfx+TBY2W4c~Z)tIS9BkS!XMgGvF?tJ2^Eq(H-C(Ai1fC8W@cL-<_oQD#T1= z2ohhfjN(3b!HC3v4Qqe6LhK$<6XqZBt$!ORa&~;`vv?{~m2EfbZn?Lo9NPtjLN4u4 zN7#Dj-{96YOVl!CjaY_k5$lj0svXC8sg_eqXzG`u5>gZ0J5TXv$m9A9c}hM*o>Hhd z>^ZMC0>`Fil+kGTdCUUIvxVb<9$?KxqG#lb(fG_{XaJ`XU;qeTfY%sM53mg(gap6L z5*p7o4as&UgdnCOEeIlk9w;47_l$_itrEr%2|C(K!=gCUG!?)Ds)0*YYa!t*tRuw0 z$@2jceAFsJp#X4R0#p=i9yBPCo}R#-&{i%0v@yUS zkpQ)KO45P*72gXXQ!vjXBTqi{#L*{@qXa^CXdhKS7cvx6`%KV5wSn zrTtp)YVhluE*{A`da{*`#@`3B<+byXS7QsYub#PhDC^i=@Vqx$Rx^M6)n^x;{pxcU z4`v+?i(08dnw{FAa$?pybC^&1EP5K;*N{Exfco4PEsEIp;{~J2a&XYM2cQ^X!Ers(itg+DzFqvTo)QWM!0slv53 zvOVI*wJusezEkUCEddUCqh5AoN2J=&&c-$0*=cxZSER=H4!(AGq!xYF7g z%RngyA_3Id8hI$vf@@pk;Ycg4?UCL{8?GD9JA#iW+t5+rH~ob3qMG&xJV`h}tW7LD z0YnG-8AoF3B=H{t`<p-Zc=%i_0?v^Km6H+l4$EGXWGKPm;RhAu1XV#25kHrzmYW zFsq>Zfkp*G0+d4XOLstgH+MSfyiZZ+S|AL(31SOP4HP13S@~ag-_BM+6a9qGf|5Qa zwSA&Zq8jd~$=IpW_iGht`4dyX$BFP*Gwqj5+ef^u_Ne#`;?=OxMXlZ<;eC$ z(^WNQBojpor)Q znkh;+hJFW3vvwB=_1XRU=M3!p5=UjJ>FW!2Q5Q6r>Xl!?z0pvDW9s z7|w=g;%DMuHl=?9z794mWd`a>aF>K%i8JS{#JmH1z&n7Q1MU7SA4s1=<@5NSiAJZH z+)OgfoPY?j7S|_HJYNlJKjZiiZdaSloGUP_Xd&@|Ans%pB05WkBEo&F{1HSiaC++8 zeVR|*CVQ1?GH7iuBZ!w2e9ryGN0xz?Mn<0N+`O3z>wIBkByaj9p-Ybs3tzdK2i7BOr85+oPfIh*&Ya~*|W{E)W z;IYHVhTO0{D8yL`uFB-&41+>kj_|nwY)dWxgT-t_JPKn3Yyo3jxJf~yc$Y*^Rr({* z7MM|mH9>=>0e!f`HWb3qR*A?%u#Yx^B6Kqz5M++oC2F@?5Qz!&Q!uAm5gNW8 z8R-+^Y{=L!s2f%OiMdi#CEG_`fbz70Si>;}DM-WH{K#Cxxm|nhIGQvv0(3%*tBTH1 zDXf`F?Bom-B3vcepn#f$DnYWgN^}x!11Q9`(!}uY4Q&mfQ1Z({QJkGVlbjOp4lcyj z%*2_^p!rHO@SghKZCVAQxdlChf`EGf{nS{U-86#LTD^-A!=#0xLrNbicM7K;p**=~ zyL3JDecGiC+8QvSeVu>}p4XT%_o(`Tkk%ze74=&Ea*vXB8kFhAq8Mq6jC5lh&8Z?i z1F6>9J~FaOtij-GLvybI-myj-3K=khOE3q3EauEsu^}TPT5CR2z1!EQcNJqT`e7Au zUEKv{G^?N&hycx2k%Q-RkAmzSlwOrDD!ozZyl~{wL6`JDL;N%4NI_TNw;Gi_D zJ%-8LV(^i$HIL)!ADt{7dsWnX@U?NzE4!UKS?ae&1E?s__y% zr_{2eWZi4Uaj|^PcFBWMe6nu+dI27V$nBDMlCa;3V-EJWm_<7FB9CT;n=87wY0iEr zh*CRIYK?qDzg}uIzdv5D=aDIRH>zK9p=hoM^|xJSu_@@kO z{WRy&aVkXDjhq^! z%y~FxyL_%+FL>rW$XO}x%#~fNL|>Mn&pPBi{d%D+ZjabbSud2$mEkRYe9E;i?9y|~ z)P?f7a^!{vUL12J_(jexe9Sp7_M^t9P-C~er(Z7}giT(9OvC*i^Zeh0`-u4)<6HE% zWlF;6H!a_?UvS2KbIyC^y5O91UaCYMvM3ZB5l2M#j72_tp$3a{Uafbw7rcg_7rb*` z-gEg-}K?03U4DkF)D(u0+#<^ZoE-0@h0q_MDeo4-8C0&JG4BS4Pr@ zs<4eFa(1YBbL0e1{6x+9W)|%{wl;muuJK;#sb5#ibm+n88haySW9Il9f`+rp@z? zTds9K>3{6oi5t5!n;uGUdgxZ?!?&J1w)EuF^r&IkT|U2Wad)b5bGEYX+QzFJ;jL0+ zX{2&pw>v{O&fYxxuCmnm2!5BsjiszEQlZASzk@c*NSt-*P8aVrhK90hCrsF zJKfNoY1o=>*m|>msbSyzk>$QYs;O?J&a!0(rJ9%B^(lAD^|4gPo@`TFrfGY+Y5V*j zs@=4go;NPL*QMO;H)>MBec3=qCa@BgDHOqMc1v* z)NW1JZcWu}yIFB_=MN5i|3GHPq4bVJOFIsywm+8gANj;?sc1tTtvfRJyK7VK=IaMj?Ypy$Et$qpx-m5WI391@P4Ct(Kgb>4l&z^nc6V0%Tz|W<{hRyd z9Ur>t*jZszrSO1j1ON0Ce!%q*zCIHc_<>o+HQtEx@hM_(5Rcg3aY%l|3f{nMWd@uS zor8=&XQw*z@nYv^lAu%tHcL*OgkL*9tK9Tggtg&`MJo+P&#)B`9%!y}VTHsjU~5X{ zD*K~ChJ!@^jCiDc87~HH%C~S)eh;TSH{kO8ub8co^FIInzoOYsUr6XI>KGhaR%9FkL5I4$ucKf z!5^fhHX#^jnK>YfVOLNuP**v?q_N6lD_A55*G2SWAyp#k6)YBvhsRjFpl}5=jB>wi zW%y0dOc$Qg$_*rE{SsV_rTj4g>k&K#scNv)t-CgNbxt6+oBpNR-P!u4H`l+uJ{xGy z1a_qZyRw~|GM#MZtknv0x;zVp`AUP;icuHMWaFFviWHa>%7TRdispMtRB@SNiz+EDkRy+3# zEoWlJioh8hle+$qp;RGx>}@AQOU{L7!z6;|N+`t|x|}U}GUsMC3#=YiU==PqfuA2g z&EtW@ou5$eK7tAWSRPABnW2BPZJqOOS`xKyVc%lQ2kypf!}?6a-gLv>ce~y@{3B(l z;i>r}w_R12%a*+jDOZCS0>bI?C_qT7H+kWk-YwO{nZ)dg3QqDbo5ObnSpy2Sg;odh0@K zC#D7wmN9te2Vgmm=zd%DiLBd3uc~dQ?;_^|>Hj(^*nZjD^a-l%79Jze4#UkrYZTcs9JP=|@v7m#jHinr-ZY1a69LWHG534j* zPGe~xoCA@cmV&d9KsEQUg$tmGF%niXISusLW^50o2YI2NC<#y{{k<64F)M*5xRs(? z^`}bPi>p5AvG{7U9^b1+ zUOBS3E$vzNn@3h`&XSgoDykPNvo&>JbN$S2sjT^=$l`Cz`ux}2Uvn>hKJ5$qwi}BI z{f$2>#=|TAzgz41-zP}j^io}KH&&qbdM)2__S)^g>q6>Z)Z^a!c4x1r_SLd*j9;0?TPaBiaoS_XVX7#?d` z1Ttm&DWOrkPF}3>*ypkOJTiso2k_AmwsSNIx_dSs(UK+YBg6^-7(SAxMPXRwJ~agg zT(&DSVun|5An=T8v*vyvBN}}ZeYxJXG5Bu@Z)6pp!x;}$Q7pBB6`#8mMLnu^$~m=2 zn3J)<2J_Tdn4wQ78>^?#A>`Z3^@byK(JWL#I)Ote40ez+MrrMu_Kh^%66awzn>6&` zJu&*@y@Q8_j|@B;crY;5`31RR;zDQwi#j@W3`w&cGn&ZZFg=<9ap+?9%!CF2$D=q? z5}>t|3Z#RG2`L|+lMQ(RuX4ewE2k2*%1!DOYn#j~=1OS&OP8#uJJ1O?hNq8p-cv#% zIw4vb2moWvWAYX<_{tn^&Uz$zqOYk2pO_%Q0DR|8h}W;qF3hG}^{ZtPV(Bs6t#`!I zW3b+YCkhVSHS(S6Q{1Ju3yycRrN}G-BWjMN53H43L8L7-5+Isq=o-<0nV7sh8zM6U z0coq(I^_6N0F)cbjq`!g8O)bSxPveT&^34!C-l`f*1z~p0b-@ zj0@=@-E_~0R+ccn8BE{?7}p8B`En#0Lfwk`8A{g1+?I2)D`>8S^xT{T$)XjF22F0E z$XCwsOthBkN}Bxp+XzfHqbPuhU(>>5D{C^98`6~4fqv7Yo^IHSA&nINwp#u9|(qb zB6cz~AY=R>rUnVvYEdI_2%*&%6Y#)64O%I(IamrZGNMV~T|pRvRe6T2*aC|P1`BUM z;0%E#M0L(AztIQ?L5*b(6BcsCo2>Nf)>y0iHHC}_!-kd^z8#ZHMqWUd7qX%}XNfn& z6oL%F#3YuWKrCVHyJZin%zigbN3A!}OT$xfA)lk;o>2IyR-#r2yC3#|h+<8lHUcaS z(SlL}4u~ZPiJ^_BS{uX~B_UlVXjXj|wb#P}J#kzZXt3mDq^tT!xP^7KZr)7pZLOwx zXzh)JiM%q1lA(q!5FZmVLL>?1G0-7O5rhv*$mznT$01V}OdNEIR_s1)!*KYJe0S37YEnY|0xz54p0u zxuWWsb6&NHva&p#b6H*|%vSe_Z_Mu-gk7^F$P5Z(wwsYzbK2XS@or9gH{Y0A@*YUJ z4y;z9*#D7G*KYui8BmuLs#HdiZ~@THDx3ocBRJq07BCTnG4HL)K#c1!T0FBgdH`>flGVMyMFqg26+sI9DVQl$BqF@& z??sUTa(Q^&+=rEOc`*{OQ2?8l#sw?g7x>2#@-IQ`|TFq2xy3vi2 z9GWmyuruMG)MrxU6GX*&mKxeUgf$}}13e?>U z0<*0W1+T4GpCSMVp#T6touwA>zB6$G!UGE0pduplX(RugB!ZZteyHU?= zoG_&42Aa0s`8Mv7ftG`G;AY_gr!rH+0_R6-3Xezkk<_(LEEv)vG?cpKUIjEhe;{={02xHQG z6;U4kEq>`HO=Ip$H2Y}!V)l6~<-U0v*_sY49Drj-wxasl`m5{bA5qOuDR(QZQQz73 z*1k;JzI5BZRO@~?&3x>wUu>sfewl{N>4wekoV(?HAluaP=5wzn|iMdWc{5Pe<#d5 z1gym%%n4XiyA!xa7n|czEU=d=htSa@h8_gN%nA{0T{!4ENx6g?=m4;OgeJ1@pK9Sy zvUG#cOMcHo9!tP6WLWKcO}NUt{T=H4ZoCV0;j=*rna(Op7_EyE;56V)@#XN&{=6`L8R*fK-I3Yx-TCb$u2k6nC(<3Jv5en$_GM=mjPyZ85u}6>ruMCRn%qV^ z;dPnux2OH>8Gl#W-*vcOfOA^bj|mEiS>=pmoKvH?DylQpNoepv1yV6;t^!REnT(ms zpa7jfMpz4vMnhWG!2%HR$LJFU9!L^85OuqdEpyO&I(X2p91RD(g*jGE{#d08rI$Uo z%PX(!{OWkh%sY7X;Nb1T!xUF*G1r+R2$QKxcU%IvLmVKr059x{@S!l+SA_l>&KV88&j^0tLL$% zz#<}W4d}&yWi?=c8>7{X51+HZa!Q?w0{~;2JuA$nfiPSh&S^7Fm0A--V~OD=cU(4J zs-V$L45kJzf|QH*RcO(k(c}Y}SK#=RjOy-9Z`*Vk7h$0tR0(rs5(^WIErqt z>Q;V+`ts{het}zsmHbz#gx~{Sw#T1YT?N(8pLYTojihMKs>hp^WLo)mb5>d>X(qN^ zbnBcTTsc)j@{&~#OK;}Kj`eb;rO6sufu^l|L|fHE4{?umai)7fKt=UykifB$(rmTo zoc*j7J55yLDX~)p2c9(JiHWLd&MCi))>$Tre)BP7QLsuB_litcb~@*Z&^=92TtbD_09OT*kFkcKS_yy{Od^w?RAMOcH$ap)1>V!5MbWNoSHC=WJM)Xa6?h zw2z{_YzI(*%?iiw!aEdoINz-{ZWnH~6gNHpZHk|spZB~~0hz`%UA&@C~S z&EcA1(uux^SU&KkAmW3dU}ROrUMes+l5-$bdWvq$CNiBoH$~Zif(&l1VB&=}+u*yo z4X;y~Y>U-`xS(ndhaq@=ZUa#^uo`MCD9mCJGAGX&gxCjoWKp0fttjD91$>8@PoAmL z^;Bx}r1?lEm)p&oBt#H)OHvUTYMK1Fh6>DKj*BThTLmOMsv`l^yCam(srKg4ODb3>g*hkzVg+$XYy;eoFM^am zg~+DE$l&Zl0D4uwCBZ>cxgY>Kz(AirDr_)F3)&J1-K{L#BBOCQEVBbHd~0IUZT`kIf_kYM3y44=@f4T z5OfLUvPL77pHrD79MobMyK5LW7$}oMzUU3IN)Qy#A(YN$1darlL?y@#D( zZx`QkcV+7W^M{wcbs29<+KYWLg12j$v;L>=*lo3y@Fl9K{M2Px*YxJ@*LSB{cfVVb z+4ER>&tprCN3OV*$qA_|UD@vKX6l2t>VsFD*{Zs0Jy(0KpZK6^(@ke) z>my5BAIY`{Gwr+5?YnM{F17buIlAJr)NOdjnXcWO?do1E%LYT4;DL1TK(?-N@dZ@B zvF6Ib#a(aqyx#MHzhkA|(y(!*-BRXz^|6J=GM=Wir)hEQL(c{@V!rR=vX<-ZAC`6I zD*PYSHedIrYPa5~!MewvVSkTp*!ZJt+t2K_l5L;BHEP?RSNN|~-Mvx;Z{PB+Oj&2T ztn+T79XDheD6hEt;UHt+ahfOp1Q>HF4e8^CR%VX^HESdThyS2m{B?OCeaoAT^kuCBlK+-uLx4`jXm#e*Ms zTd}(CJMOpKHwMyy?N^Gf%zUkE#c8Q&fwtM#IG=p=3kzRZ+>!Ebz3%_OOY_gWHRakW z7->|d9hAm@>z!64pmGI6W@Zwavl2+NZQ~cmRQ;#wD##WMjW(<6EAzwP(<&iSt!U82 zDpKJcPoT(%Q&X8qG7nA(R>zYR?I_Q;Fi&|(!Amq>SO>~2_4=|a*77VmOK%N>r=O?#otxwdr{j=-jsGB`K zN+94z>8xq#3e7t*uJvix`o-i2u6A7jLIysxanp2Q@`>SUs%Fu2V64#oHQ>)f{~r&i zL^HfXK*Zz5ya0+L!=kW7vm;oT4B(1XqJl9BtB&V1ICQ8+Z;8|3l0x3i;E*t%3C>sx zzzTo@7KZ#!HQjfkMlY$~Jf(g2ueA|XWK%7dQg35 z){VOzquBT~baaL{-Z?5_%|3?`?gU>}In~1`e8^!b9JX=bBdBqLw;^RL(&ifN`Q9 zcT2Rg-&XJVXvvT9yVR__R{;3sn|N)ufxz*(Mi$Y#l;x46RX~eFpy@41#*!$GMCgQur{LyMTv#t|r)78fcHj`Eh<0^_+ zJW;Uqv@m?#^(I4)^Ja1r&ru9qwK>@C8phrb7WvbZ(X-F`^z6KqBe$u(qw3GN+1Z?4cHJHE+AsLxI_8Bke_yIN{asChYrImx(1|1;rI=3d(rJ!P|1+K5r_=vR zr$46C@6+ku(&@)^`iM?{MkkV`b1qu$gDtiqY|K~wk{x4d+~`>b=Q((51d|GJHXUt z*9Ee{&DjHwWH$sMi)LFp>AYn}Hng4oaK8z`d%9NIPTQQF%N2DiPD;Rgp`v-En3LFx zYyCs-l zR%$s}XKCHMQqRc-OH21kBPZ8c)^)EmadN$-dH+f?Cqb7QS6VpPYN>4%t%b;3T7G8( z-#zZM6qWzfR^n{GV@GO*Q#Do2(mNaNPX9`&$Jw}C*}dY#72oFXS}CR!;xo0Zlu)YF zQd)h-O{sO3BL5v*o3n)q@~=>8%mp6hf<;*N{MyFRntohXCa@ODRJMcvwhjY^%Pm%LcE7TdHSgORAt=mXt`LetBiR&SF+2QwcNhy`> z)5`W~&B-~Q;k#MKGb@fg&TZMcohx>_K0F|nGYasR3I*(Y8sA*!qdKOzx5bFp&x@rJnR%=85-m4ZMv#YTx@?Qt<&5rV9N#NZTpxH^lM_1jH zbCMhiWF9D`)Uu!;5wQg1zBZ~5lPAE?Cm4lu*$xy#T4Z(aCb(Mg=H==#Q~C_mF;6W_ zRa^~2Y$mq`gkpyoM2lT+T!~K0WmKVxBtA%fP`>BMC?u!k9jcBUO`_JSRY?;Dzbab+ zeLI<4gfk9l-`N0z7lmpJ{Vn~pW!r=(q`uEG4%(h4S+ONZE>sb&AgXm*=$+IuVfQin z3fMKG_+-e$QjzflN708os^98W?JT&-E~wDWQ3t54A``!86FUxotL23n`ldj6%`v_i zosLaRVW*4<^4z44G8RC_8`|iI?BxJYi0b+-4Z47;qL0M^5xPYqo}-W#%QYYie7Y4RE99I~bvLHIOprqdI^l|Yy~ zMGQLYHy9kn8v6W6gAk)?7|!%PJK5x0Lya9u&TaB7a%_XyK){+<>@7VBRbbKr z0_cm}jD=Ae@GKTpg{9+_s!U^5mr4Q5Eu^~5@i_G}{^;{S-^h23A`Z}O1x}T7B+<

mFP7xW3)PHt;KO&J-HUlxtP#;Se;Cbd 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 YUasYD>m+acN|=gYo>A7x;%~zIZ~Q_XDF6Tf diff --git a/.env/Lib/site-packages/click/__pycache__/exceptions.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/exceptions.cpython-312.pyc deleted file mode 100644 index 93e1c21ad6b0f73e2cc0bc47c0de14b4ede0df41..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14769 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 Qn4|LHptXkI&zS6g0J_|D5C8xG diff --git a/.env/Lib/site-packages/click/__pycache__/globals.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/globals.cpython-312.pyc deleted file mode 100644 index d09c4c5f779d005428dd5a64663067385a019cd3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2957 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 diff --git a/.env/Lib/site-packages/click/__pycache__/testing.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/testing.cpython-312.pyc deleted file mode 100644 index b931d6c55b02789029e6cfdedb748d0f57cb4f9c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27386 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 diff --git a/.env/Lib/site-packages/click/__pycache__/utils.cpython-312.pyc b/.env/Lib/site-packages/click/__pycache__/utils.cpython-312.pyc deleted file mode 100644 index 860e2a944503d217e36b49114f0587d83172ec93..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 24861 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 QQ7)ROvfCvHF$4d90h|KZJ^%m! diff --git a/.env/Lib/site-packages/click/_compat.py b/.env/Lib/site-packages/click/_compat.py deleted file mode 100644 index f2726b9..0000000 --- a/.env/Lib/site-packages/click/_compat.py +++ /dev/null @@ -1,622 +0,0 @@ -from __future__ import annotations - -import codecs -import collections.abc as cabc -import io -import os -import re -import sys -import typing as t -from types import TracebackType -from weakref import WeakKeyDictionary - -CYGWIN = sys.platform.startswith("cygwin") -WIN = sys.platform.startswith("win") -auto_wrap_for_ansi: t.Callable[[t.TextIO], t.TextIO] | None = None -_ansi_re = re.compile(r"\033\[[;?0-9]*[a-zA-Z]") - - -def _make_text_stream( - stream: t.BinaryIO, - encoding: str | None, - errors: str | None, - force_readable: bool = False, - force_writable: bool = False, -) -> t.TextIO: - if encoding is None: - encoding = get_best_encoding(stream) - if errors is None: - errors = "replace" - return _NonClosingTextIOWrapper( - stream, - encoding, - errors, - line_buffering=True, - force_readable=force_readable, - force_writable=force_writable, - ) - - -def is_ascii_encoding(encoding: str) -> bool: - """Checks if a given encoding is ascii.""" - try: - return codecs.lookup(encoding).name == "ascii" - except LookupError: - return False - - -def get_best_encoding(stream: t.IO[t.Any]) -> str: - """Returns the default stream encoding if not found.""" - rv = getattr(stream, "encoding", None) or sys.getdefaultencoding() - if is_ascii_encoding(rv): - return "utf-8" - return rv - - -class _NonClosingTextIOWrapper(io.TextIOWrapper): - def __init__( - self, - stream: t.BinaryIO, - encoding: str | None, - errors: str | None, - force_readable: bool = False, - force_writable: bool = False, - **extra: t.Any, - ) -> None: - self._stream = stream = t.cast( - t.BinaryIO, _FixupStream(stream, force_readable, force_writable) - ) - super().__init__(stream, encoding, errors, **extra) - - def __del__(self) -> None: - try: - self.detach() - except Exception: - pass - - def isatty(self) -> bool: - # https://bitbucket.org/pypy/pypy/issue/1803 - return self._stream.isatty() - - -class _FixupStream: - """The new io interface needs more from streams than streams - traditionally implement. As such, this fix-up code is necessary in - some circumstances. - - The forcing of readable and writable flags are there because some tools - put badly patched objects on sys (one such offender are certain version - of jupyter notebook). - """ - - def __init__( - self, - stream: t.BinaryIO, - force_readable: bool = False, - force_writable: bool = False, - ): - self._stream = stream - self._force_readable = force_readable - self._force_writable = force_writable - - def __getattr__(self, name: str) -> t.Any: - return getattr(self._stream, name) - - def read1(self, size: int) -> bytes: - f = getattr(self._stream, "read1", None) - - if f is not None: - return t.cast(bytes, f(size)) - - return self._stream.read(size) - - def readable(self) -> bool: - if self._force_readable: - return True - x = getattr(self._stream, "readable", None) - if x is not None: - return t.cast(bool, x()) - try: - self._stream.read(0) - except Exception: - return False - return True - - def writable(self) -> bool: - if self._force_writable: - return True - x = getattr(self._stream, "writable", None) - if x is not None: - return t.cast(bool, x()) - try: - self._stream.write(b"") - except Exception: - try: - self._stream.write(b"") - except Exception: - return False - return True - - def seekable(self) -> bool: - x = getattr(self._stream, "seekable", None) - if x is not None: - return t.cast(bool, x()) - try: - self._stream.seek(self._stream.tell()) - except Exception: - return False - return True - - -def _is_binary_reader(stream: t.IO[t.Any], default: bool = False) -> bool: - try: - return isinstance(stream.read(0), bytes) - except Exception: - return default - # This happens in some cases where the stream was already - # closed. In this case, we assume the default. - - -def _is_binary_writer(stream: t.IO[t.Any], default: bool = False) -> bool: - try: - stream.write(b"") - except Exception: - try: - stream.write("") - return False - except Exception: - pass - return default - return True - - -def _find_binary_reader(stream: t.IO[t.Any]) -> t.BinaryIO | None: - # We need to figure out if the given stream is already binary. - # This can happen because the official docs recommend detaching - # the streams to get binary streams. Some code might do this, so - # we need to deal with this case explicitly. - if _is_binary_reader(stream, False): - return t.cast(t.BinaryIO, stream) - - buf = getattr(stream, "buffer", None) - - # Same situation here; this time we assume that the buffer is - # actually binary in case it's closed. - if buf is not None and _is_binary_reader(buf, True): - return t.cast(t.BinaryIO, buf) - - return None - - -def _find_binary_writer(stream: t.IO[t.Any]) -> t.BinaryIO | None: - # We need to figure out if the given stream is already binary. - # This can happen because the official docs recommend detaching - # the streams to get binary streams. Some code might do this, so - # we need to deal with this case explicitly. - if _is_binary_writer(stream, False): - return t.cast(t.BinaryIO, stream) - - buf = getattr(stream, "buffer", None) - - # Same situation here; this time we assume that the buffer is - # actually binary in case it's closed. - if buf is not None and _is_binary_writer(buf, True): - return t.cast(t.BinaryIO, buf) - - return None - - -def _stream_is_misconfigured(stream: t.TextIO) -> bool: - """A stream is misconfigured if its encoding is ASCII.""" - # If the stream does not have an encoding set, we assume it's set - # to ASCII. This appears to happen in certain unittest - # environments. It's not quite clear what the correct behavior is - # but this at least will force Click to recover somehow. - return is_ascii_encoding(getattr(stream, "encoding", None) or "ascii") - - -def _is_compat_stream_attr(stream: t.TextIO, attr: str, value: str | None) -> bool: - """A stream attribute is compatible if it is equal to the - desired value or the desired value is unset and the attribute - has a value. - """ - stream_value = getattr(stream, attr, None) - return stream_value == value or (value is None and stream_value is not None) - - -def _is_compatible_text_stream( - stream: t.TextIO, encoding: str | None, errors: str | None -) -> bool: - """Check if a stream's encoding and errors attributes are - compatible with the desired values. - """ - return _is_compat_stream_attr( - stream, "encoding", encoding - ) and _is_compat_stream_attr(stream, "errors", errors) - - -def _force_correct_text_stream( - text_stream: t.IO[t.Any], - encoding: str | None, - errors: str | None, - is_binary: t.Callable[[t.IO[t.Any], bool], bool], - find_binary: t.Callable[[t.IO[t.Any]], t.BinaryIO | None], - force_readable: bool = False, - force_writable: bool = False, -) -> t.TextIO: - if is_binary(text_stream, False): - binary_reader = t.cast(t.BinaryIO, text_stream) - else: - text_stream = t.cast(t.TextIO, text_stream) - # If the stream looks compatible, and won't default to a - # misconfigured ascii encoding, return it as-is. - if _is_compatible_text_stream(text_stream, encoding, errors) and not ( - encoding is None and _stream_is_misconfigured(text_stream) - ): - return text_stream - - # Otherwise, get the underlying binary reader. - possible_binary_reader = find_binary(text_stream) - - # If that's not possible, silently use the original reader - # and get mojibake instead of exceptions. - if possible_binary_reader is None: - return text_stream - - binary_reader = possible_binary_reader - - # Default errors to replace instead of strict in order to get - # something that works. - if errors is None: - errors = "replace" - - # Wrap the binary stream in a text stream with the correct - # encoding parameters. - return _make_text_stream( - binary_reader, - encoding, - errors, - force_readable=force_readable, - force_writable=force_writable, - ) - - -def _force_correct_text_reader( - text_reader: t.IO[t.Any], - encoding: str | None, - errors: str | None, - force_readable: bool = False, -) -> t.TextIO: - return _force_correct_text_stream( - text_reader, - encoding, - errors, - _is_binary_reader, - _find_binary_reader, - force_readable=force_readable, - ) - - -def _force_correct_text_writer( - text_writer: t.IO[t.Any], - encoding: str | None, - errors: str | None, - force_writable: bool = False, -) -> t.TextIO: - return _force_correct_text_stream( - text_writer, - encoding, - errors, - _is_binary_writer, - _find_binary_writer, - force_writable=force_writable, - ) - - -def get_binary_stdin() -> t.BinaryIO: - reader = _find_binary_reader(sys.stdin) - if reader is None: - raise RuntimeError("Was not able to determine binary stream for sys.stdin.") - return reader - - -def get_binary_stdout() -> t.BinaryIO: - writer = _find_binary_writer(sys.stdout) - if writer is None: - raise RuntimeError("Was not able to determine binary stream for sys.stdout.") - return writer - - -def get_binary_stderr() -> t.BinaryIO: - writer = _find_binary_writer(sys.stderr) - if writer is None: - raise RuntimeError("Was not able to determine binary stream for sys.stderr.") - return writer - - -def get_text_stdin(encoding: str | None = None, errors: str | None = None) -> t.TextIO: - rv = _get_windows_console_stream(sys.stdin, encoding, errors) - if rv is not None: - return rv - return _force_correct_text_reader(sys.stdin, encoding, errors, force_readable=True) - - -def get_text_stdout(encoding: str | None = None, errors: str | None = None) -> t.TextIO: - rv = _get_windows_console_stream(sys.stdout, encoding, errors) - if rv is not None: - return rv - return _force_correct_text_writer(sys.stdout, encoding, errors, force_writable=True) - - -def get_text_stderr(encoding: str | None = None, errors: str | None = None) -> t.TextIO: - rv = _get_windows_console_stream(sys.stderr, encoding, errors) - if rv is not None: - return rv - return _force_correct_text_writer(sys.stderr, encoding, errors, force_writable=True) - - -def _wrap_io_open( - file: str | os.PathLike[str] | int, - mode: str, - encoding: str | None, - errors: str | None, -) -> t.IO[t.Any]: - """Handles not passing ``encoding`` and ``errors`` in binary mode.""" - if "b" in mode: - return open(file, mode) - - return open(file, mode, encoding=encoding, errors=errors) - - -def open_stream( - filename: str | os.PathLike[str], - mode: str = "r", - encoding: str | None = None, - errors: str | None = "strict", - atomic: bool = False, -) -> tuple[t.IO[t.Any], bool]: - binary = "b" in mode - filename = os.fspath(filename) - - # Standard streams first. These are simple because they ignore the - # atomic flag. Use fsdecode to handle Path("-"). - if os.fsdecode(filename) == "-": - if any(m in mode for m in ["w", "a", "x"]): - if binary: - return get_binary_stdout(), False - return get_text_stdout(encoding=encoding, errors=errors), False - if binary: - return get_binary_stdin(), False - return get_text_stdin(encoding=encoding, errors=errors), False - - # Non-atomic writes directly go out through the regular open functions. - if not atomic: - return _wrap_io_open(filename, mode, encoding, errors), True - - # Some usability stuff for atomic writes - if "a" in mode: - raise ValueError( - "Appending to an existing file is not supported, because that" - " would involve an expensive `copy`-operation to a temporary" - " file. Open the file in normal `w`-mode and copy explicitly" - " if that's what you're after." - ) - if "x" in mode: - raise ValueError("Use the `overwrite`-parameter instead.") - if "w" not in mode: - raise ValueError("Atomic writes only make sense with `w`-mode.") - - # Atomic writes are more complicated. They work by opening a file - # as a proxy in the same folder and then using the fdopen - # functionality to wrap it in a Python file. Then we wrap it in an - # atomic file that moves the file over on close. - import errno - import random - - try: - perm: int | None = os.stat(filename).st_mode - except OSError: - perm = None - - flags = os.O_RDWR | os.O_CREAT | os.O_EXCL - - if binary: - flags |= getattr(os, "O_BINARY", 0) - - while True: - tmp_filename = os.path.join( - os.path.dirname(filename), - f".__atomic-write{random.randrange(1 << 32):08x}", - ) - try: - fd = os.open(tmp_filename, flags, 0o666 if perm is None else perm) - break - except OSError as e: - if e.errno == errno.EEXIST or ( - os.name == "nt" - and e.errno == errno.EACCES - and os.path.isdir(e.filename) - and os.access(e.filename, os.W_OK) - ): - continue - raise - - if perm is not None: - os.chmod(tmp_filename, perm) # in case perm includes bits in umask - - f = _wrap_io_open(fd, mode, encoding, errors) - af = _AtomicFile(f, tmp_filename, os.path.realpath(filename)) - return t.cast(t.IO[t.Any], af), True - - -class _AtomicFile: - def __init__(self, f: t.IO[t.Any], tmp_filename: str, real_filename: str) -> None: - self._f = f - self._tmp_filename = tmp_filename - self._real_filename = real_filename - self.closed = False - - @property - def name(self) -> str: - return self._real_filename - - def close(self, delete: bool = False) -> None: - if self.closed: - return - self._f.close() - os.replace(self._tmp_filename, self._real_filename) - self.closed = True - - def __getattr__(self, name: str) -> t.Any: - return getattr(self._f, name) - - def __enter__(self) -> _AtomicFile: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> None: - self.close(delete=exc_type is not None) - - def __repr__(self) -> str: - return repr(self._f) - - -def strip_ansi(value: str) -> str: - return _ansi_re.sub("", value) - - -def _is_jupyter_kernel_output(stream: t.IO[t.Any]) -> bool: - while isinstance(stream, (_FixupStream, _NonClosingTextIOWrapper)): - stream = stream._stream - - return stream.__class__.__module__.startswith("ipykernel.") - - -def should_strip_ansi( - stream: t.IO[t.Any] | None = None, color: bool | None = None -) -> bool: - if color is None: - if stream is None: - stream = sys.stdin - return not isatty(stream) and not _is_jupyter_kernel_output(stream) - return not color - - -# On Windows, wrap the output streams with colorama to support ANSI -# color codes. -# NOTE: double check is needed so mypy does not analyze this on Linux -if sys.platform.startswith("win") and WIN: - from ._winconsole import _get_windows_console_stream - - def _get_argv_encoding() -> str: - import locale - - return locale.getpreferredencoding() - - _ansi_stream_wrappers: cabc.MutableMapping[t.TextIO, t.TextIO] = WeakKeyDictionary() - - def auto_wrap_for_ansi(stream: t.TextIO, color: bool | None = None) -> t.TextIO: - """Support ANSI color and style codes on Windows by wrapping a - stream with colorama. - """ - try: - cached = _ansi_stream_wrappers.get(stream) - except Exception: - cached = None - - if cached is not None: - return cached - - import colorama - - strip = should_strip_ansi(stream, color) - ansi_wrapper = colorama.AnsiToWin32(stream, strip=strip) - rv = t.cast(t.TextIO, ansi_wrapper.stream) - _write = rv.write - - def _safe_write(s: str) -> int: - try: - return _write(s) - except BaseException: - ansi_wrapper.reset_all() - raise - - rv.write = _safe_write # type: ignore[method-assign] - - try: - _ansi_stream_wrappers[stream] = rv - except Exception: - pass - - return rv - -else: - - def _get_argv_encoding() -> str: - return getattr(sys.stdin, "encoding", None) or sys.getfilesystemencoding() - - def _get_windows_console_stream( - f: t.TextIO, encoding: str | None, errors: str | None - ) -> t.TextIO | None: - return None - - -def term_len(x: str) -> int: - return len(strip_ansi(x)) - - -def isatty(stream: t.IO[t.Any]) -> bool: - try: - return stream.isatty() - except Exception: - return False - - -def _make_cached_stream_func( - src_func: t.Callable[[], t.TextIO | None], - wrapper_func: t.Callable[[], t.TextIO], -) -> t.Callable[[], t.TextIO | None]: - cache: cabc.MutableMapping[t.TextIO, t.TextIO] = WeakKeyDictionary() - - def func() -> t.TextIO | None: - stream = src_func() - - if stream is None: - return None - - try: - rv = cache.get(stream) - except Exception: - rv = None - if rv is not None: - return rv - rv = wrapper_func() - try: - cache[stream] = rv - except Exception: - pass - return rv - - return func - - -_default_text_stdin = _make_cached_stream_func(lambda: sys.stdin, get_text_stdin) -_default_text_stdout = _make_cached_stream_func(lambda: sys.stdout, get_text_stdout) -_default_text_stderr = _make_cached_stream_func(lambda: sys.stderr, get_text_stderr) - - -binary_streams: cabc.Mapping[str, t.Callable[[], t.BinaryIO]] = { - "stdin": get_binary_stdin, - "stdout": get_binary_stdout, - "stderr": get_binary_stderr, -} - -text_streams: cabc.Mapping[str, t.Callable[[str | None, str | None], t.TextIO]] = { - "stdin": get_text_stdin, - "stdout": get_text_stdout, - "stderr": get_text_stderr, -} diff --git a/.env/Lib/site-packages/click/_termui_impl.py b/.env/Lib/site-packages/click/_termui_impl.py deleted file mode 100644 index ee8225c..0000000 --- a/.env/Lib/site-packages/click/_termui_impl.py +++ /dev/null @@ -1,852 +0,0 @@ -""" -This module contains implementations for the termui module. To keep the -import time of Click down, some infrequently used functionality is -placed in this module and only imported as needed. -""" - -from __future__ import annotations - -import collections.abc as cabc -import contextlib -import math -import os -import shlex -import sys -import time -import typing as t -from gettext import gettext as _ -from io import StringIO -from pathlib import Path -from types import TracebackType - -from ._compat import _default_text_stdout -from ._compat import CYGWIN -from ._compat import get_best_encoding -from ._compat import isatty -from ._compat import open_stream -from ._compat import strip_ansi -from ._compat import term_len -from ._compat import WIN -from .exceptions import ClickException -from .utils import echo - -V = t.TypeVar("V") - -if os.name == "nt": - BEFORE_BAR = "\r" - AFTER_BAR = "\n" -else: - BEFORE_BAR = "\r\033[?25l" - AFTER_BAR = "\033[?25h\n" - - -class ProgressBar(t.Generic[V]): - def __init__( - self, - iterable: cabc.Iterable[V] | None, - length: int | None = None, - fill_char: str = "#", - empty_char: str = " ", - bar_template: str = "%(bar)s", - info_sep: str = " ", - hidden: bool = False, - show_eta: bool = True, - show_percent: bool | None = None, - show_pos: bool = False, - item_show_func: t.Callable[[V | None], str | None] | None = None, - label: str | None = None, - file: t.TextIO | None = None, - color: bool | None = None, - update_min_steps: int = 1, - width: int = 30, - ) -> None: - self.fill_char = fill_char - self.empty_char = empty_char - self.bar_template = bar_template - self.info_sep = info_sep - self.hidden = hidden - self.show_eta = show_eta - self.show_percent = show_percent - self.show_pos = show_pos - self.item_show_func = item_show_func - self.label: str = label or "" - - if file is None: - file = _default_text_stdout() - - # There are no standard streams attached to write to. For example, - # pythonw on Windows. - if file is None: - file = StringIO() - - self.file = file - self.color = color - self.update_min_steps = update_min_steps - self._completed_intervals = 0 - self.width: int = width - self.autowidth: bool = width == 0 - - if length is None: - from operator import length_hint - - length = length_hint(iterable, -1) - - if length == -1: - length = None - if iterable is None: - if length is None: - raise TypeError("iterable or length is required") - iterable = t.cast("cabc.Iterable[V]", range(length)) - self.iter: cabc.Iterable[V] = iter(iterable) - self.length = length - self.pos: int = 0 - self.avg: list[float] = [] - self.last_eta: float - self.start: float - self.start = self.last_eta = time.time() - self.eta_known: bool = False - self.finished: bool = False - self.max_width: int | None = None - self.entered: bool = False - self.current_item: V | None = None - self._is_atty = isatty(self.file) - self._last_line: str | None = None - - def __enter__(self) -> ProgressBar[V]: - self.entered = True - self.render_progress() - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> None: - self.render_finish() - - def __iter__(self) -> cabc.Iterator[V]: - if not self.entered: - raise RuntimeError("You need to use progress bars in a with block.") - self.render_progress() - return self.generator() - - def __next__(self) -> V: - # Iteration is defined in terms of a generator function, - # returned by iter(self); use that to define next(). This works - # because `self.iter` is an iterable consumed by that generator, - # so it is re-entry safe. Calling `next(self.generator())` - # twice works and does "what you want". - return next(iter(self)) - - def render_finish(self) -> None: - if self.hidden or not self._is_atty: - return - self.file.write(AFTER_BAR) - self.file.flush() - - @property - def pct(self) -> float: - if self.finished: - return 1.0 - return min(self.pos / (float(self.length or 1) or 1), 1.0) - - @property - def time_per_iteration(self) -> float: - if not self.avg: - return 0.0 - return sum(self.avg) / float(len(self.avg)) - - @property - def eta(self) -> float: - if self.length is not None and not self.finished: - return self.time_per_iteration * (self.length - self.pos) - return 0.0 - - def format_eta(self) -> str: - if self.eta_known: - t = int(self.eta) - seconds = t % 60 - t //= 60 - minutes = t % 60 - t //= 60 - hours = t % 24 - t //= 24 - if t > 0: - return f"{t}d {hours:02}:{minutes:02}:{seconds:02}" - else: - return f"{hours:02}:{minutes:02}:{seconds:02}" - return "" - - def format_pos(self) -> str: - pos = str(self.pos) - if self.length is not None: - pos += f"/{self.length}" - return pos - - def format_pct(self) -> str: - return f"{int(self.pct * 100): 4}%"[1:] - - def format_bar(self) -> str: - if self.length is not None: - bar_length = int(self.pct * self.width) - bar = self.fill_char * bar_length - bar += self.empty_char * (self.width - bar_length) - elif self.finished: - bar = self.fill_char * self.width - else: - chars = list(self.empty_char * (self.width or 1)) - if self.time_per_iteration != 0: - chars[ - int( - (math.cos(self.pos * self.time_per_iteration) / 2.0 + 0.5) - * self.width - ) - ] = self.fill_char - bar = "".join(chars) - return bar - - def format_progress_line(self) -> str: - show_percent = self.show_percent - - info_bits = [] - if self.length is not None and show_percent is None: - show_percent = not self.show_pos - - if self.show_pos: - info_bits.append(self.format_pos()) - if show_percent: - info_bits.append(self.format_pct()) - if self.show_eta and self.eta_known and not self.finished: - info_bits.append(self.format_eta()) - if self.item_show_func is not None: - item_info = self.item_show_func(self.current_item) - if item_info is not None: - info_bits.append(item_info) - - return ( - self.bar_template - % { - "label": self.label, - "bar": self.format_bar(), - "info": self.info_sep.join(info_bits), - } - ).rstrip() - - def render_progress(self) -> None: - if self.hidden: - return - - if not self._is_atty: - # Only output the label once if the output is not a TTY. - if self._last_line != self.label: - self._last_line = self.label - echo(self.label, file=self.file, color=self.color) - return - - buf = [] - # Update width in case the terminal has been resized - if self.autowidth: - import shutil - - old_width = self.width - self.width = 0 - clutter_length = term_len(self.format_progress_line()) - new_width = max(0, shutil.get_terminal_size().columns - clutter_length) - if new_width < old_width and self.max_width is not None: - buf.append(BEFORE_BAR) - buf.append(" " * self.max_width) - self.max_width = new_width - self.width = new_width - - clear_width = self.width - if self.max_width is not None: - clear_width = self.max_width - - buf.append(BEFORE_BAR) - line = self.format_progress_line() - line_len = term_len(line) - if self.max_width is None or self.max_width < line_len: - self.max_width = line_len - - buf.append(line) - buf.append(" " * (clear_width - line_len)) - line = "".join(buf) - # Render the line only if it changed. - - if line != self._last_line: - self._last_line = line - echo(line, file=self.file, color=self.color, nl=False) - self.file.flush() - - def make_step(self, n_steps: int) -> None: - self.pos += n_steps - if self.length is not None and self.pos >= self.length: - self.finished = True - - if (time.time() - self.last_eta) < 1.0: - return - - self.last_eta = time.time() - - # self.avg is a rolling list of length <= 7 of steps where steps are - # defined as time elapsed divided by the total progress through - # self.length. - if self.pos: - step = (time.time() - self.start) / self.pos - else: - step = time.time() - self.start - - self.avg = self.avg[-6:] + [step] - - self.eta_known = self.length is not None - - def update(self, n_steps: int, current_item: V | None = None) -> None: - """Update the progress bar by advancing a specified number of - steps, and optionally set the ``current_item`` for this new - position. - - :param n_steps: Number of steps to advance. - :param current_item: Optional item to set as ``current_item`` - for the updated position. - - .. versionchanged:: 8.0 - Added the ``current_item`` optional parameter. - - .. versionchanged:: 8.0 - Only render when the number of steps meets the - ``update_min_steps`` threshold. - """ - if current_item is not None: - self.current_item = current_item - - self._completed_intervals += n_steps - - if self._completed_intervals >= self.update_min_steps: - self.make_step(self._completed_intervals) - self.render_progress() - self._completed_intervals = 0 - - def finish(self) -> None: - self.eta_known = False - self.current_item = None - self.finished = True - - def generator(self) -> cabc.Iterator[V]: - """Return a generator which yields the items added to the bar - during construction, and updates the progress bar *after* the - yielded block returns. - """ - # WARNING: the iterator interface for `ProgressBar` relies on - # this and only works because this is a simple generator which - # doesn't create or manage additional state. If this function - # changes, the impact should be evaluated both against - # `iter(bar)` and `next(bar)`. `next()` in particular may call - # `self.generator()` repeatedly, and this must remain safe in - # order for that interface to work. - if not self.entered: - raise RuntimeError("You need to use progress bars in a with block.") - - if not self._is_atty: - yield from self.iter - else: - for rv in self.iter: - self.current_item = rv - - # This allows show_item_func to be updated before the - # item is processed. Only trigger at the beginning of - # the update interval. - if self._completed_intervals == 0: - self.render_progress() - - yield rv - self.update(1) - - self.finish() - self.render_progress() - - -def pager(generator: cabc.Iterable[str], color: bool | None = None) -> None: - """Decide what method to use for paging through text.""" - stdout = _default_text_stdout() - - # There are no standard streams attached to write to. For example, - # pythonw on Windows. - if stdout is None: - stdout = StringIO() - - if not isatty(sys.stdin) or not isatty(stdout): - return _nullpager(stdout, generator, color) - - # Split and normalize the pager command into parts. - pager_cmd_parts = shlex.split(os.environ.get("PAGER", ""), posix=False) - if pager_cmd_parts: - if WIN: - if _tempfilepager(generator, pager_cmd_parts, color): - return - elif _pipepager(generator, pager_cmd_parts, color): - return - - if os.environ.get("TERM") in ("dumb", "emacs"): - return _nullpager(stdout, generator, color) - if (WIN or sys.platform.startswith("os2")) and _tempfilepager( - generator, ["more"], color - ): - return - if _pipepager(generator, ["less"], color): - return - - import tempfile - - fd, filename = tempfile.mkstemp() - os.close(fd) - try: - if _pipepager(generator, ["more"], color): - return - return _nullpager(stdout, generator, color) - finally: - os.unlink(filename) - - -def _pipepager( - generator: cabc.Iterable[str], cmd_parts: list[str], color: bool | None -) -> bool: - """Page through text by feeding it to another program. Invoking a - pager through this might support colors. - - Returns `True` if the command was found, `False` otherwise and thus another - pager should be attempted. - """ - # Split the command into the invoked CLI and its parameters. - if not cmd_parts: - return False - - import shutil - - cmd = cmd_parts[0] - cmd_params = cmd_parts[1:] - - cmd_filepath = shutil.which(cmd) - if not cmd_filepath: - return False - - # Produces a normalized absolute path string. - # multi-call binaries such as busybox derive their identity from the symlink - # less -> busybox. resolve() causes them to misbehave. (eg. less becomes busybox) - cmd_path = Path(cmd_filepath).absolute() - cmd_name = cmd_path.name - - import subprocess - - # Make a local copy of the environment to not affect the global one. - env = dict(os.environ) - - # If we're piping to less and the user hasn't decided on colors, we enable - # them by default we find the -R flag in the command line arguments. - if color is None and cmd_name == "less": - less_flags = f"{os.environ.get('LESS', '')}{' '.join(cmd_params)}" - if not less_flags: - env["LESS"] = "-R" - color = True - elif "r" in less_flags or "R" in less_flags: - color = True - - c = subprocess.Popen( - [str(cmd_path)] + cmd_params, - shell=False, - stdin=subprocess.PIPE, - env=env, - errors="replace", - text=True, - ) - assert c.stdin is not None - try: - for text in generator: - if not color: - text = strip_ansi(text) - - c.stdin.write(text) - except BrokenPipeError: - # In case the pager exited unexpectedly, ignore the broken pipe error. - pass - except Exception as e: - # In case there is an exception we want to close the pager immediately - # and let the caller handle it. - # Otherwise the pager will keep running, and the user may not notice - # the error message, or worse yet it may leave the terminal in a broken state. - c.terminate() - raise e - finally: - # We must close stdin and wait for the pager to exit before we continue - try: - c.stdin.close() - # Close implies flush, so it might throw a BrokenPipeError if the pager - # process exited already. - except BrokenPipeError: - pass - - # Less doesn't respect ^C, but catches it for its own UI purposes (aborting - # search or other commands inside less). - # - # That means when the user hits ^C, the parent process (click) terminates, - # but less is still alive, paging the output and messing up the terminal. - # - # If the user wants to make the pager exit on ^C, they should set - # `LESS='-K'`. It's not our decision to make. - while True: - try: - c.wait() - except KeyboardInterrupt: - pass - else: - break - - return True - - -def _tempfilepager( - generator: cabc.Iterable[str], cmd_parts: list[str], color: bool | None -) -> bool: - """Page through text by invoking a program on a temporary file. - - Returns `True` if the command was found, `False` otherwise and thus another - pager should be attempted. - """ - # Split the command into the invoked CLI and its parameters. - if not cmd_parts: - return False - - import shutil - - cmd = cmd_parts[0] - - cmd_filepath = shutil.which(cmd) - if not cmd_filepath: - return False - # Produces a normalized absolute path string. - # multi-call binaries such as busybox derive their identity from the symlink - # less -> busybox. resolve() causes them to misbehave. (eg. less becomes busybox) - cmd_path = Path(cmd_filepath).absolute() - - import subprocess - import tempfile - - fd, filename = tempfile.mkstemp() - # TODO: This never terminates if the passed generator never terminates. - text = "".join(generator) - if not color: - text = strip_ansi(text) - encoding = get_best_encoding(sys.stdout) - with open_stream(filename, "wb")[0] as f: - f.write(text.encode(encoding)) - try: - subprocess.call([str(cmd_path), filename]) - except OSError: - # Command not found - pass - finally: - os.close(fd) - os.unlink(filename) - - return True - - -def _nullpager( - stream: t.TextIO, generator: cabc.Iterable[str], color: bool | None -) -> None: - """Simply print unformatted text. This is the ultimate fallback.""" - for text in generator: - if not color: - text = strip_ansi(text) - stream.write(text) - - -class Editor: - def __init__( - self, - editor: str | None = None, - env: cabc.Mapping[str, str] | None = None, - require_save: bool = True, - extension: str = ".txt", - ) -> None: - self.editor = editor - self.env = env - self.require_save = require_save - self.extension = extension - - def get_editor(self) -> str: - if self.editor is not None: - return self.editor - for key in "VISUAL", "EDITOR": - rv = os.environ.get(key) - if rv: - return rv - if WIN: - return "notepad" - - from shutil import which - - for editor in "sensible-editor", "vim", "nano": - if which(editor) is not None: - return editor - return "vi" - - def edit_files(self, filenames: cabc.Iterable[str]) -> None: - import subprocess - - editor = self.get_editor() - environ: dict[str, str] | None = None - - if self.env: - environ = os.environ.copy() - environ.update(self.env) - - exc_filename = " ".join(f'"{filename}"' for filename in filenames) - - try: - c = subprocess.Popen( - args=f"{editor} {exc_filename}", env=environ, shell=True - ) - exit_code = c.wait() - if exit_code != 0: - raise ClickException( - _("{editor}: Editing failed").format(editor=editor) - ) - except OSError as e: - raise ClickException( - _("{editor}: Editing failed: {e}").format(editor=editor, e=e) - ) from e - - @t.overload - def edit(self, text: bytes | bytearray) -> bytes | None: ... - - # We cannot know whether or not the type expected is str or bytes when None - # is passed, so str is returned as that was what was done before. - @t.overload - def edit(self, text: str | None) -> str | None: ... - - def edit(self, text: str | bytes | bytearray | None) -> str | bytes | None: - import tempfile - - if text is None: - data: bytes | bytearray = b"" - elif isinstance(text, (bytes, bytearray)): - data = text - else: - if text and not text.endswith("\n"): - text += "\n" - - if WIN: - data = text.replace("\n", "\r\n").encode("utf-8-sig") - else: - data = text.encode("utf-8") - - fd, name = tempfile.mkstemp(prefix="editor-", suffix=self.extension) - f: t.BinaryIO - - try: - with os.fdopen(fd, "wb") as f: - f.write(data) - - # If the filesystem resolution is 1 second, like Mac OS - # 10.12 Extended, or 2 seconds, like FAT32, and the editor - # closes very fast, require_save can fail. Set the modified - # time to be 2 seconds in the past to work around this. - os.utime(name, (os.path.getatime(name), os.path.getmtime(name) - 2)) - # Depending on the resolution, the exact value might not be - # recorded, so get the new recorded value. - timestamp = os.path.getmtime(name) - - self.edit_files((name,)) - - if self.require_save and os.path.getmtime(name) == timestamp: - return None - - with open(name, "rb") as f: - rv = f.read() - - if isinstance(text, (bytes, bytearray)): - return rv - - return rv.decode("utf-8-sig").replace("\r\n", "\n") - finally: - os.unlink(name) - - -def open_url(url: str, wait: bool = False, locate: bool = False) -> int: - import subprocess - - def _unquote_file(url: str) -> str: - from urllib.parse import unquote - - if url.startswith("file://"): - url = unquote(url[7:]) - - return url - - if sys.platform == "darwin": - args = ["open"] - if wait: - args.append("-W") - if locate: - args.append("-R") - args.append(_unquote_file(url)) - null = open("/dev/null", "w") - try: - return subprocess.Popen(args, stderr=null).wait() - finally: - null.close() - elif WIN: - if locate: - url = _unquote_file(url) - args = ["explorer", f"/select,{url}"] - else: - args = ["start"] - if wait: - args.append("/WAIT") - args.append("") - args.append(url) - try: - return subprocess.call(args) - except OSError: - # Command not found - return 127 - elif CYGWIN: - if locate: - url = _unquote_file(url) - args = ["cygstart", os.path.dirname(url)] - else: - args = ["cygstart"] - if wait: - args.append("-w") - args.append(url) - try: - return subprocess.call(args) - except OSError: - # Command not found - return 127 - - try: - if locate: - url = os.path.dirname(_unquote_file(url)) or "." - else: - url = _unquote_file(url) - c = subprocess.Popen(["xdg-open", url]) - if wait: - return c.wait() - return 0 - except OSError: - if url.startswith(("http://", "https://")) and not locate and not wait: - import webbrowser - - webbrowser.open(url) - return 0 - return 1 - - -def _translate_ch_to_exc(ch: str) -> None: - if ch == "\x03": - raise KeyboardInterrupt() - - if ch == "\x04" and not WIN: # Unix-like, Ctrl+D - raise EOFError() - - if ch == "\x1a" and WIN: # Windows, Ctrl+Z - raise EOFError() - - return None - - -if sys.platform == "win32": - import msvcrt - - @contextlib.contextmanager - def raw_terminal() -> cabc.Iterator[int]: - yield -1 - - def getchar(echo: bool) -> str: - # The function `getch` will return a bytes object corresponding to - # the pressed character. Since Windows 10 build 1803, it will also - # return \x00 when called a second time after pressing a regular key. - # - # `getwch` does not share this probably-bugged behavior. Moreover, it - # returns a Unicode object by default, which is what we want. - # - # Either of these functions will return \x00 or \xe0 to indicate - # a special key, and you need to call the same function again to get - # the "rest" of the code. The fun part is that \u00e0 is - # "latin small letter a with grave", so if you type that on a French - # keyboard, you _also_ get a \xe0. - # E.g., consider the Up arrow. This returns \xe0 and then \x48. The - # resulting Unicode string reads as "a with grave" + "capital H". - # This is indistinguishable from when the user actually types - # "a with grave" and then "capital H". - # - # When \xe0 is returned, we assume it's part of a special-key sequence - # and call `getwch` again, but that means that when the user types - # the \u00e0 character, `getchar` doesn't return until a second - # character is typed. - # The alternative is returning immediately, but that would mess up - # cross-platform handling of arrow keys and others that start with - # \xe0. Another option is using `getch`, but then we can't reliably - # read non-ASCII characters, because return values of `getch` are - # limited to the current 8-bit codepage. - # - # Anyway, Click doesn't claim to do this Right(tm), and using `getwch` - # is doing the right thing in more situations than with `getch`. - - if echo: - func = t.cast(t.Callable[[], str], msvcrt.getwche) - else: - func = t.cast(t.Callable[[], str], msvcrt.getwch) - - rv = func() - - if rv in ("\x00", "\xe0"): - # \x00 and \xe0 are control characters that indicate special key, - # see above. - rv += func() - - _translate_ch_to_exc(rv) - return rv - -else: - import termios - import tty - - @contextlib.contextmanager - def raw_terminal() -> cabc.Iterator[int]: - f: t.TextIO | None - fd: int - - if not isatty(sys.stdin): - f = open("/dev/tty") - fd = f.fileno() - else: - fd = sys.stdin.fileno() - f = None - - try: - old_settings = termios.tcgetattr(fd) - - try: - tty.setraw(fd) - yield fd - finally: - termios.tcsetattr(fd, termios.TCSADRAIN, old_settings) - sys.stdout.flush() - - if f is not None: - f.close() - except termios.error: - pass - - def getchar(echo: bool) -> str: - with raw_terminal() as fd: - ch = os.read(fd, 32).decode(get_best_encoding(sys.stdin), "replace") - - if echo and isatty(sys.stdout): - sys.stdout.write(ch) - - _translate_ch_to_exc(ch) - return ch diff --git a/.env/Lib/site-packages/click/_textwrap.py b/.env/Lib/site-packages/click/_textwrap.py deleted file mode 100644 index 97fbee3..0000000 --- a/.env/Lib/site-packages/click/_textwrap.py +++ /dev/null @@ -1,51 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import textwrap -from contextlib import contextmanager - - -class TextWrapper(textwrap.TextWrapper): - def _handle_long_word( - self, - reversed_chunks: list[str], - cur_line: list[str], - cur_len: int, - width: int, - ) -> None: - space_left = max(width - cur_len, 1) - - if self.break_long_words: - last = reversed_chunks[-1] - cut = last[:space_left] - res = last[space_left:] - cur_line.append(cut) - reversed_chunks[-1] = res - elif not cur_line: - cur_line.append(reversed_chunks.pop()) - - @contextmanager - def extra_indent(self, indent: str) -> cabc.Iterator[None]: - old_initial_indent = self.initial_indent - old_subsequent_indent = self.subsequent_indent - self.initial_indent += indent - self.subsequent_indent += indent - - try: - yield - finally: - self.initial_indent = old_initial_indent - self.subsequent_indent = old_subsequent_indent - - def indent_only(self, text: str) -> str: - rv = [] - - for idx, line in enumerate(text.splitlines()): - indent = self.initial_indent - - if idx > 0: - indent = self.subsequent_indent - - rv.append(f"{indent}{line}") - - return "\n".join(rv) diff --git a/.env/Lib/site-packages/click/_utils.py b/.env/Lib/site-packages/click/_utils.py deleted file mode 100644 index 09fb008..0000000 --- a/.env/Lib/site-packages/click/_utils.py +++ /dev/null @@ -1,36 +0,0 @@ -from __future__ import annotations - -import enum -import typing as t - - -class Sentinel(enum.Enum): - """Enum used to define sentinel values. - - .. seealso:: - - `PEP 661 - Sentinel Values `_. - """ - - UNSET = object() - FLAG_NEEDS_VALUE = object() - - def __repr__(self) -> str: - return f"{self.__class__.__name__}.{self.name}" - - -UNSET = Sentinel.UNSET -"""Sentinel used to indicate that a value is not set.""" - -FLAG_NEEDS_VALUE = Sentinel.FLAG_NEEDS_VALUE -"""Sentinel used to indicate an option was passed as a flag without a -value but is not a flag option. - -``Option.consume_value`` uses this to prompt or use the ``flag_value``. -""" - -T_UNSET = t.Literal[UNSET] # type: ignore[valid-type] -"""Type hint for the :data:`UNSET` sentinel value.""" - -T_FLAG_NEEDS_VALUE = t.Literal[FLAG_NEEDS_VALUE] # type: ignore[valid-type] -"""Type hint for the :data:`FLAG_NEEDS_VALUE` sentinel value.""" diff --git a/.env/Lib/site-packages/click/_winconsole.py b/.env/Lib/site-packages/click/_winconsole.py deleted file mode 100644 index e56c7c6..0000000 --- a/.env/Lib/site-packages/click/_winconsole.py +++ /dev/null @@ -1,296 +0,0 @@ -# This module is based on the excellent work by Adam Bartoš who -# provided a lot of what went into the implementation here in -# the discussion to issue1602 in the Python bug tracker. -# -# There are some general differences in regards to how this works -# compared to the original patches as we do not need to patch -# the entire interpreter but just work in our little world of -# echo and prompt. -from __future__ import annotations - -import collections.abc as cabc -import io -import sys -import time -import typing as t -from ctypes import Array -from ctypes import byref -from ctypes import c_char -from ctypes import c_char_p -from ctypes import c_int -from ctypes import c_ssize_t -from ctypes import c_ulong -from ctypes import c_void_p -from ctypes import POINTER -from ctypes import py_object -from ctypes import Structure -from ctypes.wintypes import DWORD -from ctypes.wintypes import HANDLE -from ctypes.wintypes import LPCWSTR -from ctypes.wintypes import LPWSTR - -from ._compat import _NonClosingTextIOWrapper - -assert sys.platform == "win32" -import msvcrt # noqa: E402 -from ctypes import windll # noqa: E402 -from ctypes import WINFUNCTYPE # noqa: E402 - -c_ssize_p = POINTER(c_ssize_t) - -kernel32 = windll.kernel32 -GetStdHandle = kernel32.GetStdHandle -ReadConsoleW = kernel32.ReadConsoleW -WriteConsoleW = kernel32.WriteConsoleW -GetConsoleMode = kernel32.GetConsoleMode -GetLastError = kernel32.GetLastError -GetCommandLineW = WINFUNCTYPE(LPWSTR)(("GetCommandLineW", windll.kernel32)) -CommandLineToArgvW = WINFUNCTYPE(POINTER(LPWSTR), LPCWSTR, POINTER(c_int))( - ("CommandLineToArgvW", windll.shell32) -) -LocalFree = WINFUNCTYPE(c_void_p, c_void_p)(("LocalFree", windll.kernel32)) - -STDIN_HANDLE = GetStdHandle(-10) -STDOUT_HANDLE = GetStdHandle(-11) -STDERR_HANDLE = GetStdHandle(-12) - -PyBUF_SIMPLE = 0 -PyBUF_WRITABLE = 1 - -ERROR_SUCCESS = 0 -ERROR_NOT_ENOUGH_MEMORY = 8 -ERROR_OPERATION_ABORTED = 995 - -STDIN_FILENO = 0 -STDOUT_FILENO = 1 -STDERR_FILENO = 2 - -EOF = b"\x1a" -MAX_BYTES_WRITTEN = 32767 - -if t.TYPE_CHECKING: - try: - # Using `typing_extensions.Buffer` instead of `collections.abc` - # on Windows for some reason does not have `Sized` implemented. - from collections.abc import Buffer # type: ignore - except ImportError: - from typing_extensions import Buffer - -try: - from ctypes import pythonapi -except ImportError: - # On PyPy we cannot get buffers so our ability to operate here is - # severely limited. - get_buffer = None -else: - - class Py_buffer(Structure): - _fields_ = [ # noqa: RUF012 - ("buf", c_void_p), - ("obj", py_object), - ("len", c_ssize_t), - ("itemsize", c_ssize_t), - ("readonly", c_int), - ("ndim", c_int), - ("format", c_char_p), - ("shape", c_ssize_p), - ("strides", c_ssize_p), - ("suboffsets", c_ssize_p), - ("internal", c_void_p), - ] - - PyObject_GetBuffer = pythonapi.PyObject_GetBuffer - PyBuffer_Release = pythonapi.PyBuffer_Release - - def get_buffer(obj: Buffer, writable: bool = False) -> Array[c_char]: - buf = Py_buffer() - flags: int = PyBUF_WRITABLE if writable else PyBUF_SIMPLE - PyObject_GetBuffer(py_object(obj), byref(buf), flags) - - try: - buffer_type = c_char * buf.len - out: Array[c_char] = buffer_type.from_address(buf.buf) - return out - finally: - PyBuffer_Release(byref(buf)) - - -class _WindowsConsoleRawIOBase(io.RawIOBase): - def __init__(self, handle: int | None) -> None: - self.handle = handle - - def isatty(self) -> t.Literal[True]: - super().isatty() - return True - - -class _WindowsConsoleReader(_WindowsConsoleRawIOBase): - def readable(self) -> t.Literal[True]: - return True - - def readinto(self, b: Buffer) -> int: - bytes_to_be_read = len(b) - if not bytes_to_be_read: - return 0 - elif bytes_to_be_read % 2: - raise ValueError( - "cannot read odd number of bytes from UTF-16-LE encoded console" - ) - - buffer = get_buffer(b, writable=True) - code_units_to_be_read = bytes_to_be_read // 2 - code_units_read = c_ulong() - - rv = ReadConsoleW( - HANDLE(self.handle), - buffer, - code_units_to_be_read, - byref(code_units_read), - None, - ) - if GetLastError() == ERROR_OPERATION_ABORTED: - # wait for KeyboardInterrupt - time.sleep(0.1) - if not rv: - raise OSError(f"Windows error: {GetLastError()}") - - if buffer[0] == EOF: - return 0 - return 2 * code_units_read.value - - -class _WindowsConsoleWriter(_WindowsConsoleRawIOBase): - def writable(self) -> t.Literal[True]: - return True - - @staticmethod - def _get_error_message(errno: int) -> str: - if errno == ERROR_SUCCESS: - return "ERROR_SUCCESS" - elif errno == ERROR_NOT_ENOUGH_MEMORY: - return "ERROR_NOT_ENOUGH_MEMORY" - return f"Windows error {errno}" - - def write(self, b: Buffer) -> int: - bytes_to_be_written = len(b) - buf = get_buffer(b) - code_units_to_be_written = min(bytes_to_be_written, MAX_BYTES_WRITTEN) // 2 - code_units_written = c_ulong() - - WriteConsoleW( - HANDLE(self.handle), - buf, - code_units_to_be_written, - byref(code_units_written), - None, - ) - bytes_written = 2 * code_units_written.value - - if bytes_written == 0 and bytes_to_be_written > 0: - raise OSError(self._get_error_message(GetLastError())) - return bytes_written - - -class ConsoleStream: - def __init__(self, text_stream: t.TextIO, byte_stream: t.BinaryIO) -> None: - self._text_stream = text_stream - self.buffer = byte_stream - - @property - def name(self) -> str: - return self.buffer.name - - def write(self, x: t.AnyStr) -> int: - if isinstance(x, str): - return self._text_stream.write(x) - try: - self.flush() - except Exception: - pass - return self.buffer.write(x) - - def writelines(self, lines: cabc.Iterable[t.AnyStr]) -> None: - for line in lines: - self.write(line) - - def __getattr__(self, name: str) -> t.Any: - return getattr(self._text_stream, name) - - def isatty(self) -> bool: - return self.buffer.isatty() - - def __repr__(self) -> str: - return f"" - - -def _get_text_stdin(buffer_stream: t.BinaryIO) -> t.TextIO: - text_stream = _NonClosingTextIOWrapper( - io.BufferedReader(_WindowsConsoleReader(STDIN_HANDLE)), - "utf-16-le", - "strict", - line_buffering=True, - ) - return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream)) - - -def _get_text_stdout(buffer_stream: t.BinaryIO) -> t.TextIO: - text_stream = _NonClosingTextIOWrapper( - io.BufferedWriter(_WindowsConsoleWriter(STDOUT_HANDLE)), - "utf-16-le", - "strict", - line_buffering=True, - ) - return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream)) - - -def _get_text_stderr(buffer_stream: t.BinaryIO) -> t.TextIO: - text_stream = _NonClosingTextIOWrapper( - io.BufferedWriter(_WindowsConsoleWriter(STDERR_HANDLE)), - "utf-16-le", - "strict", - line_buffering=True, - ) - return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream)) - - -_stream_factories: cabc.Mapping[int, t.Callable[[t.BinaryIO], t.TextIO]] = { - 0: _get_text_stdin, - 1: _get_text_stdout, - 2: _get_text_stderr, -} - - -def _is_console(f: t.TextIO) -> bool: - if not hasattr(f, "fileno"): - return False - - try: - fileno = f.fileno() - except (OSError, io.UnsupportedOperation): - return False - - handle = msvcrt.get_osfhandle(fileno) - return bool(GetConsoleMode(handle, byref(DWORD()))) - - -def _get_windows_console_stream( - f: t.TextIO, encoding: str | None, errors: str | None -) -> t.TextIO | None: - if ( - get_buffer is None - or encoding not in {"utf-16-le", None} - or errors not in {"strict", None} - or not _is_console(f) - ): - return None - - func = _stream_factories.get(f.fileno()) - if func is None: - return None - - b = getattr(f, "buffer", None) - - if b is None: - return None - - return func(b) diff --git a/.env/Lib/site-packages/click/core.py b/.env/Lib/site-packages/click/core.py deleted file mode 100644 index 57f549c..0000000 --- a/.env/Lib/site-packages/click/core.py +++ /dev/null @@ -1,3415 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import enum -import errno -import inspect -import os -import sys -import typing as t -from collections import abc -from collections import Counter -from contextlib import AbstractContextManager -from contextlib import contextmanager -from contextlib import ExitStack -from functools import update_wrapper -from gettext import gettext as _ -from gettext import ngettext -from itertools import repeat -from types import TracebackType - -from . import types -from ._utils import FLAG_NEEDS_VALUE -from ._utils import UNSET -from .exceptions import Abort -from .exceptions import BadParameter -from .exceptions import ClickException -from .exceptions import Exit -from .exceptions import MissingParameter -from .exceptions import NoArgsIsHelpError -from .exceptions import UsageError -from .formatting import HelpFormatter -from .formatting import join_options -from .globals import pop_context -from .globals import push_context -from .parser import _OptionParser -from .parser import _split_opt -from .termui import confirm -from .termui import prompt -from .termui import style -from .utils import _detect_program_name -from .utils import _expand_args -from .utils import echo -from .utils import make_default_short_help -from .utils import make_str -from .utils import PacifyFlushWrapper - -if t.TYPE_CHECKING: - from .shell_completion import CompletionItem - -F = t.TypeVar("F", bound="t.Callable[..., t.Any]") -V = t.TypeVar("V") - - -def _complete_visible_commands( - ctx: Context, incomplete: str -) -> cabc.Iterator[tuple[str, Command]]: - """List all the subcommands of a group that start with the - incomplete value and aren't hidden. - - :param ctx: Invocation context for the group. - :param incomplete: Value being completed. May be empty. - """ - multi = t.cast(Group, ctx.command) - - for name in multi.list_commands(ctx): - if name.startswith(incomplete): - command = multi.get_command(ctx, name) - - if command is not None and not command.hidden: - yield name, command - - -def _check_nested_chain( - base_command: Group, cmd_name: str, cmd: Command, register: bool = False -) -> None: - if not base_command.chain or not isinstance(cmd, Group): - return - - if register: - message = ( - f"It is not possible to add the group {cmd_name!r} to another" - f" group {base_command.name!r} that is in chain mode." - ) - else: - message = ( - f"Found the group {cmd_name!r} as subcommand to another group " - f" {base_command.name!r} that is in chain mode. This is not supported." - ) - - raise RuntimeError(message) - - -def batch(iterable: cabc.Iterable[V], batch_size: int) -> list[tuple[V, ...]]: - return list(zip(*repeat(iter(iterable), batch_size), strict=False)) - - -@contextmanager -def augment_usage_errors( - ctx: Context, param: Parameter | None = None -) -> cabc.Iterator[None]: - """Context manager that attaches extra information to exceptions.""" - try: - yield - except BadParameter as e: - if e.ctx is None: - e.ctx = ctx - if param is not None and e.param is None: - e.param = param - raise - except UsageError as e: - if e.ctx is None: - e.ctx = ctx - raise - - -def iter_params_for_processing( - invocation_order: cabc.Sequence[Parameter], - declaration_order: cabc.Sequence[Parameter], -) -> list[Parameter]: - """Returns all declared parameters in the order they should be processed. - - The declared parameters are re-shuffled depending on the order in which - they were invoked, as well as the eagerness of each parameters. - - The invocation order takes precedence over the declaration order. I.e. the - order in which the user provided them to the CLI is respected. - - This behavior and its effect on callback evaluation is detailed at: - https://click.palletsprojects.com/en/stable/advanced/#callback-evaluation-order - """ - - def sort_key(item: Parameter) -> tuple[bool, float]: - try: - idx: float = invocation_order.index(item) - except ValueError: - idx = float("inf") - - return not item.is_eager, idx - - return sorted(declaration_order, key=sort_key) - - -class ParameterSource(enum.Enum): - """This is an :class:`~enum.Enum` that indicates the source of a - parameter's value. - - Use :meth:`click.Context.get_parameter_source` to get the - source for a parameter by name. - - .. versionchanged:: 8.0 - Use :class:`~enum.Enum` and drop the ``validate`` method. - - .. versionchanged:: 8.0 - Added the ``PROMPT`` value. - """ - - COMMANDLINE = enum.auto() - """The value was provided by the command line args.""" - ENVIRONMENT = enum.auto() - """The value was provided with an environment variable.""" - DEFAULT = enum.auto() - """Used the default specified by the parameter.""" - DEFAULT_MAP = enum.auto() - """Used a default provided by :attr:`Context.default_map`.""" - PROMPT = enum.auto() - """Used a prompt to confirm a default or provide a value.""" - - -class Context: - """The context is a special internal object that holds state relevant - for the script execution at every single level. It's normally invisible - to commands unless they opt-in to getting access to it. - - The context is useful as it can pass internal objects around and can - control special execution features such as reading data from - environment variables. - - A context can be used as context manager in which case it will call - :meth:`close` on teardown. - - :param command: the command class for this context. - :param parent: the parent context. - :param info_name: the info name for this invocation. Generally this - is the most descriptive name for the script or - command. For the toplevel script it is usually - the name of the script, for commands below it it's - the name of the script. - :param obj: an arbitrary object of user data. - :param auto_envvar_prefix: the prefix to use for automatic environment - variables. If this is `None` then reading - from environment variables is disabled. This - does not affect manually set environment - variables which are always read. - :param default_map: a dictionary (like object) with default values - for parameters. - :param terminal_width: the width of the terminal. The default is - inherit from parent context. If no context - defines the terminal width then auto - detection will be applied. - :param max_content_width: the maximum width for content rendered by - Click (this currently only affects help - pages). This defaults to 80 characters if - not overridden. In other words: even if the - terminal is larger than that, Click will not - format things wider than 80 characters by - default. In addition to that, formatters might - add some safety mapping on the right. - :param resilient_parsing: if this flag is enabled then Click will - parse without any interactivity or callback - invocation. Default values will also be - ignored. This is useful for implementing - things such as completion support. - :param allow_extra_args: if this is set to `True` then extra arguments - at the end will not raise an error and will be - kept on the context. The default is to inherit - from the command. - :param allow_interspersed_args: if this is set to `False` then options - and arguments cannot be mixed. The - default is to inherit from the command. - :param ignore_unknown_options: instructs click to ignore options it does - not know and keeps them for later - processing. - :param help_option_names: optionally a list of strings that define how - the default help parameter is named. The - default is ``['--help']``. - :param token_normalize_func: an optional function that is used to - normalize tokens (options, choices, - etc.). This for instance can be used to - implement case insensitive behavior. - :param color: controls if the terminal supports ANSI colors or not. The - default is autodetection. This is only needed if ANSI - codes are used in texts that Click prints which is by - default not the case. This for instance would affect - help output. - :param show_default: Show the default value for commands. If this - value is not set, it defaults to the value from the parent - context. ``Command.show_default`` overrides this default for the - specific command. - - .. versionchanged:: 8.2 - The ``protected_args`` attribute is deprecated and will be removed in - Click 9.0. ``args`` will contain remaining unparsed tokens. - - .. versionchanged:: 8.1 - The ``show_default`` parameter is overridden by - ``Command.show_default``, instead of the other way around. - - .. versionchanged:: 8.0 - The ``show_default`` parameter defaults to the value from the - parent context. - - .. versionchanged:: 7.1 - Added the ``show_default`` parameter. - - .. versionchanged:: 4.0 - Added the ``color``, ``ignore_unknown_options``, and - ``max_content_width`` parameters. - - .. versionchanged:: 3.0 - Added the ``allow_extra_args`` and ``allow_interspersed_args`` - parameters. - - .. versionchanged:: 2.0 - Added the ``resilient_parsing``, ``help_option_names``, and - ``token_normalize_func`` parameters. - """ - - #: The formatter class to create with :meth:`make_formatter`. - #: - #: .. versionadded:: 8.0 - formatter_class: type[HelpFormatter] = HelpFormatter - - def __init__( - self, - command: Command, - parent: Context | None = None, - info_name: str | None = None, - obj: t.Any | None = None, - auto_envvar_prefix: str | None = None, - default_map: cabc.MutableMapping[str, t.Any] | None = None, - terminal_width: int | None = None, - max_content_width: int | None = None, - resilient_parsing: bool = False, - allow_extra_args: bool | None = None, - allow_interspersed_args: bool | None = None, - ignore_unknown_options: bool | None = None, - help_option_names: list[str] | None = None, - token_normalize_func: t.Callable[[str], str] | None = None, - color: bool | None = None, - show_default: bool | None = None, - ) -> None: - #: the parent context or `None` if none exists. - self.parent = parent - #: the :class:`Command` for this context. - self.command = command - #: the descriptive information name - self.info_name = info_name - #: Map of parameter names to their parsed values. Parameters - #: with ``expose_value=False`` are not stored. - self.params: dict[str, t.Any] = {} - #: the leftover arguments. - self.args: list[str] = [] - #: protected arguments. These are arguments that are prepended - #: to `args` when certain parsing scenarios are encountered but - #: must be never propagated to another arguments. This is used - #: to implement nested parsing. - self._protected_args: list[str] = [] - #: the collected prefixes of the command's options. - self._opt_prefixes: set[str] = set(parent._opt_prefixes) if parent else set() - - if obj is None and parent is not None: - obj = parent.obj - - #: the user object stored. - self.obj: t.Any = obj - self._meta: dict[str, t.Any] = getattr(parent, "meta", {}) - - #: A dictionary (-like object) with defaults for parameters. - if ( - default_map is None - and info_name is not None - and parent is not None - and parent.default_map is not None - ): - default_map = parent.default_map.get(info_name) - - self.default_map: cabc.MutableMapping[str, t.Any] | None = default_map - - #: This flag indicates if a subcommand is going to be executed. A - #: group callback can use this information to figure out if it's - #: being executed directly or because the execution flow passes - #: onwards to a subcommand. By default it's None, but it can be - #: the name of the subcommand to execute. - #: - #: If chaining is enabled this will be set to ``'*'`` in case - #: any commands are executed. It is however not possible to - #: figure out which ones. If you require this knowledge you - #: should use a :func:`result_callback`. - self.invoked_subcommand: str | None = None - - if terminal_width is None and parent is not None: - terminal_width = parent.terminal_width - - #: The width of the terminal (None is autodetection). - self.terminal_width: int | None = terminal_width - - if max_content_width is None and parent is not None: - max_content_width = parent.max_content_width - - #: The maximum width of formatted content (None implies a sensible - #: default which is 80 for most things). - self.max_content_width: int | None = max_content_width - - if allow_extra_args is None: - allow_extra_args = command.allow_extra_args - - #: Indicates if the context allows extra args or if it should - #: fail on parsing. - #: - #: .. versionadded:: 3.0 - self.allow_extra_args = allow_extra_args - - if allow_interspersed_args is None: - allow_interspersed_args = command.allow_interspersed_args - - #: Indicates if the context allows mixing of arguments and - #: options or not. - #: - #: .. versionadded:: 3.0 - self.allow_interspersed_args: bool = allow_interspersed_args - - if ignore_unknown_options is None: - ignore_unknown_options = command.ignore_unknown_options - - #: Instructs click to ignore options that a command does not - #: understand and will store it on the context for later - #: processing. This is primarily useful for situations where you - #: want to call into external programs. Generally this pattern is - #: strongly discouraged because it's not possibly to losslessly - #: forward all arguments. - #: - #: .. versionadded:: 4.0 - self.ignore_unknown_options: bool = ignore_unknown_options - - if help_option_names is None: - if parent is not None: - help_option_names = parent.help_option_names - else: - help_option_names = ["--help"] - - #: The names for the help options. - self.help_option_names: list[str] = help_option_names - - if token_normalize_func is None and parent is not None: - token_normalize_func = parent.token_normalize_func - - #: An optional normalization function for tokens. This is - #: options, choices, commands etc. - self.token_normalize_func: t.Callable[[str], str] | None = token_normalize_func - - #: Indicates if resilient parsing is enabled. In that case Click - #: will do its best to not cause any failures and default values - #: will be ignored. Useful for completion. - self.resilient_parsing: bool = resilient_parsing - - # If there is no envvar prefix yet, but the parent has one and - # the command on this level has a name, we can expand the envvar - # prefix automatically. - if auto_envvar_prefix is None: - if ( - parent is not None - and parent.auto_envvar_prefix is not None - and self.info_name is not None - ): - auto_envvar_prefix = ( - f"{parent.auto_envvar_prefix}_{self.info_name.upper()}" - ) - else: - auto_envvar_prefix = auto_envvar_prefix.upper() - - if auto_envvar_prefix is not None: - auto_envvar_prefix = auto_envvar_prefix.replace("-", "_") - - self.auto_envvar_prefix: str | None = auto_envvar_prefix - - if color is None and parent is not None: - color = parent.color - - #: Controls if styling output is wanted or not. - self.color: bool | None = color - - if show_default is None and parent is not None: - show_default = parent.show_default - - #: Show option default values when formatting help text. - self.show_default: bool | None = show_default - - self._close_callbacks: list[t.Callable[[], t.Any]] = [] - self._depth = 0 - self._parameter_source: dict[str, ParameterSource] = {} - self._exit_stack = ExitStack() - - @property - def protected_args(self) -> list[str]: - import warnings - - warnings.warn( - "'protected_args' is deprecated and will be removed in Click 9.0." - " 'args' will contain remaining unparsed tokens.", - DeprecationWarning, - stacklevel=2, - ) - return self._protected_args - - def to_info_dict(self) -> dict[str, t.Any]: - """Gather information that could be useful for a tool generating - user-facing documentation. This traverses the entire CLI - structure. - - .. code-block:: python - - with Context(cli) as ctx: - info = ctx.to_info_dict() - - .. versionadded:: 8.0 - """ - return { - "command": self.command.to_info_dict(self), - "info_name": self.info_name, - "allow_extra_args": self.allow_extra_args, - "allow_interspersed_args": self.allow_interspersed_args, - "ignore_unknown_options": self.ignore_unknown_options, - "auto_envvar_prefix": self.auto_envvar_prefix, - } - - def __enter__(self) -> Context: - self._depth += 1 - push_context(self) - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> bool | None: - self._depth -= 1 - exit_result: bool | None = None - if self._depth == 0: - exit_result = self._close_with_exception_info(exc_type, exc_value, tb) - pop_context() - - return exit_result - - @contextmanager - def scope(self, cleanup: bool = True) -> cabc.Iterator[Context]: - """This helper method can be used with the context object to promote - it to the current thread local (see :func:`get_current_context`). - The default behavior of this is to invoke the cleanup functions which - can be disabled by setting `cleanup` to `False`. The cleanup - functions are typically used for things such as closing file handles. - - If the cleanup is intended the context object can also be directly - used as a context manager. - - Example usage:: - - with ctx.scope(): - assert get_current_context() is ctx - - This is equivalent:: - - with ctx: - assert get_current_context() is ctx - - .. versionadded:: 5.0 - - :param cleanup: controls if the cleanup functions should be run or - not. The default is to run these functions. In - some situations the context only wants to be - temporarily pushed in which case this can be disabled. - Nested pushes automatically defer the cleanup. - """ - if not cleanup: - self._depth += 1 - try: - with self as rv: - yield rv - finally: - if not cleanup: - self._depth -= 1 - - @property - def meta(self) -> dict[str, t.Any]: - """This is a dictionary which is shared with all the contexts - that are nested. It exists so that click utilities can store some - state here if they need to. It is however the responsibility of - that code to manage this dictionary well. - - The keys are supposed to be unique dotted strings. For instance - module paths are a good choice for it. What is stored in there is - irrelevant for the operation of click. However what is important is - that code that places data here adheres to the general semantics of - the system. - - Example usage:: - - LANG_KEY = f'{__name__}.lang' - - def set_language(value): - ctx = get_current_context() - ctx.meta[LANG_KEY] = value - - def get_language(): - return get_current_context().meta.get(LANG_KEY, 'en_US') - - .. versionadded:: 5.0 - """ - return self._meta - - def make_formatter(self) -> HelpFormatter: - """Creates the :class:`~click.HelpFormatter` for the help and - usage output. - - To quickly customize the formatter class used without overriding - this method, set the :attr:`formatter_class` attribute. - - .. versionchanged:: 8.0 - Added the :attr:`formatter_class` attribute. - """ - return self.formatter_class( - width=self.terminal_width, max_width=self.max_content_width - ) - - def with_resource(self, context_manager: AbstractContextManager[V]) -> V: - """Register a resource as if it were used in a ``with`` - statement. The resource will be cleaned up when the context is - popped. - - Uses :meth:`contextlib.ExitStack.enter_context`. It calls the - resource's ``__enter__()`` method and returns the result. When - the context is popped, it closes the stack, which calls the - resource's ``__exit__()`` method. - - To register a cleanup function for something that isn't a - context manager, use :meth:`call_on_close`. Or use something - from :mod:`contextlib` to turn it into a context manager first. - - .. code-block:: python - - @click.group() - @click.option("--name") - @click.pass_context - def cli(ctx): - ctx.obj = ctx.with_resource(connect_db(name)) - - :param context_manager: The context manager to enter. - :return: Whatever ``context_manager.__enter__()`` returns. - - .. versionadded:: 8.0 - """ - return self._exit_stack.enter_context(context_manager) - - def call_on_close(self, f: t.Callable[..., t.Any]) -> t.Callable[..., t.Any]: - """Register a function to be called when the context tears down. - - This can be used to close resources opened during the script - execution. Resources that support Python's context manager - protocol which would be used in a ``with`` statement should be - registered with :meth:`with_resource` instead. - - :param f: The function to execute on teardown. - """ - return self._exit_stack.callback(f) - - def close(self) -> None: - """Invoke all close callbacks registered with - :meth:`call_on_close`, and exit all context managers entered - with :meth:`with_resource`. - """ - self._close_with_exception_info(None, None, None) - - def _close_with_exception_info( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> bool | None: - """Unwind the exit stack by calling its :meth:`__exit__` providing the exception - information to allow for exception handling by the various resources registered - using :meth;`with_resource` - - :return: Whatever ``exit_stack.__exit__()`` returns. - """ - exit_result = self._exit_stack.__exit__(exc_type, exc_value, tb) - # In case the context is reused, create a new exit stack. - self._exit_stack = ExitStack() - - return exit_result - - @property - def command_path(self) -> str: - """The computed command path. This is used for the ``usage`` - information on the help page. It's automatically created by - combining the info names of the chain of contexts to the root. - """ - rv = "" - if self.info_name is not None: - rv = self.info_name - if self.parent is not None: - parent_command_path = [self.parent.command_path] - - if isinstance(self.parent.command, Command): - for param in self.parent.command.get_params(self): - parent_command_path.extend(param.get_usage_pieces(self)) - - rv = f"{' '.join(parent_command_path)} {rv}" - return rv.lstrip() - - def find_root(self) -> Context: - """Finds the outermost context.""" - node = self - while node.parent is not None: - node = node.parent - return node - - def find_object(self, object_type: type[V]) -> V | None: - """Finds the closest object of a given type.""" - node: Context | None = self - - while node is not None: - if isinstance(node.obj, object_type): - return node.obj - - node = node.parent - - return None - - def ensure_object(self, object_type: type[V]) -> V: - """Like :meth:`find_object` but sets the innermost object to a - new instance of `object_type` if it does not exist. - """ - rv = self.find_object(object_type) - if rv is None: - self.obj = rv = object_type() - return rv - - @t.overload - def lookup_default( - self, name: str, call: t.Literal[True] = True - ) -> t.Any | None: ... - - @t.overload - def lookup_default( - self, name: str, call: t.Literal[False] = ... - ) -> t.Any | t.Callable[[], t.Any] | None: ... - - def lookup_default(self, name: str, call: bool = True) -> t.Any | None: - """Get the default for a parameter from :attr:`default_map`. - - :param name: Name of the parameter. - :param call: If the default is a callable, call it. Disable to - return the callable instead. - - .. versionchanged:: 8.0 - Added the ``call`` parameter. - """ - if self.default_map is not None: - value = self.default_map.get(name, UNSET) - - if call and callable(value): - return value() - - return value - - return UNSET - - def fail(self, message: str) -> t.NoReturn: - """Aborts the execution of the program with a specific error - message. - - :param message: the error message to fail with. - """ - raise UsageError(message, self) - - def abort(self) -> t.NoReturn: - """Aborts the script.""" - raise Abort() - - def exit(self, code: int = 0) -> t.NoReturn: - """Exits the application with a given exit code. - - .. versionchanged:: 8.2 - Callbacks and context managers registered with :meth:`call_on_close` - and :meth:`with_resource` are closed before exiting. - """ - self.close() - raise Exit(code) - - def get_usage(self) -> str: - """Helper method to get formatted usage string for the current - context and command. - """ - return self.command.get_usage(self) - - def get_help(self) -> str: - """Helper method to get formatted help page for the current - context and command. - """ - return self.command.get_help(self) - - def _make_sub_context(self, command: Command) -> Context: - """Create a new context of the same type as this context, but - for a new command. - - :meta private: - """ - return type(self)(command, info_name=command.name, parent=self) - - @t.overload - def invoke( - self, callback: t.Callable[..., V], /, *args: t.Any, **kwargs: t.Any - ) -> V: ... - - @t.overload - def invoke(self, callback: Command, /, *args: t.Any, **kwargs: t.Any) -> t.Any: ... - - def invoke( - self, callback: Command | t.Callable[..., V], /, *args: t.Any, **kwargs: t.Any - ) -> t.Any | V: - """Invokes a command callback in exactly the way it expects. There - are two ways to invoke this method: - - 1. the first argument can be a callback and all other arguments and - keyword arguments are forwarded directly to the function. - 2. the first argument is a click command object. In that case all - arguments are forwarded as well but proper click parameters - (options and click arguments) must be keyword arguments and Click - will fill in defaults. - - .. versionchanged:: 8.0 - All ``kwargs`` are tracked in :attr:`params` so they will be - passed if :meth:`forward` is called at multiple levels. - - .. versionchanged:: 3.2 - A new context is created, and missing arguments use default values. - """ - if isinstance(callback, Command): - other_cmd = callback - - if other_cmd.callback is None: - raise TypeError( - "The given command does not have a callback that can be invoked." - ) - else: - callback = t.cast("t.Callable[..., V]", other_cmd.callback) - - ctx = self._make_sub_context(other_cmd) - - for param in other_cmd.params: - if param.name not in kwargs and param.expose_value: - default_value = param.get_default(ctx) - # We explicitly hide the :attr:`UNSET` value to the user, as we - # choose to make it an implementation detail. And because ``invoke`` - # has been designed as part of Click public API, we return ``None`` - # instead. Refs: - # https://github.com/pallets/click/issues/3066 - # https://github.com/pallets/click/issues/3065 - # https://github.com/pallets/click/pull/3068 - if default_value is UNSET: - default_value = None - kwargs[param.name] = param.type_cast_value( # type: ignore - ctx, default_value - ) - - # Track all kwargs as params, so that forward() will pass - # them on in subsequent calls. - ctx.params.update(kwargs) - else: - ctx = self - - with augment_usage_errors(self): - with ctx: - return callback(*args, **kwargs) - - def forward(self, cmd: Command, /, *args: t.Any, **kwargs: t.Any) -> t.Any: - """Similar to :meth:`invoke` but fills in default keyword - arguments from the current context if the other command expects - it. This cannot invoke callbacks directly, only other commands. - - .. versionchanged:: 8.0 - All ``kwargs`` are tracked in :attr:`params` so they will be - passed if ``forward`` is called at multiple levels. - """ - # Can only forward to other commands, not direct callbacks. - if not isinstance(cmd, Command): - raise TypeError("Callback is not a command.") - - for param in self.params: - if param not in kwargs: - kwargs[param] = self.params[param] - - return self.invoke(cmd, *args, **kwargs) - - def set_parameter_source(self, name: str, source: ParameterSource) -> None: - """Set the source of a parameter. This indicates the location - from which the value of the parameter was obtained. - - :param name: The name of the parameter. - :param source: A member of :class:`~click.core.ParameterSource`. - """ - self._parameter_source[name] = source - - def get_parameter_source(self, name: str) -> ParameterSource | None: - """Get the source of a parameter. This indicates the location - from which the value of the parameter was obtained. - - This can be useful for determining when a user specified a value - on the command line that is the same as the default value. It - will be :attr:`~click.core.ParameterSource.DEFAULT` only if the - value was actually taken from the default. - - :param name: The name of the parameter. - :rtype: ParameterSource - - .. versionchanged:: 8.0 - Returns ``None`` if the parameter was not provided from any - source. - """ - return self._parameter_source.get(name) - - -class Command: - """Commands are the basic building block of command line interfaces in - Click. A basic command handles command line parsing and might dispatch - more parsing to commands nested below it. - - :param name: the name of the command to use unless a group overrides it. - :param context_settings: an optional dictionary with defaults that are - passed to the context object. - :param callback: the callback to invoke. This is optional. - :param params: the parameters to register with this command. This can - be either :class:`Option` or :class:`Argument` objects. - :param help: the help string to use for this command. - :param epilog: like the help string but it's printed at the end of the - help page after everything else. - :param short_help: the short help to use for this command. This is - shown on the command listing of the parent command. - :param add_help_option: by default each command registers a ``--help`` - option. This can be disabled by this parameter. - :param no_args_is_help: this controls what happens if no arguments are - provided. This option is disabled by default. - If enabled this will add ``--help`` as argument - if no arguments are passed - :param hidden: hide this command from help outputs. - :param deprecated: If ``True`` or non-empty string, issues a message - indicating that the command is deprecated and highlights - its deprecation in --help. The message can be customized - by using a string as the value. - - .. versionchanged:: 8.2 - This is the base class for all commands, not ``BaseCommand``. - ``deprecated`` can be set to a string as well to customize the - deprecation message. - - .. versionchanged:: 8.1 - ``help``, ``epilog``, and ``short_help`` are stored unprocessed, - all formatting is done when outputting help text, not at init, - and is done even if not using the ``@command`` decorator. - - .. versionchanged:: 8.0 - Added a ``repr`` showing the command name. - - .. versionchanged:: 7.1 - Added the ``no_args_is_help`` parameter. - - .. versionchanged:: 2.0 - Added the ``context_settings`` parameter. - """ - - #: The context class to create with :meth:`make_context`. - #: - #: .. versionadded:: 8.0 - context_class: type[Context] = Context - - #: the default for the :attr:`Context.allow_extra_args` flag. - allow_extra_args = False - - #: the default for the :attr:`Context.allow_interspersed_args` flag. - allow_interspersed_args = True - - #: the default for the :attr:`Context.ignore_unknown_options` flag. - ignore_unknown_options = False - - def __init__( - self, - name: str | None, - context_settings: cabc.MutableMapping[str, t.Any] | None = None, - callback: t.Callable[..., t.Any] | None = None, - params: list[Parameter] | None = None, - help: str | None = None, - epilog: str | None = None, - short_help: str | None = None, - options_metavar: str | None = "[OPTIONS]", - add_help_option: bool = True, - no_args_is_help: bool = False, - hidden: bool = False, - deprecated: bool | str = False, - ) -> None: - #: the name the command thinks it has. Upon registering a command - #: on a :class:`Group` the group will default the command name - #: with this information. You should instead use the - #: :class:`Context`\'s :attr:`~Context.info_name` attribute. - self.name = name - - if context_settings is None: - context_settings = {} - - #: an optional dictionary with defaults passed to the context. - self.context_settings: cabc.MutableMapping[str, t.Any] = context_settings - - #: the callback to execute when the command fires. This might be - #: `None` in which case nothing happens. - self.callback = callback - #: the list of parameters for this command in the order they - #: should show up in the help page and execute. Eager parameters - #: will automatically be handled before non eager ones. - self.params: list[Parameter] = params or [] - self.help = help - self.epilog = epilog - self.options_metavar = options_metavar - self.short_help = short_help - self.add_help_option = add_help_option - self._help_option = None - self.no_args_is_help = no_args_is_help - self.hidden = hidden - self.deprecated = deprecated - - def to_info_dict(self, ctx: Context) -> dict[str, t.Any]: - return { - "name": self.name, - "params": [param.to_info_dict() for param in self.get_params(ctx)], - "help": self.help, - "epilog": self.epilog, - "short_help": self.short_help, - "hidden": self.hidden, - "deprecated": self.deprecated, - } - - def __repr__(self) -> str: - return f"<{self.__class__.__name__} {self.name}>" - - def get_usage(self, ctx: Context) -> str: - """Formats the usage line into a string and returns it. - - Calls :meth:`format_usage` internally. - """ - formatter = ctx.make_formatter() - self.format_usage(ctx, formatter) - return formatter.getvalue().rstrip("\n") - - def get_params(self, ctx: Context) -> list[Parameter]: - params = self.params - help_option = self.get_help_option(ctx) - - if help_option is not None: - params = [*params, help_option] - - if __debug__: - import warnings - - opts = [opt for param in params for opt in param.opts] - opts_counter = Counter(opts) - duplicate_opts = (opt for opt, count in opts_counter.items() if count > 1) - - for duplicate_opt in duplicate_opts: - warnings.warn( - ( - f"The parameter {duplicate_opt} is used more than once. " - "Remove its duplicate as parameters should be unique." - ), - stacklevel=3, - ) - - return params - - def format_usage(self, ctx: Context, formatter: HelpFormatter) -> None: - """Writes the usage line into the formatter. - - This is a low-level method called by :meth:`get_usage`. - """ - pieces = self.collect_usage_pieces(ctx) - formatter.write_usage(ctx.command_path, " ".join(pieces)) - - def collect_usage_pieces(self, ctx: Context) -> list[str]: - """Returns all the pieces that go into the usage line and returns - it as a list of strings. - """ - rv = [self.options_metavar] if self.options_metavar else [] - - for param in self.get_params(ctx): - rv.extend(param.get_usage_pieces(ctx)) - - return rv - - def get_help_option_names(self, ctx: Context) -> list[str]: - """Returns the names for the help option.""" - all_names = set(ctx.help_option_names) - for param in self.params: - all_names.difference_update(param.opts) - all_names.difference_update(param.secondary_opts) - return list(all_names) - - def get_help_option(self, ctx: Context) -> Option | None: - """Returns the help option object. - - Skipped if :attr:`add_help_option` is ``False``. - - .. versionchanged:: 8.1.8 - The help option is now cached to avoid creating it multiple times. - """ - help_option_names = self.get_help_option_names(ctx) - - if not help_option_names or not self.add_help_option: - return None - - # Cache the help option object in private _help_option attribute to - # avoid creating it multiple times. Not doing this will break the - # callback odering by iter_params_for_processing(), which relies on - # object comparison. - if self._help_option is None: - # Avoid circular import. - from .decorators import help_option - - # Apply help_option decorator and pop resulting option - help_option(*help_option_names)(self) - self._help_option = self.params.pop() # type: ignore[assignment] - - return self._help_option - - def make_parser(self, ctx: Context) -> _OptionParser: - """Creates the underlying option parser for this command.""" - parser = _OptionParser(ctx) - for param in self.get_params(ctx): - param.add_to_parser(parser, ctx) - return parser - - def get_help(self, ctx: Context) -> str: - """Formats the help into a string and returns it. - - Calls :meth:`format_help` internally. - """ - formatter = ctx.make_formatter() - self.format_help(ctx, formatter) - return formatter.getvalue().rstrip("\n") - - def get_short_help_str(self, limit: int = 45) -> str: - """Gets short help for the command or makes it by shortening the - long help string. - """ - if self.short_help: - text = inspect.cleandoc(self.short_help) - elif self.help: - text = make_default_short_help(self.help, limit) - else: - text = "" - - if self.deprecated: - deprecated_message = ( - f"(DEPRECATED: {self.deprecated})" - if isinstance(self.deprecated, str) - else "(DEPRECATED)" - ) - text = _("{text} {deprecated_message}").format( - text=text, deprecated_message=deprecated_message - ) - - return text.strip() - - def format_help(self, ctx: Context, formatter: HelpFormatter) -> None: - """Writes the help into the formatter if it exists. - - This is a low-level method called by :meth:`get_help`. - - This calls the following methods: - - - :meth:`format_usage` - - :meth:`format_help_text` - - :meth:`format_options` - - :meth:`format_epilog` - """ - self.format_usage(ctx, formatter) - self.format_help_text(ctx, formatter) - self.format_options(ctx, formatter) - self.format_epilog(ctx, formatter) - - def format_help_text(self, ctx: Context, formatter: HelpFormatter) -> None: - """Writes the help text to the formatter if it exists.""" - if self.help is not None: - # truncate the help text to the first form feed - text = inspect.cleandoc(self.help).partition("\f")[0] - else: - text = "" - - if self.deprecated: - deprecated_message = ( - f"(DEPRECATED: {self.deprecated})" - if isinstance(self.deprecated, str) - else "(DEPRECATED)" - ) - text = _("{text} {deprecated_message}").format( - text=text, deprecated_message=deprecated_message - ) - - if text: - formatter.write_paragraph() - - with formatter.indentation(): - formatter.write_text(text) - - def format_options(self, ctx: Context, formatter: HelpFormatter) -> None: - """Writes all the options into the formatter if they exist.""" - opts = [] - for param in self.get_params(ctx): - rv = param.get_help_record(ctx) - if rv is not None: - opts.append(rv) - - if opts: - with formatter.section(_("Options")): - formatter.write_dl(opts) - - def format_epilog(self, ctx: Context, formatter: HelpFormatter) -> None: - """Writes the epilog into the formatter if it exists.""" - if self.epilog: - epilog = inspect.cleandoc(self.epilog) - formatter.write_paragraph() - - with formatter.indentation(): - formatter.write_text(epilog) - - def make_context( - self, - info_name: str | None, - args: list[str], - parent: Context | None = None, - **extra: t.Any, - ) -> Context: - """This function when given an info name and arguments will kick - off the parsing and create a new :class:`Context`. It does not - invoke the actual command callback though. - - To quickly customize the context class used without overriding - this method, set the :attr:`context_class` attribute. - - :param info_name: the info name for this invocation. Generally this - is the most descriptive name for the script or - command. For the toplevel script it's usually - the name of the script, for commands below it's - the name of the command. - :param args: the arguments to parse as list of strings. - :param parent: the parent context if available. - :param extra: extra keyword arguments forwarded to the context - constructor. - - .. versionchanged:: 8.0 - Added the :attr:`context_class` attribute. - """ - for key, value in self.context_settings.items(): - if key not in extra: - extra[key] = value - - ctx = self.context_class(self, info_name=info_name, parent=parent, **extra) - - with ctx.scope(cleanup=False): - self.parse_args(ctx, args) - return ctx - - def parse_args(self, ctx: Context, args: list[str]) -> list[str]: - if not args and self.no_args_is_help and not ctx.resilient_parsing: - raise NoArgsIsHelpError(ctx) - - parser = self.make_parser(ctx) - opts, args, param_order = parser.parse_args(args=args) - - for param in iter_params_for_processing(param_order, self.get_params(ctx)): - _, args = param.handle_parse_result(ctx, opts, args) - - # We now have all parameters' values into `ctx.params`, but the data may contain - # the `UNSET` sentinel. - # Convert `UNSET` to `None` to ensure that the user doesn't see `UNSET`. - # - # Waiting until after the initial parse to convert allows us to treat `UNSET` - # more like a missing value when multiple params use the same name. - # Refs: - # https://github.com/pallets/click/issues/3071 - # https://github.com/pallets/click/pull/3079 - for name, value in ctx.params.items(): - if value is UNSET: - ctx.params[name] = None - - if args and not ctx.allow_extra_args and not ctx.resilient_parsing: - ctx.fail( - ngettext( - "Got unexpected extra argument ({args})", - "Got unexpected extra arguments ({args})", - len(args), - ).format(args=" ".join(map(str, args))) - ) - - ctx.args = args - ctx._opt_prefixes.update(parser._opt_prefixes) - return args - - def invoke(self, ctx: Context) -> t.Any: - """Given a context, this invokes the attached callback (if it exists) - in the right way. - """ - if self.deprecated: - extra_message = ( - f" {self.deprecated}" if isinstance(self.deprecated, str) else "" - ) - message = _( - "DeprecationWarning: The command {name!r} is deprecated.{extra_message}" - ).format(name=self.name, extra_message=extra_message) - echo(style(message, fg="red"), err=True) - - if self.callback is not None: - return ctx.invoke(self.callback, **ctx.params) - - def shell_complete(self, ctx: Context, incomplete: str) -> list[CompletionItem]: - """Return a list of completions for the incomplete value. Looks - at the names of options and chained multi-commands. - - Any command could be part of a chained multi-command, so sibling - commands are valid at any point during command completion. - - :param ctx: Invocation context for this command. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - from click.shell_completion import CompletionItem - - results: list[CompletionItem] = [] - - if incomplete and not incomplete[0].isalnum(): - for param in self.get_params(ctx): - if ( - not isinstance(param, Option) - or param.hidden - or ( - not param.multiple - and ctx.get_parameter_source(param.name) # type: ignore - is ParameterSource.COMMANDLINE - ) - ): - continue - - results.extend( - CompletionItem(name, help=param.help) - for name in [*param.opts, *param.secondary_opts] - if name.startswith(incomplete) - ) - - while ctx.parent is not None: - ctx = ctx.parent - - if isinstance(ctx.command, Group) and ctx.command.chain: - results.extend( - CompletionItem(name, help=command.get_short_help_str()) - for name, command in _complete_visible_commands(ctx, incomplete) - if name not in ctx._protected_args - ) - - return results - - @t.overload - def main( - self, - args: cabc.Sequence[str] | None = None, - prog_name: str | None = None, - complete_var: str | None = None, - standalone_mode: t.Literal[True] = True, - **extra: t.Any, - ) -> t.NoReturn: ... - - @t.overload - def main( - self, - args: cabc.Sequence[str] | None = None, - prog_name: str | None = None, - complete_var: str | None = None, - standalone_mode: bool = ..., - **extra: t.Any, - ) -> t.Any: ... - - def main( - self, - args: cabc.Sequence[str] | None = None, - prog_name: str | None = None, - complete_var: str | None = None, - standalone_mode: bool = True, - windows_expand_args: bool = True, - **extra: t.Any, - ) -> t.Any: - """This is the way to invoke a script with all the bells and - whistles as a command line application. This will always terminate - the application after a call. If this is not wanted, ``SystemExit`` - needs to be caught. - - This method is also available by directly calling the instance of - a :class:`Command`. - - :param args: the arguments that should be used for parsing. If not - provided, ``sys.argv[1:]`` is used. - :param prog_name: the program name that should be used. By default - the program name is constructed by taking the file - name from ``sys.argv[0]``. - :param complete_var: the environment variable that controls the - bash completion support. The default is - ``"__COMPLETE"`` with prog_name in - uppercase. - :param standalone_mode: the default behavior is to invoke the script - in standalone mode. Click will then - handle exceptions and convert them into - error messages and the function will never - return but shut down the interpreter. If - this is set to `False` they will be - propagated to the caller and the return - value of this function is the return value - of :meth:`invoke`. - :param windows_expand_args: Expand glob patterns, user dir, and - env vars in command line args on Windows. - :param extra: extra keyword arguments are forwarded to the context - constructor. See :class:`Context` for more information. - - .. versionchanged:: 8.0.1 - Added the ``windows_expand_args`` parameter to allow - disabling command line arg expansion on Windows. - - .. versionchanged:: 8.0 - When taking arguments from ``sys.argv`` on Windows, glob - patterns, user dir, and env vars are expanded. - - .. versionchanged:: 3.0 - Added the ``standalone_mode`` parameter. - """ - if args is None: - args = sys.argv[1:] - - if os.name == "nt" and windows_expand_args: - args = _expand_args(args) - else: - args = list(args) - - if prog_name is None: - prog_name = _detect_program_name() - - # Process shell completion requests and exit early. - self._main_shell_completion(extra, prog_name, complete_var) - - try: - try: - with self.make_context(prog_name, args, **extra) as ctx: - rv = self.invoke(ctx) - if not standalone_mode: - return rv - # it's not safe to `ctx.exit(rv)` here! - # note that `rv` may actually contain data like "1" which - # has obvious effects - # more subtle case: `rv=[None, None]` can come out of - # chained commands which all returned `None` -- so it's not - # even always obvious that `rv` indicates success/failure - # by its truthiness/falsiness - ctx.exit() - except (EOFError, KeyboardInterrupt) as e: - echo(file=sys.stderr) - raise Abort() from e - except ClickException as e: - if not standalone_mode: - raise - e.show() - sys.exit(e.exit_code) - except OSError as e: - if e.errno == errno.EPIPE: - sys.stdout = t.cast(t.TextIO, PacifyFlushWrapper(sys.stdout)) - sys.stderr = t.cast(t.TextIO, PacifyFlushWrapper(sys.stderr)) - sys.exit(1) - else: - raise - except Exit as e: - if standalone_mode: - sys.exit(e.exit_code) - else: - # in non-standalone mode, return the exit code - # note that this is only reached if `self.invoke` above raises - # an Exit explicitly -- thus bypassing the check there which - # would return its result - # the results of non-standalone execution may therefore be - # somewhat ambiguous: if there are codepaths which lead to - # `ctx.exit(1)` and to `return 1`, the caller won't be able to - # tell the difference between the two - return e.exit_code - except Abort: - if not standalone_mode: - raise - echo(_("Aborted!"), file=sys.stderr) - sys.exit(1) - - def _main_shell_completion( - self, - ctx_args: cabc.MutableMapping[str, t.Any], - prog_name: str, - complete_var: str | None = None, - ) -> None: - """Check if the shell is asking for tab completion, process - that, then exit early. Called from :meth:`main` before the - program is invoked. - - :param prog_name: Name of the executable in the shell. - :param complete_var: Name of the environment variable that holds - the completion instruction. Defaults to - ``_{PROG_NAME}_COMPLETE``. - - .. versionchanged:: 8.2.0 - Dots (``.``) in ``prog_name`` are replaced with underscores (``_``). - """ - if complete_var is None: - complete_name = prog_name.replace("-", "_").replace(".", "_") - complete_var = f"_{complete_name}_COMPLETE".upper() - - instruction = os.environ.get(complete_var) - - if not instruction: - return - - from .shell_completion import shell_complete - - rv = shell_complete(self, ctx_args, prog_name, complete_var, instruction) - sys.exit(rv) - - def __call__(self, *args: t.Any, **kwargs: t.Any) -> t.Any: - """Alias for :meth:`main`.""" - return self.main(*args, **kwargs) - - -class _FakeSubclassCheck(type): - def __subclasscheck__(cls, subclass: type) -> bool: - return issubclass(subclass, cls.__bases__[0]) - - def __instancecheck__(cls, instance: t.Any) -> bool: - return isinstance(instance, cls.__bases__[0]) - - -class _BaseCommand(Command, metaclass=_FakeSubclassCheck): - """ - .. deprecated:: 8.2 - Will be removed in Click 9.0. Use ``Command`` instead. - """ - - -class Group(Command): - """A group is a command that nests other commands (or more groups). - - :param name: The name of the group command. - :param commands: Map names to :class:`Command` objects. Can be a list, which - will use :attr:`Command.name` as the keys. - :param invoke_without_command: Invoke the group's callback even if a - subcommand is not given. - :param no_args_is_help: If no arguments are given, show the group's help and - exit. Defaults to the opposite of ``invoke_without_command``. - :param subcommand_metavar: How to represent the subcommand argument in help. - The default will represent whether ``chain`` is set or not. - :param chain: Allow passing more than one subcommand argument. After parsing - a command's arguments, if any arguments remain another command will be - matched, and so on. - :param result_callback: A function to call after the group's and - subcommand's callbacks. The value returned by the subcommand is passed. - If ``chain`` is enabled, the value will be a list of values returned by - all the commands. If ``invoke_without_command`` is enabled, the value - will be the value returned by the group's callback, or an empty list if - ``chain`` is enabled. - :param kwargs: Other arguments passed to :class:`Command`. - - .. versionchanged:: 8.0 - The ``commands`` argument can be a list of command objects. - - .. versionchanged:: 8.2 - Merged with and replaces the ``MultiCommand`` base class. - """ - - allow_extra_args = True - allow_interspersed_args = False - - #: If set, this is used by the group's :meth:`command` decorator - #: as the default :class:`Command` class. This is useful to make all - #: subcommands use a custom command class. - #: - #: .. versionadded:: 8.0 - command_class: type[Command] | None = None - - #: If set, this is used by the group's :meth:`group` decorator - #: as the default :class:`Group` class. This is useful to make all - #: subgroups use a custom group class. - #: - #: If set to the special value :class:`type` (literally - #: ``group_class = type``), this group's class will be used as the - #: default class. This makes a custom group class continue to make - #: custom groups. - #: - #: .. versionadded:: 8.0 - group_class: type[Group] | type[type] | None = None - # Literal[type] isn't valid, so use Type[type] - - def __init__( - self, - name: str | None = None, - commands: cabc.MutableMapping[str, Command] - | cabc.Sequence[Command] - | None = None, - invoke_without_command: bool = False, - no_args_is_help: bool | None = None, - subcommand_metavar: str | None = None, - chain: bool = False, - result_callback: t.Callable[..., t.Any] | None = None, - **kwargs: t.Any, - ) -> None: - super().__init__(name, **kwargs) - - if commands is None: - commands = {} - elif isinstance(commands, abc.Sequence): - commands = {c.name: c for c in commands if c.name is not None} - - #: The registered subcommands by their exported names. - self.commands: cabc.MutableMapping[str, Command] = commands - - if no_args_is_help is None: - no_args_is_help = not invoke_without_command - - self.no_args_is_help = no_args_is_help - self.invoke_without_command = invoke_without_command - - if subcommand_metavar is None: - if chain: - subcommand_metavar = "COMMAND1 [ARGS]... [COMMAND2 [ARGS]...]..." - else: - subcommand_metavar = "COMMAND [ARGS]..." - - self.subcommand_metavar = subcommand_metavar - self.chain = chain - # The result callback that is stored. This can be set or - # overridden with the :func:`result_callback` decorator. - self._result_callback = result_callback - - if self.chain: - for param in self.params: - if isinstance(param, Argument) and not param.required: - raise RuntimeError( - "A group in chain mode cannot have optional arguments." - ) - - def to_info_dict(self, ctx: Context) -> dict[str, t.Any]: - info_dict = super().to_info_dict(ctx) - commands = {} - - for name in self.list_commands(ctx): - command = self.get_command(ctx, name) - - if command is None: - continue - - sub_ctx = ctx._make_sub_context(command) - - with sub_ctx.scope(cleanup=False): - commands[name] = command.to_info_dict(sub_ctx) - - info_dict.update(commands=commands, chain=self.chain) - return info_dict - - def add_command(self, cmd: Command, name: str | None = None) -> None: - """Registers another :class:`Command` with this group. If the name - is not provided, the name of the command is used. - """ - name = name or cmd.name - if name is None: - raise TypeError("Command has no name.") - _check_nested_chain(self, name, cmd, register=True) - self.commands[name] = cmd - - @t.overload - def command(self, __func: t.Callable[..., t.Any]) -> Command: ... - - @t.overload - def command( - self, *args: t.Any, **kwargs: t.Any - ) -> t.Callable[[t.Callable[..., t.Any]], Command]: ... - - def command( - self, *args: t.Any, **kwargs: t.Any - ) -> t.Callable[[t.Callable[..., t.Any]], Command] | Command: - """A shortcut decorator for declaring and attaching a command to - the group. This takes the same arguments as :func:`command` and - immediately registers the created command with this group by - calling :meth:`add_command`. - - To customize the command class used, set the - :attr:`command_class` attribute. - - .. versionchanged:: 8.1 - This decorator can be applied without parentheses. - - .. versionchanged:: 8.0 - Added the :attr:`command_class` attribute. - """ - from .decorators import command - - func: t.Callable[..., t.Any] | None = None - - if args and callable(args[0]): - assert len(args) == 1 and not kwargs, ( - "Use 'command(**kwargs)(callable)' to provide arguments." - ) - (func,) = args - args = () - - if self.command_class and kwargs.get("cls") is None: - kwargs["cls"] = self.command_class - - def decorator(f: t.Callable[..., t.Any]) -> Command: - cmd: Command = command(*args, **kwargs)(f) - self.add_command(cmd) - return cmd - - if func is not None: - return decorator(func) - - return decorator - - @t.overload - def group(self, __func: t.Callable[..., t.Any]) -> Group: ... - - @t.overload - def group( - self, *args: t.Any, **kwargs: t.Any - ) -> t.Callable[[t.Callable[..., t.Any]], Group]: ... - - def group( - self, *args: t.Any, **kwargs: t.Any - ) -> t.Callable[[t.Callable[..., t.Any]], Group] | Group: - """A shortcut decorator for declaring and attaching a group to - the group. This takes the same arguments as :func:`group` and - immediately registers the created group with this group by - calling :meth:`add_command`. - - To customize the group class used, set the :attr:`group_class` - attribute. - - .. versionchanged:: 8.1 - This decorator can be applied without parentheses. - - .. versionchanged:: 8.0 - Added the :attr:`group_class` attribute. - """ - from .decorators import group - - func: t.Callable[..., t.Any] | None = None - - if args and callable(args[0]): - assert len(args) == 1 and not kwargs, ( - "Use 'group(**kwargs)(callable)' to provide arguments." - ) - (func,) = args - args = () - - if self.group_class is not None and kwargs.get("cls") is None: - if self.group_class is type: - kwargs["cls"] = type(self) - else: - kwargs["cls"] = self.group_class - - def decorator(f: t.Callable[..., t.Any]) -> Group: - cmd: Group = group(*args, **kwargs)(f) - self.add_command(cmd) - return cmd - - if func is not None: - return decorator(func) - - return decorator - - def result_callback(self, replace: bool = False) -> t.Callable[[F], F]: - """Adds a result callback to the command. By default if a - result callback is already registered this will chain them but - this can be disabled with the `replace` parameter. The result - callback is invoked with the return value of the subcommand - (or the list of return values from all subcommands if chaining - is enabled) as well as the parameters as they would be passed - to the main callback. - - Example:: - - @click.group() - @click.option('-i', '--input', default=23) - def cli(input): - return 42 - - @cli.result_callback() - def process_result(result, input): - return result + input - - :param replace: if set to `True` an already existing result - callback will be removed. - - .. versionchanged:: 8.0 - Renamed from ``resultcallback``. - - .. versionadded:: 3.0 - """ - - def decorator(f: F) -> F: - old_callback = self._result_callback - - if old_callback is None or replace: - self._result_callback = f - return f - - def function(value: t.Any, /, *args: t.Any, **kwargs: t.Any) -> t.Any: - inner = old_callback(value, *args, **kwargs) - return f(inner, *args, **kwargs) - - self._result_callback = rv = update_wrapper(t.cast(F, function), f) - return rv # type: ignore[return-value] - - return decorator - - def get_command(self, ctx: Context, cmd_name: str) -> Command | None: - """Given a context and a command name, this returns a :class:`Command` - object if it exists or returns ``None``. - """ - return self.commands.get(cmd_name) - - def list_commands(self, ctx: Context) -> list[str]: - """Returns a list of subcommand names in the order they should appear.""" - return sorted(self.commands) - - def collect_usage_pieces(self, ctx: Context) -> list[str]: - rv = super().collect_usage_pieces(ctx) - rv.append(self.subcommand_metavar) - return rv - - def format_options(self, ctx: Context, formatter: HelpFormatter) -> None: - super().format_options(ctx, formatter) - self.format_commands(ctx, formatter) - - def format_commands(self, ctx: Context, formatter: HelpFormatter) -> None: - """Extra format methods for multi methods that adds all the commands - after the options. - """ - commands = [] - for subcommand in self.list_commands(ctx): - cmd = self.get_command(ctx, subcommand) - # What is this, the tool lied about a command. Ignore it - if cmd is None: - continue - if cmd.hidden: - continue - - commands.append((subcommand, cmd)) - - # allow for 3 times the default spacing - if len(commands): - limit = formatter.width - 6 - max(len(cmd[0]) for cmd in commands) - - rows = [] - for subcommand, cmd in commands: - help = cmd.get_short_help_str(limit) - rows.append((subcommand, help)) - - if rows: - with formatter.section(_("Commands")): - formatter.write_dl(rows) - - def parse_args(self, ctx: Context, args: list[str]) -> list[str]: - if not args and self.no_args_is_help and not ctx.resilient_parsing: - raise NoArgsIsHelpError(ctx) - - rest = super().parse_args(ctx, args) - - if self.chain: - ctx._protected_args = rest - ctx.args = [] - elif rest: - ctx._protected_args, ctx.args = rest[:1], rest[1:] - - return ctx.args - - def invoke(self, ctx: Context) -> t.Any: - def _process_result(value: t.Any) -> t.Any: - if self._result_callback is not None: - value = ctx.invoke(self._result_callback, value, **ctx.params) - return value - - if not ctx._protected_args: - if self.invoke_without_command: - # No subcommand was invoked, so the result callback is - # invoked with the group return value for regular - # groups, or an empty list for chained groups. - with ctx: - rv = super().invoke(ctx) - return _process_result([] if self.chain else rv) - ctx.fail(_("Missing command.")) - - # Fetch args back out - args = [*ctx._protected_args, *ctx.args] - ctx.args = [] - ctx._protected_args = [] - - # If we're not in chain mode, we only allow the invocation of a - # single command but we also inform the current context about the - # name of the command to invoke. - if not self.chain: - # Make sure the context is entered so we do not clean up - # resources until the result processor has worked. - with ctx: - cmd_name, cmd, args = self.resolve_command(ctx, args) - assert cmd is not None - ctx.invoked_subcommand = cmd_name - super().invoke(ctx) - sub_ctx = cmd.make_context(cmd_name, args, parent=ctx) - with sub_ctx: - return _process_result(sub_ctx.command.invoke(sub_ctx)) - - # In chain mode we create the contexts step by step, but after the - # base command has been invoked. Because at that point we do not - # know the subcommands yet, the invoked subcommand attribute is - # set to ``*`` to inform the command that subcommands are executed - # but nothing else. - with ctx: - ctx.invoked_subcommand = "*" if args else None - super().invoke(ctx) - - # Otherwise we make every single context and invoke them in a - # chain. In that case the return value to the result processor - # is the list of all invoked subcommand's results. - contexts = [] - while args: - cmd_name, cmd, args = self.resolve_command(ctx, args) - assert cmd is not None - sub_ctx = cmd.make_context( - cmd_name, - args, - parent=ctx, - allow_extra_args=True, - allow_interspersed_args=False, - ) - contexts.append(sub_ctx) - args, sub_ctx.args = sub_ctx.args, [] - - rv = [] - for sub_ctx in contexts: - with sub_ctx: - rv.append(sub_ctx.command.invoke(sub_ctx)) - return _process_result(rv) - - def resolve_command( - self, ctx: Context, args: list[str] - ) -> tuple[str | None, Command | None, list[str]]: - cmd_name = make_str(args[0]) - original_cmd_name = cmd_name - - # Get the command - cmd = self.get_command(ctx, cmd_name) - - # If we can't find the command but there is a normalization - # function available, we try with that one. - if cmd is None and ctx.token_normalize_func is not None: - cmd_name = ctx.token_normalize_func(cmd_name) - cmd = self.get_command(ctx, cmd_name) - - # If we don't find the command we want to show an error message - # to the user that it was not provided. However, there is - # something else we should do: if the first argument looks like - # an option we want to kick off parsing again for arguments to - # resolve things like --help which now should go to the main - # place. - if cmd is None and not ctx.resilient_parsing: - if _split_opt(cmd_name)[0]: - self.parse_args(ctx, args) - ctx.fail(_("No such command {name!r}.").format(name=original_cmd_name)) - return cmd_name if cmd else None, cmd, args[1:] - - def shell_complete(self, ctx: Context, incomplete: str) -> list[CompletionItem]: - """Return a list of completions for the incomplete value. Looks - at the names of options, subcommands, and chained - multi-commands. - - :param ctx: Invocation context for this command. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - from click.shell_completion import CompletionItem - - results = [ - CompletionItem(name, help=command.get_short_help_str()) - for name, command in _complete_visible_commands(ctx, incomplete) - ] - results.extend(super().shell_complete(ctx, incomplete)) - return results - - -class _MultiCommand(Group, metaclass=_FakeSubclassCheck): - """ - .. deprecated:: 8.2 - Will be removed in Click 9.0. Use ``Group`` instead. - """ - - -class CommandCollection(Group): - """A :class:`Group` that looks up subcommands on other groups. If a command - is not found on this group, each registered source is checked in order. - Parameters on a source are not added to this group, and a source's callback - is not invoked when invoking its commands. In other words, this "flattens" - commands in many groups into this one group. - - :param name: The name of the group command. - :param sources: A list of :class:`Group` objects to look up commands from. - :param kwargs: Other arguments passed to :class:`Group`. - - .. versionchanged:: 8.2 - This is a subclass of ``Group``. Commands are looked up first on this - group, then each of its sources. - """ - - def __init__( - self, - name: str | None = None, - sources: list[Group] | None = None, - **kwargs: t.Any, - ) -> None: - super().__init__(name, **kwargs) - #: The list of registered groups. - self.sources: list[Group] = sources or [] - - def add_source(self, group: Group) -> None: - """Add a group as a source of commands.""" - self.sources.append(group) - - def get_command(self, ctx: Context, cmd_name: str) -> Command | None: - rv = super().get_command(ctx, cmd_name) - - if rv is not None: - return rv - - for source in self.sources: - rv = source.get_command(ctx, cmd_name) - - if rv is not None: - if self.chain: - _check_nested_chain(self, cmd_name, rv) - - return rv - - return None - - def list_commands(self, ctx: Context) -> list[str]: - rv: set[str] = set(super().list_commands(ctx)) - - for source in self.sources: - rv.update(source.list_commands(ctx)) - - return sorted(rv) - - -def _check_iter(value: t.Any) -> cabc.Iterator[t.Any]: - """Check if the value is iterable but not a string. Raises a type - error, or return an iterator over the value. - """ - if isinstance(value, str): - raise TypeError - - return iter(value) - - -class Parameter: - r"""A parameter to a command comes in two versions: they are either - :class:`Option`\s or :class:`Argument`\s. Other subclasses are currently - not supported by design as some of the internals for parsing are - intentionally not finalized. - - Some settings are supported by both options and arguments. - - :param param_decls: the parameter declarations for this option or - argument. This is a list of flags or argument - names. - :param type: the type that should be used. Either a :class:`ParamType` - or a Python type. The latter is converted into the former - automatically if supported. - :param required: controls if this is optional or not. - :param default: the default value if omitted. This can also be a callable, - in which case it's invoked when the default is needed - without any arguments. - :param callback: A function to further process or validate the value - after type conversion. It is called as ``f(ctx, param, value)`` - and must return the value. It is called for all sources, - including prompts. - :param nargs: the number of arguments to match. If not ``1`` the return - value is a tuple instead of single value. The default for - nargs is ``1`` (except if the type is a tuple, then it's - the arity of the tuple). If ``nargs=-1``, all remaining - parameters are collected. - :param metavar: how the value is represented in the help page. - :param expose_value: if this is `True` then the value is passed onwards - to the command callback and stored on the context, - otherwise it's skipped. - :param is_eager: eager values are processed before non eager ones. This - should not be set for arguments or it will inverse the - order of processing. - :param envvar: environment variable(s) that are used to provide a default value for - this parameter. This can be a string or a sequence of strings. If a sequence is - given, only the first non-empty environment variable is used for the parameter. - :param shell_complete: A function that returns custom shell - completions. Used instead of the param's type completion if - given. Takes ``ctx, param, incomplete`` and must return a list - of :class:`~click.shell_completion.CompletionItem` or a list of - strings. - :param deprecated: If ``True`` or non-empty string, issues a message - indicating that the argument is deprecated and highlights - its deprecation in --help. The message can be customized - by using a string as the value. A deprecated parameter - cannot be required, a ValueError will be raised otherwise. - - .. versionchanged:: 8.2.0 - Introduction of ``deprecated``. - - .. versionchanged:: 8.2 - Adding duplicate parameter names to a :class:`~click.core.Command` will - result in a ``UserWarning`` being shown. - - .. versionchanged:: 8.2 - Adding duplicate parameter names to a :class:`~click.core.Command` will - result in a ``UserWarning`` being shown. - - .. versionchanged:: 8.0 - ``process_value`` validates required parameters and bounded - ``nargs``, and invokes the parameter callback before returning - the value. This allows the callback to validate prompts. - ``full_process_value`` is removed. - - .. versionchanged:: 8.0 - ``autocompletion`` is renamed to ``shell_complete`` and has new - semantics described above. The old name is deprecated and will - be removed in 8.1, until then it will be wrapped to match the - new requirements. - - .. versionchanged:: 8.0 - For ``multiple=True, nargs>1``, the default must be a list of - tuples. - - .. versionchanged:: 8.0 - Setting a default is no longer required for ``nargs>1``, it will - default to ``None``. ``multiple=True`` or ``nargs=-1`` will - default to ``()``. - - .. versionchanged:: 7.1 - Empty environment variables are ignored rather than taking the - empty string value. This makes it possible for scripts to clear - variables if they can't unset them. - - .. versionchanged:: 2.0 - Changed signature for parameter callback to also be passed the - parameter. The old callback format will still work, but it will - raise a warning to give you a chance to migrate the code easier. - """ - - param_type_name = "parameter" - - def __init__( - self, - param_decls: cabc.Sequence[str] | None = None, - type: types.ParamType | t.Any | None = None, - required: bool = False, - # XXX The default historically embed two concepts: - # - the declaration of a Parameter object carrying the default (handy to - # arbitrage the default value of coupled Parameters sharing the same - # self.name, like flag options), - # - and the actual value of the default. - # It is confusing and is the source of many issues discussed in: - # https://github.com/pallets/click/pull/3030 - # In the future, we might think of splitting it in two, not unlike - # Option.is_flag and Option.flag_value: we could have something like - # Parameter.is_default and Parameter.default_value. - default: t.Any | t.Callable[[], t.Any] | None = UNSET, - callback: t.Callable[[Context, Parameter, t.Any], t.Any] | None = None, - nargs: int | None = None, - multiple: bool = False, - metavar: str | None = None, - expose_value: bool = True, - is_eager: bool = False, - envvar: str | cabc.Sequence[str] | None = None, - shell_complete: t.Callable[ - [Context, Parameter, str], list[CompletionItem] | list[str] - ] - | None = None, - deprecated: bool | str = False, - ) -> None: - self.name: str | None - self.opts: list[str] - self.secondary_opts: list[str] - self.name, self.opts, self.secondary_opts = self._parse_decls( - param_decls or (), expose_value - ) - self.type: types.ParamType = types.convert_type(type, default) - - # Default nargs to what the type tells us if we have that - # information available. - if nargs is None: - if self.type.is_composite: - nargs = self.type.arity - else: - nargs = 1 - - self.required = required - self.callback = callback - self.nargs = nargs - self.multiple = multiple - self.expose_value = expose_value - self.default: t.Any | t.Callable[[], t.Any] | None = default - self.is_eager = is_eager - self.metavar = metavar - self.envvar = envvar - self._custom_shell_complete = shell_complete - self.deprecated = deprecated - - if __debug__: - if self.type.is_composite and nargs != self.type.arity: - raise ValueError( - f"'nargs' must be {self.type.arity} (or None) for" - f" type {self.type!r}, but it was {nargs}." - ) - - if required and deprecated: - raise ValueError( - f"The {self.param_type_name} '{self.human_readable_name}' " - "is deprecated and still required. A deprecated " - f"{self.param_type_name} cannot be required." - ) - - def to_info_dict(self) -> dict[str, t.Any]: - """Gather information that could be useful for a tool generating - user-facing documentation. - - Use :meth:`click.Context.to_info_dict` to traverse the entire - CLI structure. - - .. versionchanged:: 8.3.0 - Returns ``None`` for the :attr:`default` if it was not set. - - .. versionadded:: 8.0 - """ - return { - "name": self.name, - "param_type_name": self.param_type_name, - "opts": self.opts, - "secondary_opts": self.secondary_opts, - "type": self.type.to_info_dict(), - "required": self.required, - "nargs": self.nargs, - "multiple": self.multiple, - # We explicitly hide the :attr:`UNSET` value to the user, as we choose to - # make it an implementation detail. And because ``to_info_dict`` has been - # designed for documentation purposes, we return ``None`` instead. - "default": self.default if self.default is not UNSET else None, - "envvar": self.envvar, - } - - def __repr__(self) -> str: - return f"<{self.__class__.__name__} {self.name}>" - - def _parse_decls( - self, decls: cabc.Sequence[str], expose_value: bool - ) -> tuple[str | None, list[str], list[str]]: - raise NotImplementedError() - - @property - def human_readable_name(self) -> str: - """Returns the human readable name of this parameter. This is the - same as the name for options, but the metavar for arguments. - """ - return self.name # type: ignore - - def make_metavar(self, ctx: Context) -> str: - if self.metavar is not None: - return self.metavar - - metavar = self.type.get_metavar(param=self, ctx=ctx) - - if metavar is None: - metavar = self.type.name.upper() - - if self.nargs != 1: - metavar += "..." - - return metavar - - @t.overload - def get_default( - self, ctx: Context, call: t.Literal[True] = True - ) -> t.Any | None: ... - - @t.overload - def get_default( - self, ctx: Context, call: bool = ... - ) -> t.Any | t.Callable[[], t.Any] | None: ... - - def get_default( - self, ctx: Context, call: bool = True - ) -> t.Any | t.Callable[[], t.Any] | None: - """Get the default for the parameter. Tries - :meth:`Context.lookup_default` first, then the local default. - - :param ctx: Current context. - :param call: If the default is a callable, call it. Disable to - return the callable instead. - - .. versionchanged:: 8.0.2 - Type casting is no longer performed when getting a default. - - .. versionchanged:: 8.0.1 - Type casting can fail in resilient parsing mode. Invalid - defaults will not prevent showing help text. - - .. versionchanged:: 8.0 - Looks at ``ctx.default_map`` first. - - .. versionchanged:: 8.0 - Added the ``call`` parameter. - """ - value = ctx.lookup_default(self.name, call=False) # type: ignore - - if value is UNSET: - value = self.default - - if call and callable(value): - value = value() - - return value - - def add_to_parser(self, parser: _OptionParser, ctx: Context) -> None: - raise NotImplementedError() - - def consume_value( - self, ctx: Context, opts: cabc.Mapping[str, t.Any] - ) -> tuple[t.Any, ParameterSource]: - """Returns the parameter value produced by the parser. - - If the parser did not produce a value from user input, the value is either - sourced from the environment variable, the default map, or the parameter's - default value. In that order of precedence. - - If no value is found, an internal sentinel value is returned. - - :meta private: - """ - # Collect from the parse the value passed by the user to the CLI. - value = opts.get(self.name, UNSET) # type: ignore - # If the value is set, it means it was sourced from the command line by the - # parser, otherwise it left unset by default. - source = ( - ParameterSource.COMMANDLINE - if value is not UNSET - else ParameterSource.DEFAULT - ) - - if value is UNSET: - envvar_value = self.value_from_envvar(ctx) - if envvar_value is not None: - value = envvar_value - source = ParameterSource.ENVIRONMENT - - if value is UNSET: - default_map_value = ctx.lookup_default(self.name) # type: ignore - if default_map_value is not UNSET: - value = default_map_value - source = ParameterSource.DEFAULT_MAP - - if value is UNSET: - default_value = self.get_default(ctx) - if default_value is not UNSET: - value = default_value - source = ParameterSource.DEFAULT - - return value, source - - def type_cast_value(self, ctx: Context, value: t.Any) -> t.Any: - """Convert and validate a value against the parameter's - :attr:`type`, :attr:`multiple`, and :attr:`nargs`. - """ - if value is None: - if self.multiple or self.nargs == -1: - return () - else: - return value - - def check_iter(value: t.Any) -> cabc.Iterator[t.Any]: - try: - return _check_iter(value) - except TypeError: - # This should only happen when passing in args manually, - # the parser should construct an iterable when parsing - # the command line. - raise BadParameter( - _("Value must be an iterable."), ctx=ctx, param=self - ) from None - - # Define the conversion function based on nargs and type. - - if self.nargs == 1 or self.type.is_composite: - - def convert(value: t.Any) -> t.Any: - return self.type(value, param=self, ctx=ctx) - - elif self.nargs == -1: - - def convert(value: t.Any) -> t.Any: # tuple[t.Any, ...] - return tuple(self.type(x, self, ctx) for x in check_iter(value)) - - else: # nargs > 1 - - def convert(value: t.Any) -> t.Any: # tuple[t.Any, ...] - value = tuple(check_iter(value)) - - if len(value) != self.nargs: - raise BadParameter( - ngettext( - "Takes {nargs} values but 1 was given.", - "Takes {nargs} values but {len} were given.", - len(value), - ).format(nargs=self.nargs, len=len(value)), - ctx=ctx, - param=self, - ) - - return tuple(self.type(x, self, ctx) for x in value) - - if self.multiple: - return tuple(convert(x) for x in check_iter(value)) - - return convert(value) - - def value_is_missing(self, value: t.Any) -> bool: - """A value is considered missing if: - - - it is :attr:`UNSET`, - - or if it is an empty sequence while the parameter is suppose to have - non-single value (i.e. :attr:`nargs` is not ``1`` or :attr:`multiple` is - set). - - :meta private: - """ - if value is UNSET: - return True - - if (self.nargs != 1 or self.multiple) and value == (): - return True - - return False - - def process_value(self, ctx: Context, value: t.Any) -> t.Any: - """Process the value of this parameter: - - 1. Type cast the value using :meth:`type_cast_value`. - 2. Check if the value is missing (see: :meth:`value_is_missing`), and raise - :exc:`MissingParameter` if it is required. - 3. If a :attr:`callback` is set, call it to have the value replaced by the - result of the callback. If the value was not set, the callback receive - ``None``. This keep the legacy behavior as it was before the introduction of - the :attr:`UNSET` sentinel. - - :meta private: - """ - # shelter `type_cast_value` from ever seeing an `UNSET` value by handling the - # cases in which `UNSET` gets special treatment explicitly at this layer - # - # Refs: - # https://github.com/pallets/click/issues/3069 - if value is UNSET: - if self.multiple or self.nargs == -1: - value = () - else: - value = self.type_cast_value(ctx, value) - - if self.required and self.value_is_missing(value): - raise MissingParameter(ctx=ctx, param=self) - - if self.callback is not None: - # Legacy case: UNSET is not exposed directly to the callback, but converted - # to None. - if value is UNSET: - value = None - - # Search for parameters with UNSET values in the context. - unset_keys = {k: None for k, v in ctx.params.items() if v is UNSET} - # No UNSET values, call the callback as usual. - if not unset_keys: - value = self.callback(ctx, self, value) - - # Legacy case: provide a temporarily manipulated context to the callback - # to hide UNSET values as None. - # - # Refs: - # https://github.com/pallets/click/issues/3136 - # https://github.com/pallets/click/pull/3137 - else: - # Add another layer to the context stack to clearly hint that the - # context is temporarily modified. - with ctx: - # Update the context parameters to replace UNSET with None. - ctx.params.update(unset_keys) - # Feed these fake context parameters to the callback. - value = self.callback(ctx, self, value) - # Restore the UNSET values in the context parameters. - ctx.params.update( - { - k: UNSET - for k in unset_keys - # Only restore keys that are present and still None, in case - # the callback modified other parameters. - if k in ctx.params and ctx.params[k] is None - } - ) - - return value - - def resolve_envvar_value(self, ctx: Context) -> str | None: - """Returns the value found in the environment variable(s) attached to this - parameter. - - Environment variables values are `always returned as strings - `_. - - This method returns ``None`` if: - - - the :attr:`envvar` property is not set on the :class:`Parameter`, - - the environment variable is not found in the environment, - - the variable is found in the environment but its value is empty (i.e. the - environment variable is present but has an empty string). - - If :attr:`envvar` is setup with multiple environment variables, - then only the first non-empty value is returned. - - .. caution:: - - The raw value extracted from the environment is not normalized and is - returned as-is. Any normalization or reconciliation is performed later by - the :class:`Parameter`'s :attr:`type`. - - :meta private: - """ - if not self.envvar: - return None - - if isinstance(self.envvar, str): - rv = os.environ.get(self.envvar) - - if rv: - return rv - else: - for envvar in self.envvar: - rv = os.environ.get(envvar) - - # Return the first non-empty value of the list of environment variables. - if rv: - return rv - # Else, absence of value is interpreted as an environment variable that - # is not set, so proceed to the next one. - - return None - - def value_from_envvar(self, ctx: Context) -> str | cabc.Sequence[str] | None: - """Process the raw environment variable string for this parameter. - - Returns the string as-is or splits it into a sequence of strings if the - parameter is expecting multiple values (i.e. its :attr:`nargs` property is set - to a value other than ``1``). - - :meta private: - """ - rv = self.resolve_envvar_value(ctx) - - if rv is not None and self.nargs != 1: - return self.type.split_envvar_value(rv) - - return rv - - def handle_parse_result( - self, ctx: Context, opts: cabc.Mapping[str, t.Any], args: list[str] - ) -> tuple[t.Any, list[str]]: - """Process the value produced by the parser from user input. - - Always process the value through the Parameter's :attr:`type`, wherever it - comes from. - - If the parameter is deprecated, this method warn the user about it. But only if - the value has been explicitly set by the user (and as such, is not coming from - a default). - - :meta private: - """ - with augment_usage_errors(ctx, param=self): - value, source = self.consume_value(ctx, opts) - - ctx.set_parameter_source(self.name, source) # type: ignore - - # Display a deprecation warning if necessary. - if ( - self.deprecated - and value is not UNSET - and source not in (ParameterSource.DEFAULT, ParameterSource.DEFAULT_MAP) - ): - extra_message = ( - f" {self.deprecated}" if isinstance(self.deprecated, str) else "" - ) - message = _( - "DeprecationWarning: The {param_type} {name!r} is deprecated." - "{extra_message}" - ).format( - param_type=self.param_type_name, - name=self.human_readable_name, - extra_message=extra_message, - ) - echo(style(message, fg="red"), err=True) - - # Process the value through the parameter's type. - try: - value = self.process_value(ctx, value) - except Exception: - if not ctx.resilient_parsing: - raise - # In resilient parsing mode, we do not want to fail the command if the - # value is incompatible with the parameter type, so we reset the value - # to UNSET, which will be interpreted as a missing value. - value = UNSET - - # Add parameter's value to the context. - if ( - self.expose_value - # We skip adding the value if it was previously set by another parameter - # targeting the same variable name. This prevents parameters competing for - # the same name to override each other. - and (self.name not in ctx.params or ctx.params[self.name] is UNSET) - ): - # Click is logically enforcing that the name is None if the parameter is - # not to be exposed. We still assert it here to please the type checker. - assert self.name is not None, ( - f"{self!r} parameter's name should not be None when exposing value." - ) - ctx.params[self.name] = value - - return value, args - - def get_help_record(self, ctx: Context) -> tuple[str, str] | None: - pass - - def get_usage_pieces(self, ctx: Context) -> list[str]: - return [] - - def get_error_hint(self, ctx: Context) -> str: - """Get a stringified version of the param for use in error messages to - indicate which param caused the error. - """ - hint_list = self.opts or [self.human_readable_name] - return " / ".join(f"'{x}'" for x in hint_list) - - def shell_complete(self, ctx: Context, incomplete: str) -> list[CompletionItem]: - """Return a list of completions for the incomplete value. If a - ``shell_complete`` function was given during init, it is used. - Otherwise, the :attr:`type` - :meth:`~click.types.ParamType.shell_complete` function is used. - - :param ctx: Invocation context for this command. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - if self._custom_shell_complete is not None: - results = self._custom_shell_complete(ctx, self, incomplete) - - if results and isinstance(results[0], str): - from click.shell_completion import CompletionItem - - results = [CompletionItem(c) for c in results] - - return t.cast("list[CompletionItem]", results) - - return self.type.shell_complete(ctx, self, incomplete) - - -class Option(Parameter): - """Options are usually optional values on the command line and - have some extra features that arguments don't have. - - All other parameters are passed onwards to the parameter constructor. - - :param show_default: Show the default value for this option in its - help text. Values are not shown by default, unless - :attr:`Context.show_default` is ``True``. If this value is a - string, it shows that string in parentheses instead of the - actual value. This is particularly useful for dynamic options. - For single option boolean flags, the default remains hidden if - its value is ``False``. - :param show_envvar: Controls if an environment variable should be - shown on the help page and error messages. - Normally, environment variables are not shown. - :param prompt: If set to ``True`` or a non empty string then the - user will be prompted for input. If set to ``True`` the prompt - will be the option name capitalized. A deprecated option cannot be - prompted. - :param confirmation_prompt: Prompt a second time to confirm the - value if it was prompted for. Can be set to a string instead of - ``True`` to customize the message. - :param prompt_required: If set to ``False``, the user will be - prompted for input only when the option was specified as a flag - without a value. - :param hide_input: If this is ``True`` then the input on the prompt - will be hidden from the user. This is useful for password input. - :param is_flag: forces this option to act as a flag. The default is - auto detection. - :param flag_value: which value should be used for this flag if it's - enabled. This is set to a boolean automatically if - the option string contains a slash to mark two options. - :param multiple: if this is set to `True` then the argument is accepted - multiple times and recorded. This is similar to ``nargs`` - in how it works but supports arbitrary number of - arguments. - :param count: this flag makes an option increment an integer. - :param allow_from_autoenv: if this is enabled then the value of this - parameter will be pulled from an environment - variable in case a prefix is defined on the - context. - :param help: the help string. - :param hidden: hide this option from help outputs. - :param attrs: Other command arguments described in :class:`Parameter`. - - .. versionchanged:: 8.2 - ``envvar`` used with ``flag_value`` will always use the ``flag_value``, - previously it would use the value of the environment variable. - - .. versionchanged:: 8.1 - Help text indentation is cleaned here instead of only in the - ``@option`` decorator. - - .. versionchanged:: 8.1 - The ``show_default`` parameter overrides - ``Context.show_default``. - - .. versionchanged:: 8.1 - The default of a single option boolean flag is not shown if the - default value is ``False``. - - .. versionchanged:: 8.0.1 - ``type`` is detected from ``flag_value`` if given. - """ - - param_type_name = "option" - - def __init__( - self, - param_decls: cabc.Sequence[str] | None = None, - show_default: bool | str | None = None, - prompt: bool | str = False, - confirmation_prompt: bool | str = False, - prompt_required: bool = True, - hide_input: bool = False, - is_flag: bool | None = None, - flag_value: t.Any = UNSET, - multiple: bool = False, - count: bool = False, - allow_from_autoenv: bool = True, - type: types.ParamType | t.Any | None = None, - help: str | None = None, - hidden: bool = False, - show_choices: bool = True, - show_envvar: bool = False, - deprecated: bool | str = False, - **attrs: t.Any, - ) -> None: - if help: - help = inspect.cleandoc(help) - - super().__init__( - param_decls, type=type, multiple=multiple, deprecated=deprecated, **attrs - ) - - if prompt is True: - if self.name is None: - raise TypeError("'name' is required with 'prompt=True'.") - - prompt_text: str | None = self.name.replace("_", " ").capitalize() - elif prompt is False: - prompt_text = None - else: - prompt_text = prompt - - if deprecated: - deprecated_message = ( - f"(DEPRECATED: {deprecated})" - if isinstance(deprecated, str) - else "(DEPRECATED)" - ) - help = help + deprecated_message if help is not None else deprecated_message - - self.prompt = prompt_text - self.confirmation_prompt = confirmation_prompt - self.prompt_required = prompt_required - self.hide_input = hide_input - self.hidden = hidden - - # The _flag_needs_value property tells the parser that this option is a flag - # that cannot be used standalone and needs a value. With this information, the - # parser can determine whether to consider the next user-provided argument in - # the CLI as a value for this flag or as a new option. - # If prompt is enabled but not required, then it opens the possibility for the - # option to gets its value from the user. - self._flag_needs_value = self.prompt is not None and not self.prompt_required - - # Auto-detect if this is a flag or not. - if is_flag is None: - # Implicitly a flag because flag_value was set. - if flag_value is not UNSET: - is_flag = True - # Not a flag, but when used as a flag it shows a prompt. - elif self._flag_needs_value: - is_flag = False - # Implicitly a flag because secondary options names were given. - elif self.secondary_opts: - is_flag = True - # The option is explicitly not a flag. But we do not know yet if it needs a - # value or not. So we look at the default value to determine it. - elif is_flag is False and not self._flag_needs_value: - self._flag_needs_value = self.default is UNSET - - if is_flag: - # Set missing default for flags if not explicitly required or prompted. - if self.default is UNSET and not self.required and not self.prompt: - if multiple: - self.default = () - - # Auto-detect the type of the flag based on the flag_value. - if type is None: - # A flag without a flag_value is a boolean flag. - if flag_value is UNSET: - self.type: types.ParamType = types.BoolParamType() - # If the flag value is a boolean, use BoolParamType. - elif isinstance(flag_value, bool): - self.type = types.BoolParamType() - # Otherwise, guess the type from the flag value. - else: - self.type = types.convert_type(None, flag_value) - - self.is_flag: bool = bool(is_flag) - self.is_bool_flag: bool = bool( - is_flag and isinstance(self.type, types.BoolParamType) - ) - self.flag_value: t.Any = flag_value - - # Set boolean flag default to False if unset and not required. - if self.is_bool_flag: - if self.default is UNSET and not self.required: - self.default = False - - # Support the special case of aligning the default value with the flag_value - # for flags whose default is explicitly set to True. Note that as long as we - # have this condition, there is no way a flag can have a default set to True, - # and a flag_value set to something else. Refs: - # https://github.com/pallets/click/issues/3024#issuecomment-3146199461 - # https://github.com/pallets/click/pull/3030/commits/06847da - if self.default is True and self.flag_value is not UNSET: - self.default = self.flag_value - - # Set the default flag_value if it is not set. - if self.flag_value is UNSET: - if self.is_flag: - self.flag_value = True - else: - self.flag_value = None - - # Counting. - self.count = count - if count: - if type is None: - self.type = types.IntRange(min=0) - if self.default is UNSET: - self.default = 0 - - self.allow_from_autoenv = allow_from_autoenv - self.help = help - self.show_default = show_default - self.show_choices = show_choices - self.show_envvar = show_envvar - - if __debug__: - if deprecated and prompt: - raise ValueError("`deprecated` options cannot use `prompt`.") - - if self.nargs == -1: - raise TypeError("nargs=-1 is not supported for options.") - - if not self.is_bool_flag and self.secondary_opts: - raise TypeError("Secondary flag is not valid for non-boolean flag.") - - if self.is_bool_flag and self.hide_input and self.prompt is not None: - raise TypeError( - "'prompt' with 'hide_input' is not valid for boolean flag." - ) - - if self.count: - if self.multiple: - raise TypeError("'count' is not valid with 'multiple'.") - - if self.is_flag: - raise TypeError("'count' is not valid with 'is_flag'.") - - def to_info_dict(self) -> dict[str, t.Any]: - """ - .. versionchanged:: 8.3.0 - Returns ``None`` for the :attr:`flag_value` if it was not set. - """ - info_dict = super().to_info_dict() - info_dict.update( - help=self.help, - prompt=self.prompt, - is_flag=self.is_flag, - # We explicitly hide the :attr:`UNSET` value to the user, as we choose to - # make it an implementation detail. And because ``to_info_dict`` has been - # designed for documentation purposes, we return ``None`` instead. - flag_value=self.flag_value if self.flag_value is not UNSET else None, - count=self.count, - hidden=self.hidden, - ) - return info_dict - - def get_error_hint(self, ctx: Context) -> str: - result = super().get_error_hint(ctx) - if self.show_envvar and self.envvar is not None: - result += f" (env var: '{self.envvar}')" - return result - - def _parse_decls( - self, decls: cabc.Sequence[str], expose_value: bool - ) -> tuple[str | None, list[str], list[str]]: - opts = [] - secondary_opts = [] - name = None - possible_names = [] - - for decl in decls: - if decl.isidentifier(): - if name is not None: - raise TypeError(f"Name '{name}' defined twice") - name = decl - else: - split_char = ";" if decl[:1] == "/" else "/" - if split_char in decl: - first, second = decl.split(split_char, 1) - first = first.rstrip() - if first: - possible_names.append(_split_opt(first)) - opts.append(first) - second = second.lstrip() - if second: - secondary_opts.append(second.lstrip()) - if first == second: - raise ValueError( - f"Boolean option {decl!r} cannot use the" - " same flag for true/false." - ) - else: - possible_names.append(_split_opt(decl)) - opts.append(decl) - - if name is None and possible_names: - possible_names.sort(key=lambda x: -len(x[0])) # group long options first - name = possible_names[0][1].replace("-", "_").lower() - if not name.isidentifier(): - name = None - - if name is None: - if not expose_value: - return None, opts, secondary_opts - raise TypeError( - f"Could not determine name for option with declarations {decls!r}" - ) - - if not opts and not secondary_opts: - raise TypeError( - f"No options defined but a name was passed ({name})." - " Did you mean to declare an argument instead? Did" - f" you mean to pass '--{name}'?" - ) - - return name, opts, secondary_opts - - def add_to_parser(self, parser: _OptionParser, ctx: Context) -> None: - if self.multiple: - action = "append" - elif self.count: - action = "count" - else: - action = "store" - - if self.is_flag: - action = f"{action}_const" - - if self.is_bool_flag and self.secondary_opts: - parser.add_option( - obj=self, opts=self.opts, dest=self.name, action=action, const=True - ) - parser.add_option( - obj=self, - opts=self.secondary_opts, - dest=self.name, - action=action, - const=False, - ) - else: - parser.add_option( - obj=self, - opts=self.opts, - dest=self.name, - action=action, - const=self.flag_value, - ) - else: - parser.add_option( - obj=self, - opts=self.opts, - dest=self.name, - action=action, - nargs=self.nargs, - ) - - def get_help_record(self, ctx: Context) -> tuple[str, str] | None: - if self.hidden: - return None - - any_prefix_is_slash = False - - def _write_opts(opts: cabc.Sequence[str]) -> str: - nonlocal any_prefix_is_slash - - rv, any_slashes = join_options(opts) - - if any_slashes: - any_prefix_is_slash = True - - if not self.is_flag and not self.count: - rv += f" {self.make_metavar(ctx=ctx)}" - - return rv - - rv = [_write_opts(self.opts)] - - if self.secondary_opts: - rv.append(_write_opts(self.secondary_opts)) - - help = self.help or "" - - extra = self.get_help_extra(ctx) - extra_items = [] - if "envvars" in extra: - extra_items.append( - _("env var: {var}").format(var=", ".join(extra["envvars"])) - ) - if "default" in extra: - extra_items.append(_("default: {default}").format(default=extra["default"])) - if "range" in extra: - extra_items.append(extra["range"]) - if "required" in extra: - extra_items.append(_(extra["required"])) - - if extra_items: - extra_str = "; ".join(extra_items) - help = f"{help} [{extra_str}]" if help else f"[{extra_str}]" - - return ("; " if any_prefix_is_slash else " / ").join(rv), help - - def get_help_extra(self, ctx: Context) -> types.OptionHelpExtra: - extra: types.OptionHelpExtra = {} - - if self.show_envvar: - envvar = self.envvar - - if envvar is None: - if ( - self.allow_from_autoenv - and ctx.auto_envvar_prefix is not None - and self.name is not None - ): - envvar = f"{ctx.auto_envvar_prefix}_{self.name.upper()}" - - if envvar is not None: - if isinstance(envvar, str): - extra["envvars"] = (envvar,) - else: - extra["envvars"] = tuple(str(d) for d in envvar) - - # Temporarily enable resilient parsing to avoid type casting - # failing for the default. Might be possible to extend this to - # help formatting in general. - resilient = ctx.resilient_parsing - ctx.resilient_parsing = True - - try: - default_value = self.get_default(ctx, call=False) - finally: - ctx.resilient_parsing = resilient - - show_default = False - show_default_is_str = False - - if self.show_default is not None: - if isinstance(self.show_default, str): - show_default_is_str = show_default = True - else: - show_default = self.show_default - elif ctx.show_default is not None: - show_default = ctx.show_default - - if show_default_is_str or ( - show_default and (default_value not in (None, UNSET)) - ): - if show_default_is_str: - default_string = f"({self.show_default})" - elif isinstance(default_value, (list, tuple)): - default_string = ", ".join(str(d) for d in default_value) - elif isinstance(default_value, enum.Enum): - default_string = default_value.name - elif inspect.isfunction(default_value): - default_string = _("(dynamic)") - elif self.is_bool_flag and self.secondary_opts: - # For boolean flags that have distinct True/False opts, - # use the opt without prefix instead of the value. - default_string = _split_opt( - (self.opts if default_value else self.secondary_opts)[0] - )[1] - elif self.is_bool_flag and not self.secondary_opts and not default_value: - default_string = "" - elif default_value == "": - default_string = '""' - else: - default_string = str(default_value) - - if default_string: - extra["default"] = default_string - - if ( - isinstance(self.type, types._NumberRangeBase) - # skip count with default range type - and not (self.count and self.type.min == 0 and self.type.max is None) - ): - range_str = self.type._describe_range() - - if range_str: - extra["range"] = range_str - - if self.required: - extra["required"] = "required" - - return extra - - def prompt_for_value(self, ctx: Context) -> t.Any: - """This is an alternative flow that can be activated in the full - value processing if a value does not exist. It will prompt the - user until a valid value exists and then returns the processed - value as result. - """ - assert self.prompt is not None - - # Calculate the default before prompting anything to lock in the value before - # attempting any user interaction. - default = self.get_default(ctx) - - # A boolean flag can use a simplified [y/n] confirmation prompt. - if self.is_bool_flag: - # If we have no boolean default, we force the user to explicitly provide - # one. - if default in (UNSET, None): - default = None - # Nothing prevent you to declare an option that is simultaneously: - # 1) auto-detected as a boolean flag, - # 2) allowed to prompt, and - # 3) still declare a non-boolean default. - # This forced casting into a boolean is necessary to align any non-boolean - # default to the prompt, which is going to be a [y/n]-style confirmation - # because the option is still a boolean flag. That way, instead of [y/n], - # we get [Y/n] or [y/N] depending on the truthy value of the default. - # Refs: https://github.com/pallets/click/pull/3030#discussion_r2289180249 - else: - default = bool(default) - return confirm(self.prompt, default) - - # If show_default is set to True/False, provide this to `prompt` as well. For - # non-bool values of `show_default`, we use `prompt`'s default behavior - prompt_kwargs: t.Any = {} - if isinstance(self.show_default, bool): - prompt_kwargs["show_default"] = self.show_default - - return prompt( - self.prompt, - # Use ``None`` to inform the prompt() function to reiterate until a valid - # value is provided by the user if we have no default. - default=None if default is UNSET else default, - type=self.type, - hide_input=self.hide_input, - show_choices=self.show_choices, - confirmation_prompt=self.confirmation_prompt, - value_proc=lambda x: self.process_value(ctx, x), - **prompt_kwargs, - ) - - def resolve_envvar_value(self, ctx: Context) -> str | None: - """:class:`Option` resolves its environment variable the same way as - :func:`Parameter.resolve_envvar_value`, but it also supports - :attr:`Context.auto_envvar_prefix`. If we could not find an environment from - the :attr:`envvar` property, we fallback on :attr:`Context.auto_envvar_prefix` - to build dynamiccaly the environment variable name using the - :python:`{ctx.auto_envvar_prefix}_{self.name.upper()}` template. - - :meta private: - """ - rv = super().resolve_envvar_value(ctx) - - if rv is not None: - return rv - - if ( - self.allow_from_autoenv - and ctx.auto_envvar_prefix is not None - and self.name is not None - ): - envvar = f"{ctx.auto_envvar_prefix}_{self.name.upper()}" - rv = os.environ.get(envvar) - - if rv: - return rv - - return None - - def value_from_envvar(self, ctx: Context) -> t.Any: - """For :class:`Option`, this method processes the raw environment variable - string the same way as :func:`Parameter.value_from_envvar` does. - - But in the case of non-boolean flags, the value is analyzed to determine if the - flag is activated or not, and returns a boolean of its activation, or the - :attr:`flag_value` if the latter is set. - - This method also takes care of repeated options (i.e. options with - :attr:`multiple` set to ``True``). - - :meta private: - """ - rv = self.resolve_envvar_value(ctx) - - # Absent environment variable or an empty string is interpreted as unset. - if rv is None: - return None - - # Non-boolean flags are more liberal in what they accept. But a flag being a - # flag, its envvar value still needs to be analyzed to determine if the flag is - # activated or not. - if self.is_flag and not self.is_bool_flag: - # If the flag_value is set and match the envvar value, return it - # directly. - if self.flag_value is not UNSET and rv == self.flag_value: - return self.flag_value - # Analyze the envvar value as a boolean to know if the flag is - # activated or not. - return types.BoolParamType.str_to_bool(rv) - - # Split the envvar value if it is allowed to be repeated. - value_depth = (self.nargs != 1) + bool(self.multiple) - if value_depth > 0: - multi_rv = self.type.split_envvar_value(rv) - if self.multiple and self.nargs != 1: - multi_rv = batch(multi_rv, self.nargs) # type: ignore[assignment] - - return multi_rv - - return rv - - def consume_value( - self, ctx: Context, opts: cabc.Mapping[str, Parameter] - ) -> tuple[t.Any, ParameterSource]: - """For :class:`Option`, the value can be collected from an interactive prompt - if the option is a flag that needs a value (and the :attr:`prompt` property is - set). - - Additionally, this method handles flag option that are activated without a - value, in which case the :attr:`flag_value` is returned. - - :meta private: - """ - value, source = super().consume_value(ctx, opts) - - # The parser will emit a sentinel value if the option is allowed to as a flag - # without a value. - if value is FLAG_NEEDS_VALUE: - # If the option allows for a prompt, we start an interaction with the user. - if self.prompt is not None and not ctx.resilient_parsing: - value = self.prompt_for_value(ctx) - source = ParameterSource.PROMPT - # Else the flag takes its flag_value as value. - else: - value = self.flag_value - source = ParameterSource.COMMANDLINE - - # A flag which is activated always returns the flag value, unless the value - # comes from the explicitly sets default. - elif ( - self.is_flag - and value is True - and not self.is_bool_flag - and source not in (ParameterSource.DEFAULT, ParameterSource.DEFAULT_MAP) - ): - value = self.flag_value - - # Re-interpret a multiple option which has been sent as-is by the parser. - # Here we replace each occurrence of value-less flags (marked by the - # FLAG_NEEDS_VALUE sentinel) with the flag_value. - elif ( - self.multiple - and value is not UNSET - and source not in (ParameterSource.DEFAULT, ParameterSource.DEFAULT_MAP) - and any(v is FLAG_NEEDS_VALUE for v in value) - ): - value = [self.flag_value if v is FLAG_NEEDS_VALUE else v for v in value] - source = ParameterSource.COMMANDLINE - - # The value wasn't set, or used the param's default, prompt for one to the user - # if prompting is enabled. - elif ( - ( - value is UNSET - or source in (ParameterSource.DEFAULT, ParameterSource.DEFAULT_MAP) - ) - and self.prompt is not None - and (self.required or self.prompt_required) - and not ctx.resilient_parsing - ): - value = self.prompt_for_value(ctx) - source = ParameterSource.PROMPT - - return value, source - - def process_value(self, ctx: Context, value: t.Any) -> t.Any: - # process_value has to be overridden on Options in order to capture - # `value == UNSET` cases before `type_cast_value()` gets called. - # - # Refs: - # https://github.com/pallets/click/issues/3069 - if self.is_flag and not self.required and self.is_bool_flag and value is UNSET: - value = False - - if self.callback is not None: - value = self.callback(ctx, self, value) - - return value - - # in the normal case, rely on Parameter.process_value - return super().process_value(ctx, value) - - -class Argument(Parameter): - """Arguments are positional parameters to a command. They generally - provide fewer features than options but can have infinite ``nargs`` - and are required by default. - - All parameters are passed onwards to the constructor of :class:`Parameter`. - """ - - param_type_name = "argument" - - def __init__( - self, - param_decls: cabc.Sequence[str], - required: bool | None = None, - **attrs: t.Any, - ) -> None: - # Auto-detect the requirement status of the argument if not explicitly set. - if required is None: - # The argument gets automatically required if it has no explicit default - # value set and is setup to match at least one value. - if attrs.get("default", UNSET) is UNSET: - required = attrs.get("nargs", 1) > 0 - # If the argument has a default value, it is not required. - else: - required = False - - if "multiple" in attrs: - raise TypeError("__init__() got an unexpected keyword argument 'multiple'.") - - super().__init__(param_decls, required=required, **attrs) - - @property - def human_readable_name(self) -> str: - if self.metavar is not None: - return self.metavar - return self.name.upper() # type: ignore - - def make_metavar(self, ctx: Context) -> str: - if self.metavar is not None: - return self.metavar - var = self.type.get_metavar(param=self, ctx=ctx) - if not var: - var = self.name.upper() # type: ignore - if self.deprecated: - var += "!" - if not self.required: - var = f"[{var}]" - if self.nargs != 1: - var += "..." - return var - - def _parse_decls( - self, decls: cabc.Sequence[str], expose_value: bool - ) -> tuple[str | None, list[str], list[str]]: - if not decls: - if not expose_value: - return None, [], [] - raise TypeError("Argument is marked as exposed, but does not have a name.") - if len(decls) == 1: - name = arg = decls[0] - name = name.replace("-", "_").lower() - else: - raise TypeError( - "Arguments take exactly one parameter declaration, got" - f" {len(decls)}: {decls}." - ) - return name, [arg], [] - - def get_usage_pieces(self, ctx: Context) -> list[str]: - return [self.make_metavar(ctx)] - - def get_error_hint(self, ctx: Context) -> str: - return f"'{self.make_metavar(ctx)}'" - - def add_to_parser(self, parser: _OptionParser, ctx: Context) -> None: - parser.add_argument(dest=self.name, nargs=self.nargs, obj=self) - - -def __getattr__(name: str) -> object: - import warnings - - if name == "BaseCommand": - warnings.warn( - "'BaseCommand' is deprecated and will be removed in Click 9.0. Use" - " 'Command' instead.", - DeprecationWarning, - stacklevel=2, - ) - return _BaseCommand - - if name == "MultiCommand": - warnings.warn( - "'MultiCommand' is deprecated and will be removed in Click 9.0. Use" - " 'Group' instead.", - DeprecationWarning, - stacklevel=2, - ) - return _MultiCommand - - raise AttributeError(name) diff --git a/.env/Lib/site-packages/click/decorators.py b/.env/Lib/site-packages/click/decorators.py deleted file mode 100644 index 21f4c34..0000000 --- a/.env/Lib/site-packages/click/decorators.py +++ /dev/null @@ -1,551 +0,0 @@ -from __future__ import annotations - -import inspect -import typing as t -from functools import update_wrapper -from gettext import gettext as _ - -from .core import Argument -from .core import Command -from .core import Context -from .core import Group -from .core import Option -from .core import Parameter -from .globals import get_current_context -from .utils import echo - -if t.TYPE_CHECKING: - import typing_extensions as te - - P = te.ParamSpec("P") - -R = t.TypeVar("R") -T = t.TypeVar("T") -_AnyCallable = t.Callable[..., t.Any] -FC = t.TypeVar("FC", bound="_AnyCallable | Command") - - -def pass_context(f: t.Callable[te.Concatenate[Context, P], R]) -> t.Callable[P, R]: - """Marks a callback as wanting to receive the current context - object as first argument. - """ - - def new_func(*args: P.args, **kwargs: P.kwargs) -> R: - return f(get_current_context(), *args, **kwargs) - - return update_wrapper(new_func, f) - - -def pass_obj(f: t.Callable[te.Concatenate[T, P], R]) -> t.Callable[P, R]: - """Similar to :func:`pass_context`, but only pass the object on the - context onwards (:attr:`Context.obj`). This is useful if that object - represents the state of a nested system. - """ - - def new_func(*args: P.args, **kwargs: P.kwargs) -> R: - return f(get_current_context().obj, *args, **kwargs) - - return update_wrapper(new_func, f) - - -def make_pass_decorator( - object_type: type[T], ensure: bool = False -) -> t.Callable[[t.Callable[te.Concatenate[T, P], R]], t.Callable[P, R]]: - """Given an object type this creates a decorator that will work - similar to :func:`pass_obj` but instead of passing the object of the - current context, it will find the innermost context of type - :func:`object_type`. - - This generates a decorator that works roughly like this:: - - from functools import update_wrapper - - def decorator(f): - @pass_context - def new_func(ctx, *args, **kwargs): - obj = ctx.find_object(object_type) - return ctx.invoke(f, obj, *args, **kwargs) - return update_wrapper(new_func, f) - return decorator - - :param object_type: the type of the object to pass. - :param ensure: if set to `True`, a new object will be created and - remembered on the context if it's not there yet. - """ - - def decorator(f: t.Callable[te.Concatenate[T, P], R]) -> t.Callable[P, R]: - def new_func(*args: P.args, **kwargs: P.kwargs) -> R: - ctx = get_current_context() - - obj: T | None - if ensure: - obj = ctx.ensure_object(object_type) - else: - obj = ctx.find_object(object_type) - - if obj is None: - raise RuntimeError( - "Managed to invoke callback without a context" - f" object of type {object_type.__name__!r}" - " existing." - ) - - return ctx.invoke(f, obj, *args, **kwargs) - - return update_wrapper(new_func, f) - - return decorator - - -def pass_meta_key( - key: str, *, doc_description: str | None = None -) -> t.Callable[[t.Callable[te.Concatenate[T, P], R]], t.Callable[P, R]]: - """Create a decorator that passes a key from - :attr:`click.Context.meta` as the first argument to the decorated - function. - - :param key: Key in ``Context.meta`` to pass. - :param doc_description: Description of the object being passed, - inserted into the decorator's docstring. Defaults to "the 'key' - key from Context.meta". - - .. versionadded:: 8.0 - """ - - def decorator(f: t.Callable[te.Concatenate[T, P], R]) -> t.Callable[P, R]: - def new_func(*args: P.args, **kwargs: P.kwargs) -> R: - ctx = get_current_context() - obj = ctx.meta[key] - return ctx.invoke(f, obj, *args, **kwargs) - - return update_wrapper(new_func, f) - - if doc_description is None: - doc_description = f"the {key!r} key from :attr:`click.Context.meta`" - - decorator.__doc__ = ( - f"Decorator that passes {doc_description} as the first argument" - " to the decorated function." - ) - return decorator - - -CmdType = t.TypeVar("CmdType", bound=Command) - - -# variant: no call, directly as decorator for a function. -@t.overload -def command(name: _AnyCallable) -> Command: ... - - -# variant: with positional name and with positional or keyword cls argument: -# @command(namearg, CommandCls, ...) or @command(namearg, cls=CommandCls, ...) -@t.overload -def command( - name: str | None, - cls: type[CmdType], - **attrs: t.Any, -) -> t.Callable[[_AnyCallable], CmdType]: ... - - -# variant: name omitted, cls _must_ be a keyword argument, @command(cls=CommandCls, ...) -@t.overload -def command( - name: None = None, - *, - cls: type[CmdType], - **attrs: t.Any, -) -> t.Callable[[_AnyCallable], CmdType]: ... - - -# variant: with optional string name, no cls argument provided. -@t.overload -def command( - name: str | None = ..., cls: None = None, **attrs: t.Any -) -> t.Callable[[_AnyCallable], Command]: ... - - -def command( - name: str | _AnyCallable | None = None, - cls: type[CmdType] | None = None, - **attrs: t.Any, -) -> Command | t.Callable[[_AnyCallable], Command | CmdType]: - r"""Creates a new :class:`Command` and uses the decorated function as - callback. This will also automatically attach all decorated - :func:`option`\s and :func:`argument`\s as parameters to the command. - - The name of the command defaults to the name of the function, converted to - lowercase, with underscores ``_`` replaced by dashes ``-``, and the suffixes - ``_command``, ``_cmd``, ``_group``, and ``_grp`` are removed. For example, - ``init_data_command`` becomes ``init-data``. - - All keyword arguments are forwarded to the underlying command class. - For the ``params`` argument, any decorated params are appended to - the end of the list. - - Once decorated the function turns into a :class:`Command` instance - that can be invoked as a command line utility or be attached to a - command :class:`Group`. - - :param name: The name of the command. Defaults to modifying the function's - name as described above. - :param cls: The command class to create. Defaults to :class:`Command`. - - .. versionchanged:: 8.2 - The suffixes ``_command``, ``_cmd``, ``_group``, and ``_grp`` are - removed when generating the name. - - .. versionchanged:: 8.1 - This decorator can be applied without parentheses. - - .. versionchanged:: 8.1 - The ``params`` argument can be used. Decorated params are - appended to the end of the list. - """ - - func: t.Callable[[_AnyCallable], t.Any] | None = None - - if callable(name): - func = name - name = None - assert cls is None, "Use 'command(cls=cls)(callable)' to specify a class." - assert not attrs, "Use 'command(**kwargs)(callable)' to provide arguments." - - if cls is None: - cls = t.cast("type[CmdType]", Command) - - def decorator(f: _AnyCallable) -> CmdType: - if isinstance(f, Command): - raise TypeError("Attempted to convert a callback into a command twice.") - - attr_params = attrs.pop("params", None) - params = attr_params if attr_params is not None else [] - - try: - decorator_params = f.__click_params__ # type: ignore - except AttributeError: - pass - else: - del f.__click_params__ # type: ignore - params.extend(reversed(decorator_params)) - - if attrs.get("help") is None: - attrs["help"] = f.__doc__ - - if t.TYPE_CHECKING: - assert cls is not None - assert not callable(name) - - if name is not None: - cmd_name = name - else: - cmd_name = f.__name__.lower().replace("_", "-") - cmd_left, sep, suffix = cmd_name.rpartition("-") - - if sep and suffix in {"command", "cmd", "group", "grp"}: - cmd_name = cmd_left - - cmd = cls(name=cmd_name, callback=f, params=params, **attrs) - cmd.__doc__ = f.__doc__ - return cmd - - if func is not None: - return decorator(func) - - return decorator - - -GrpType = t.TypeVar("GrpType", bound=Group) - - -# variant: no call, directly as decorator for a function. -@t.overload -def group(name: _AnyCallable) -> Group: ... - - -# variant: with positional name and with positional or keyword cls argument: -# @group(namearg, GroupCls, ...) or @group(namearg, cls=GroupCls, ...) -@t.overload -def group( - name: str | None, - cls: type[GrpType], - **attrs: t.Any, -) -> t.Callable[[_AnyCallable], GrpType]: ... - - -# variant: name omitted, cls _must_ be a keyword argument, @group(cmd=GroupCls, ...) -@t.overload -def group( - name: None = None, - *, - cls: type[GrpType], - **attrs: t.Any, -) -> t.Callable[[_AnyCallable], GrpType]: ... - - -# variant: with optional string name, no cls argument provided. -@t.overload -def group( - name: str | None = ..., cls: None = None, **attrs: t.Any -) -> t.Callable[[_AnyCallable], Group]: ... - - -def group( - name: str | _AnyCallable | None = None, - cls: type[GrpType] | None = None, - **attrs: t.Any, -) -> Group | t.Callable[[_AnyCallable], Group | GrpType]: - """Creates a new :class:`Group` with a function as callback. This - works otherwise the same as :func:`command` just that the `cls` - parameter is set to :class:`Group`. - - .. versionchanged:: 8.1 - This decorator can be applied without parentheses. - """ - if cls is None: - cls = t.cast("type[GrpType]", Group) - - if callable(name): - return command(cls=cls, **attrs)(name) - - return command(name, cls, **attrs) - - -def _param_memo(f: t.Callable[..., t.Any], param: Parameter) -> None: - if isinstance(f, Command): - f.params.append(param) - else: - if not hasattr(f, "__click_params__"): - f.__click_params__ = [] # type: ignore - - f.__click_params__.append(param) # type: ignore - - -def argument( - *param_decls: str, cls: type[Argument] | None = None, **attrs: t.Any -) -> t.Callable[[FC], FC]: - """Attaches an argument to the command. All positional arguments are - passed as parameter declarations to :class:`Argument`; all keyword - arguments are forwarded unchanged (except ``cls``). - This is equivalent to creating an :class:`Argument` instance manually - and attaching it to the :attr:`Command.params` list. - - For the default argument class, refer to :class:`Argument` and - :class:`Parameter` for descriptions of parameters. - - :param cls: the argument class to instantiate. This defaults to - :class:`Argument`. - :param param_decls: Passed as positional arguments to the constructor of - ``cls``. - :param attrs: Passed as keyword arguments to the constructor of ``cls``. - """ - if cls is None: - cls = Argument - - def decorator(f: FC) -> FC: - _param_memo(f, cls(param_decls, **attrs)) - return f - - return decorator - - -def option( - *param_decls: str, cls: type[Option] | None = None, **attrs: t.Any -) -> t.Callable[[FC], FC]: - """Attaches an option to the command. All positional arguments are - passed as parameter declarations to :class:`Option`; all keyword - arguments are forwarded unchanged (except ``cls``). - This is equivalent to creating an :class:`Option` instance manually - and attaching it to the :attr:`Command.params` list. - - For the default option class, refer to :class:`Option` and - :class:`Parameter` for descriptions of parameters. - - :param cls: the option class to instantiate. This defaults to - :class:`Option`. - :param param_decls: Passed as positional arguments to the constructor of - ``cls``. - :param attrs: Passed as keyword arguments to the constructor of ``cls``. - """ - if cls is None: - cls = Option - - def decorator(f: FC) -> FC: - _param_memo(f, cls(param_decls, **attrs)) - return f - - return decorator - - -def confirmation_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]: - """Add a ``--yes`` option which shows a prompt before continuing if - not passed. If the prompt is declined, the program will exit. - - :param param_decls: One or more option names. Defaults to the single - value ``"--yes"``. - :param kwargs: Extra arguments are passed to :func:`option`. - """ - - def callback(ctx: Context, param: Parameter, value: bool) -> None: - if not value: - ctx.abort() - - if not param_decls: - param_decls = ("--yes",) - - kwargs.setdefault("is_flag", True) - kwargs.setdefault("callback", callback) - kwargs.setdefault("expose_value", False) - kwargs.setdefault("prompt", "Do you want to continue?") - kwargs.setdefault("help", "Confirm the action without prompting.") - return option(*param_decls, **kwargs) - - -def password_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]: - """Add a ``--password`` option which prompts for a password, hiding - input and asking to enter the value again for confirmation. - - :param param_decls: One or more option names. Defaults to the single - value ``"--password"``. - :param kwargs: Extra arguments are passed to :func:`option`. - """ - if not param_decls: - param_decls = ("--password",) - - kwargs.setdefault("prompt", True) - kwargs.setdefault("confirmation_prompt", True) - kwargs.setdefault("hide_input", True) - return option(*param_decls, **kwargs) - - -def version_option( - version: str | None = None, - *param_decls: str, - package_name: str | None = None, - prog_name: str | None = None, - message: str | None = None, - **kwargs: t.Any, -) -> t.Callable[[FC], FC]: - """Add a ``--version`` option which immediately prints the version - number and exits the program. - - If ``version`` is not provided, Click will try to detect it using - :func:`importlib.metadata.version` to get the version for the - ``package_name``. - - If ``package_name`` is not provided, Click will try to detect it by - inspecting the stack frames. This will be used to detect the - version, so it must match the name of the installed package. - - :param version: The version number to show. If not provided, Click - will try to detect it. - :param param_decls: One or more option names. Defaults to the single - value ``"--version"``. - :param package_name: The package name to detect the version from. If - not provided, Click will try to detect it. - :param prog_name: The name of the CLI to show in the message. If not - provided, it will be detected from the command. - :param message: The message to show. The values ``%(prog)s``, - ``%(package)s``, and ``%(version)s`` are available. Defaults to - ``"%(prog)s, version %(version)s"``. - :param kwargs: Extra arguments are passed to :func:`option`. - :raise RuntimeError: ``version`` could not be detected. - - .. versionchanged:: 8.0 - Add the ``package_name`` parameter, and the ``%(package)s`` - value for messages. - - .. versionchanged:: 8.0 - Use :mod:`importlib.metadata` instead of ``pkg_resources``. The - version is detected based on the package name, not the entry - point name. The Python package name must match the installed - package name, or be passed with ``package_name=``. - """ - if message is None: - message = _("%(prog)s, version %(version)s") - - if version is None and package_name is None: - frame = inspect.currentframe() - f_back = frame.f_back if frame is not None else None - f_globals = f_back.f_globals if f_back is not None else None - # break reference cycle - # https://docs.python.org/3/library/inspect.html#the-interpreter-stack - del frame - - if f_globals is not None: - package_name = f_globals.get("__name__") - - if package_name == "__main__": - package_name = f_globals.get("__package__") - - if package_name: - package_name = package_name.partition(".")[0] - - def callback(ctx: Context, param: Parameter, value: bool) -> None: - if not value or ctx.resilient_parsing: - return - - nonlocal prog_name - nonlocal version - - if prog_name is None: - prog_name = ctx.find_root().info_name - - if version is None and package_name is not None: - import importlib.metadata - - try: - version = importlib.metadata.version(package_name) - except importlib.metadata.PackageNotFoundError: - raise RuntimeError( - f"{package_name!r} is not installed. Try passing" - " 'package_name' instead." - ) from None - - if version is None: - raise RuntimeError( - f"Could not determine the version for {package_name!r} automatically." - ) - - echo( - message % {"prog": prog_name, "package": package_name, "version": version}, - color=ctx.color, - ) - ctx.exit() - - if not param_decls: - param_decls = ("--version",) - - kwargs.setdefault("is_flag", True) - kwargs.setdefault("expose_value", False) - kwargs.setdefault("is_eager", True) - kwargs.setdefault("help", _("Show the version and exit.")) - kwargs["callback"] = callback - return option(*param_decls, **kwargs) - - -def help_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]: - """Pre-configured ``--help`` option which immediately prints the help page - and exits the program. - - :param param_decls: One or more option names. Defaults to the single - value ``"--help"``. - :param kwargs: Extra arguments are passed to :func:`option`. - """ - - def show_help(ctx: Context, param: Parameter, value: bool) -> None: - """Callback that print the help page on ```` and exits.""" - if value and not ctx.resilient_parsing: - echo(ctx.get_help(), color=ctx.color) - ctx.exit() - - if not param_decls: - param_decls = ("--help",) - - kwargs.setdefault("is_flag", True) - kwargs.setdefault("expose_value", False) - kwargs.setdefault("is_eager", True) - kwargs.setdefault("help", _("Show this message and exit.")) - kwargs.setdefault("callback", show_help) - - return option(*param_decls, **kwargs) diff --git a/.env/Lib/site-packages/click/exceptions.py b/.env/Lib/site-packages/click/exceptions.py deleted file mode 100644 index 4d782ee..0000000 --- a/.env/Lib/site-packages/click/exceptions.py +++ /dev/null @@ -1,308 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import typing as t -from gettext import gettext as _ -from gettext import ngettext - -from ._compat import get_text_stderr -from .globals import resolve_color_default -from .utils import echo -from .utils import format_filename - -if t.TYPE_CHECKING: - from .core import Command - from .core import Context - from .core import Parameter - - -def _join_param_hints(param_hint: cabc.Sequence[str] | str | None) -> str | None: - if param_hint is not None and not isinstance(param_hint, str): - return " / ".join(repr(x) for x in param_hint) - - return param_hint - - -class ClickException(Exception): - """An exception that Click can handle and show to the user.""" - - #: The exit code for this exception. - exit_code = 1 - - def __init__(self, message: str) -> None: - super().__init__(message) - # The context will be removed by the time we print the message, so cache - # the color settings here to be used later on (in `show`) - self.show_color: bool | None = resolve_color_default() - self.message = message - - def format_message(self) -> str: - return self.message - - def __str__(self) -> str: - return self.message - - def show(self, file: t.IO[t.Any] | None = None) -> None: - if file is None: - file = get_text_stderr() - - echo( - _("Error: {message}").format(message=self.format_message()), - file=file, - color=self.show_color, - ) - - -class UsageError(ClickException): - """An internal exception that signals a usage error. This typically - aborts any further handling. - - :param message: the error message to display. - :param ctx: optionally the context that caused this error. Click will - fill in the context automatically in some situations. - """ - - exit_code = 2 - - def __init__(self, message: str, ctx: Context | None = None) -> None: - super().__init__(message) - self.ctx = ctx - self.cmd: Command | None = self.ctx.command if self.ctx else None - - def show(self, file: t.IO[t.Any] | None = None) -> None: - if file is None: - file = get_text_stderr() - color = None - hint = "" - if ( - self.ctx is not None - and self.ctx.command.get_help_option(self.ctx) is not None - ): - hint = _("Try '{command} {option}' for help.").format( - command=self.ctx.command_path, option=self.ctx.help_option_names[0] - ) - hint = f"{hint}\n" - if self.ctx is not None: - color = self.ctx.color - echo(f"{self.ctx.get_usage()}\n{hint}", file=file, color=color) - echo( - _("Error: {message}").format(message=self.format_message()), - file=file, - color=color, - ) - - -class BadParameter(UsageError): - """An exception that formats out a standardized error message for a - bad parameter. This is useful when thrown from a callback or type as - Click will attach contextual information to it (for instance, which - parameter it is). - - .. versionadded:: 2.0 - - :param param: the parameter object that caused this error. This can - be left out, and Click will attach this info itself - if possible. - :param param_hint: a string that shows up as parameter name. This - can be used as alternative to `param` in cases - where custom validation should happen. If it is - a string it's used as such, if it's a list then - each item is quoted and separated. - """ - - def __init__( - self, - message: str, - ctx: Context | None = None, - param: Parameter | None = None, - param_hint: cabc.Sequence[str] | str | None = None, - ) -> None: - super().__init__(message, ctx) - self.param = param - self.param_hint = param_hint - - def format_message(self) -> str: - if self.param_hint is not None: - param_hint = self.param_hint - elif self.param is not None: - param_hint = self.param.get_error_hint(self.ctx) # type: ignore - else: - return _("Invalid value: {message}").format(message=self.message) - - return _("Invalid value for {param_hint}: {message}").format( - param_hint=_join_param_hints(param_hint), message=self.message - ) - - -class MissingParameter(BadParameter): - """Raised if click required an option or argument but it was not - provided when invoking the script. - - .. versionadded:: 4.0 - - :param param_type: a string that indicates the type of the parameter. - The default is to inherit the parameter type from - the given `param`. Valid values are ``'parameter'``, - ``'option'`` or ``'argument'``. - """ - - def __init__( - self, - message: str | None = None, - ctx: Context | None = None, - param: Parameter | None = None, - param_hint: cabc.Sequence[str] | str | None = None, - param_type: str | None = None, - ) -> None: - super().__init__(message or "", ctx, param, param_hint) - self.param_type = param_type - - def format_message(self) -> str: - if self.param_hint is not None: - param_hint: cabc.Sequence[str] | str | None = self.param_hint - elif self.param is not None: - param_hint = self.param.get_error_hint(self.ctx) # type: ignore - else: - param_hint = None - - param_hint = _join_param_hints(param_hint) - param_hint = f" {param_hint}" if param_hint else "" - - param_type = self.param_type - if param_type is None and self.param is not None: - param_type = self.param.param_type_name - - msg = self.message - if self.param is not None: - msg_extra = self.param.type.get_missing_message( - param=self.param, ctx=self.ctx - ) - if msg_extra: - if msg: - msg += f". {msg_extra}" - else: - msg = msg_extra - - msg = f" {msg}" if msg else "" - - # Translate param_type for known types. - if param_type == "argument": - missing = _("Missing argument") - elif param_type == "option": - missing = _("Missing option") - elif param_type == "parameter": - missing = _("Missing parameter") - else: - missing = _("Missing {param_type}").format(param_type=param_type) - - return f"{missing}{param_hint}.{msg}" - - def __str__(self) -> str: - if not self.message: - param_name = self.param.name if self.param else None - return _("Missing parameter: {param_name}").format(param_name=param_name) - else: - return self.message - - -class NoSuchOption(UsageError): - """Raised if click attempted to handle an option that does not - exist. - - .. versionadded:: 4.0 - """ - - def __init__( - self, - option_name: str, - message: str | None = None, - possibilities: cabc.Sequence[str] | None = None, - ctx: Context | None = None, - ) -> None: - if message is None: - message = _("No such option: {name}").format(name=option_name) - - super().__init__(message, ctx) - self.option_name = option_name - self.possibilities = possibilities - - def format_message(self) -> str: - if not self.possibilities: - return self.message - - possibility_str = ", ".join(sorted(self.possibilities)) - suggest = ngettext( - "Did you mean {possibility}?", - "(Possible options: {possibilities})", - len(self.possibilities), - ).format(possibility=possibility_str, possibilities=possibility_str) - return f"{self.message} {suggest}" - - -class BadOptionUsage(UsageError): - """Raised if an option is generally supplied but the use of the option - was incorrect. This is for instance raised if the number of arguments - for an option is not correct. - - .. versionadded:: 4.0 - - :param option_name: the name of the option being used incorrectly. - """ - - def __init__( - self, option_name: str, message: str, ctx: Context | None = None - ) -> None: - super().__init__(message, ctx) - self.option_name = option_name - - -class BadArgumentUsage(UsageError): - """Raised if an argument is generally supplied but the use of the argument - was incorrect. This is for instance raised if the number of values - for an argument is not correct. - - .. versionadded:: 6.0 - """ - - -class NoArgsIsHelpError(UsageError): - def __init__(self, ctx: Context) -> None: - self.ctx: Context - super().__init__(ctx.get_help(), ctx=ctx) - - def show(self, file: t.IO[t.Any] | None = None) -> None: - echo(self.format_message(), file=file, err=True, color=self.ctx.color) - - -class FileError(ClickException): - """Raised if a file cannot be opened.""" - - def __init__(self, filename: str, hint: str | None = None) -> None: - if hint is None: - hint = _("unknown error") - - super().__init__(hint) - self.ui_filename: str = format_filename(filename) - self.filename = filename - - def format_message(self) -> str: - return _("Could not open file {filename!r}: {message}").format( - filename=self.ui_filename, message=self.message - ) - - -class Abort(RuntimeError): - """An internal signalling exception that signals Click to abort.""" - - -class Exit(RuntimeError): - """An exception that indicates that the application should exit with some - status code. - - :param code: the status code to exit with. - """ - - __slots__ = ("exit_code",) - - def __init__(self, code: int = 0) -> None: - self.exit_code: int = code diff --git a/.env/Lib/site-packages/click/formatting.py b/.env/Lib/site-packages/click/formatting.py deleted file mode 100644 index 0b64f83..0000000 --- a/.env/Lib/site-packages/click/formatting.py +++ /dev/null @@ -1,301 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -from contextlib import contextmanager -from gettext import gettext as _ - -from ._compat import term_len -from .parser import _split_opt - -# Can force a width. This is used by the test system -FORCED_WIDTH: int | None = None - - -def measure_table(rows: cabc.Iterable[tuple[str, str]]) -> tuple[int, ...]: - widths: dict[int, int] = {} - - for row in rows: - for idx, col in enumerate(row): - widths[idx] = max(widths.get(idx, 0), term_len(col)) - - return tuple(y for x, y in sorted(widths.items())) - - -def iter_rows( - rows: cabc.Iterable[tuple[str, str]], col_count: int -) -> cabc.Iterator[tuple[str, ...]]: - for row in rows: - yield row + ("",) * (col_count - len(row)) - - -def wrap_text( - text: str, - width: int = 78, - initial_indent: str = "", - subsequent_indent: str = "", - preserve_paragraphs: bool = False, -) -> str: - """A helper function that intelligently wraps text. By default, it - assumes that it operates on a single paragraph of text but if the - `preserve_paragraphs` parameter is provided it will intelligently - handle paragraphs (defined by two empty lines). - - If paragraphs are handled, a paragraph can be prefixed with an empty - line containing the ``\\b`` character (``\\x08``) to indicate that - no rewrapping should happen in that block. - - :param text: the text that should be rewrapped. - :param width: the maximum width for the text. - :param initial_indent: the initial indent that should be placed on the - first line as a string. - :param subsequent_indent: the indent string that should be placed on - each consecutive line. - :param preserve_paragraphs: if this flag is set then the wrapping will - intelligently handle paragraphs. - """ - from ._textwrap import TextWrapper - - text = text.expandtabs() - wrapper = TextWrapper( - width, - initial_indent=initial_indent, - subsequent_indent=subsequent_indent, - replace_whitespace=False, - ) - if not preserve_paragraphs: - return wrapper.fill(text) - - p: list[tuple[int, bool, str]] = [] - buf: list[str] = [] - indent = None - - def _flush_par() -> None: - if not buf: - return - if buf[0].strip() == "\b": - p.append((indent or 0, True, "\n".join(buf[1:]))) - else: - p.append((indent or 0, False, " ".join(buf))) - del buf[:] - - for line in text.splitlines(): - if not line: - _flush_par() - indent = None - else: - if indent is None: - orig_len = term_len(line) - line = line.lstrip() - indent = orig_len - term_len(line) - buf.append(line) - _flush_par() - - rv = [] - for indent, raw, text in p: - with wrapper.extra_indent(" " * indent): - if raw: - rv.append(wrapper.indent_only(text)) - else: - rv.append(wrapper.fill(text)) - - return "\n\n".join(rv) - - -class HelpFormatter: - """This class helps with formatting text-based help pages. It's - usually just needed for very special internal cases, but it's also - exposed so that developers can write their own fancy outputs. - - At present, it always writes into memory. - - :param indent_increment: the additional increment for each level. - :param width: the width for the text. This defaults to the terminal - width clamped to a maximum of 78. - """ - - def __init__( - self, - indent_increment: int = 2, - width: int | None = None, - max_width: int | None = None, - ) -> None: - self.indent_increment = indent_increment - if max_width is None: - max_width = 80 - if width is None: - import shutil - - width = FORCED_WIDTH - if width is None: - width = max(min(shutil.get_terminal_size().columns, max_width) - 2, 50) - self.width = width - self.current_indent: int = 0 - self.buffer: list[str] = [] - - def write(self, string: str) -> None: - """Writes a unicode string into the internal buffer.""" - self.buffer.append(string) - - def indent(self) -> None: - """Increases the indentation.""" - self.current_indent += self.indent_increment - - def dedent(self) -> None: - """Decreases the indentation.""" - self.current_indent -= self.indent_increment - - def write_usage(self, prog: str, args: str = "", prefix: str | None = None) -> None: - """Writes a usage line into the buffer. - - :param prog: the program name. - :param args: whitespace separated list of arguments. - :param prefix: The prefix for the first line. Defaults to - ``"Usage: "``. - """ - if prefix is None: - prefix = f"{_('Usage:')} " - - usage_prefix = f"{prefix:>{self.current_indent}}{prog} " - text_width = self.width - self.current_indent - - if text_width >= (term_len(usage_prefix) + 20): - # The arguments will fit to the right of the prefix. - indent = " " * term_len(usage_prefix) - self.write( - wrap_text( - args, - text_width, - initial_indent=usage_prefix, - subsequent_indent=indent, - ) - ) - else: - # The prefix is too long, put the arguments on the next line. - self.write(usage_prefix) - self.write("\n") - indent = " " * (max(self.current_indent, term_len(prefix)) + 4) - self.write( - wrap_text( - args, text_width, initial_indent=indent, subsequent_indent=indent - ) - ) - - self.write("\n") - - def write_heading(self, heading: str) -> None: - """Writes a heading into the buffer.""" - self.write(f"{'':>{self.current_indent}}{heading}:\n") - - def write_paragraph(self) -> None: - """Writes a paragraph into the buffer.""" - if self.buffer: - self.write("\n") - - def write_text(self, text: str) -> None: - """Writes re-indented text into the buffer. This rewraps and - preserves paragraphs. - """ - indent = " " * self.current_indent - self.write( - wrap_text( - text, - self.width, - initial_indent=indent, - subsequent_indent=indent, - preserve_paragraphs=True, - ) - ) - self.write("\n") - - def write_dl( - self, - rows: cabc.Sequence[tuple[str, str]], - col_max: int = 30, - col_spacing: int = 2, - ) -> None: - """Writes a definition list into the buffer. This is how options - and commands are usually formatted. - - :param rows: a list of two item tuples for the terms and values. - :param col_max: the maximum width of the first column. - :param col_spacing: the number of spaces between the first and - second column. - """ - rows = list(rows) - widths = measure_table(rows) - if len(widths) != 2: - raise TypeError("Expected two columns for definition list") - - first_col = min(widths[0], col_max) + col_spacing - - for first, second in iter_rows(rows, len(widths)): - self.write(f"{'':>{self.current_indent}}{first}") - if not second: - self.write("\n") - continue - if term_len(first) <= first_col - col_spacing: - self.write(" " * (first_col - term_len(first))) - else: - self.write("\n") - self.write(" " * (first_col + self.current_indent)) - - text_width = max(self.width - first_col - 2, 10) - wrapped_text = wrap_text(second, text_width, preserve_paragraphs=True) - lines = wrapped_text.splitlines() - - if lines: - self.write(f"{lines[0]}\n") - - for line in lines[1:]: - self.write(f"{'':>{first_col + self.current_indent}}{line}\n") - else: - self.write("\n") - - @contextmanager - def section(self, name: str) -> cabc.Iterator[None]: - """Helpful context manager that writes a paragraph, a heading, - and the indents. - - :param name: the section name that is written as heading. - """ - self.write_paragraph() - self.write_heading(name) - self.indent() - try: - yield - finally: - self.dedent() - - @contextmanager - def indentation(self) -> cabc.Iterator[None]: - """A context manager that increases the indentation.""" - self.indent() - try: - yield - finally: - self.dedent() - - def getvalue(self) -> str: - """Returns the buffer contents.""" - return "".join(self.buffer) - - -def join_options(options: cabc.Sequence[str]) -> tuple[str, bool]: - """Given a list of option strings this joins them in the most appropriate - way and returns them in the form ``(formatted_string, - any_prefix_is_slash)`` where the second item in the tuple is a flag that - indicates if any of the option prefixes was a slash. - """ - rv = [] - any_prefix_is_slash = False - - for opt in options: - prefix = _split_opt(opt)[0] - - if prefix == "/": - any_prefix_is_slash = True - - rv.append((len(prefix), opt)) - - rv.sort(key=lambda x: x[0]) - return ", ".join(x[1] for x in rv), any_prefix_is_slash diff --git a/.env/Lib/site-packages/click/globals.py b/.env/Lib/site-packages/click/globals.py deleted file mode 100644 index a2f9172..0000000 --- a/.env/Lib/site-packages/click/globals.py +++ /dev/null @@ -1,67 +0,0 @@ -from __future__ import annotations - -import typing as t -from threading import local - -if t.TYPE_CHECKING: - from .core import Context - -_local = local() - - -@t.overload -def get_current_context(silent: t.Literal[False] = False) -> Context: ... - - -@t.overload -def get_current_context(silent: bool = ...) -> Context | None: ... - - -def get_current_context(silent: bool = False) -> Context | None: - """Returns the current click context. This can be used as a way to - access the current context object from anywhere. This is a more implicit - alternative to the :func:`pass_context` decorator. This function is - primarily useful for helpers such as :func:`echo` which might be - interested in changing its behavior based on the current context. - - To push the current context, :meth:`Context.scope` can be used. - - .. versionadded:: 5.0 - - :param silent: if set to `True` the return value is `None` if no context - is available. The default behavior is to raise a - :exc:`RuntimeError`. - """ - try: - return t.cast("Context", _local.stack[-1]) - except (AttributeError, IndexError) as e: - if not silent: - raise RuntimeError("There is no active click context.") from e - - return None - - -def push_context(ctx: Context) -> None: - """Pushes a new context to the current stack.""" - _local.__dict__.setdefault("stack", []).append(ctx) - - -def pop_context() -> None: - """Removes the top level from the stack.""" - _local.stack.pop() - - -def resolve_color_default(color: bool | None = None) -> bool | None: - """Internal helper to get the default value of the color flag. If a - value is passed it's returned unchanged, otherwise it's looked up from - the current context. - """ - if color is not None: - return color - - ctx = get_current_context(silent=True) - - if ctx is not None: - return ctx.color - - return None diff --git a/.env/Lib/site-packages/click/parser.py b/.env/Lib/site-packages/click/parser.py deleted file mode 100644 index 1ea1f71..0000000 --- a/.env/Lib/site-packages/click/parser.py +++ /dev/null @@ -1,532 +0,0 @@ -""" -This module started out as largely a copy paste from the stdlib's -optparse module with the features removed that we do not need from -optparse because we implement them in Click on a higher level (for -instance type handling, help formatting and a lot more). - -The plan is to remove more and more from here over time. - -The reason this is a different module and not optparse from the stdlib -is that there are differences in 2.x and 3.x about the error messages -generated and optparse in the stdlib uses gettext for no good reason -and might cause us issues. - -Click uses parts of optparse written by Gregory P. Ward and maintained -by the Python Software Foundation. This is limited to code in parser.py. - -Copyright 2001-2006 Gregory P. Ward. All rights reserved. -Copyright 2002-2006 Python Software Foundation. All rights reserved. -""" - -# This code uses parts of optparse written by Gregory P. Ward and -# maintained by the Python Software Foundation. -# Copyright 2001-2006 Gregory P. Ward -# Copyright 2002-2006 Python Software Foundation -from __future__ import annotations - -import collections.abc as cabc -import typing as t -from collections import deque -from gettext import gettext as _ -from gettext import ngettext - -from ._utils import FLAG_NEEDS_VALUE -from ._utils import UNSET -from .exceptions import BadArgumentUsage -from .exceptions import BadOptionUsage -from .exceptions import NoSuchOption -from .exceptions import UsageError - -if t.TYPE_CHECKING: - from ._utils import T_FLAG_NEEDS_VALUE - from ._utils import T_UNSET - from .core import Argument as CoreArgument - from .core import Context - from .core import Option as CoreOption - from .core import Parameter as CoreParameter - -V = t.TypeVar("V") - - -def _unpack_args( - args: cabc.Sequence[str], nargs_spec: cabc.Sequence[int] -) -> tuple[cabc.Sequence[str | cabc.Sequence[str | None] | None], list[str]]: - """Given an iterable of arguments and an iterable of nargs specifications, - it returns a tuple with all the unpacked arguments at the first index - and all remaining arguments as the second. - - The nargs specification is the number of arguments that should be consumed - or `-1` to indicate that this position should eat up all the remainders. - - Missing items are filled with ``UNSET``. - """ - args = deque(args) - nargs_spec = deque(nargs_spec) - rv: list[str | tuple[str | T_UNSET, ...] | T_UNSET] = [] - spos: int | None = None - - def _fetch(c: deque[V]) -> V | T_UNSET: - try: - if spos is None: - return c.popleft() - else: - return c.pop() - except IndexError: - return UNSET - - while nargs_spec: - nargs = _fetch(nargs_spec) - - if nargs is None: - continue - - if nargs == 1: - rv.append(_fetch(args)) # type: ignore[arg-type] - elif nargs > 1: - x = [_fetch(args) for _ in range(nargs)] - - # If we're reversed, we're pulling in the arguments in reverse, - # so we need to turn them around. - if spos is not None: - x.reverse() - - rv.append(tuple(x)) - elif nargs < 0: - if spos is not None: - raise TypeError("Cannot have two nargs < 0") - - spos = len(rv) - rv.append(UNSET) - - # spos is the position of the wildcard (star). If it's not `None`, - # we fill it with the remainder. - if spos is not None: - rv[spos] = tuple(args) - args = [] - rv[spos + 1 :] = reversed(rv[spos + 1 :]) - - return tuple(rv), list(args) - - -def _split_opt(opt: str) -> tuple[str, str]: - first = opt[:1] - if first.isalnum(): - return "", opt - if opt[1:2] == first: - return opt[:2], opt[2:] - return first, opt[1:] - - -def _normalize_opt(opt: str, ctx: Context | None) -> str: - if ctx is None or ctx.token_normalize_func is None: - return opt - prefix, opt = _split_opt(opt) - return f"{prefix}{ctx.token_normalize_func(opt)}" - - -class _Option: - def __init__( - self, - obj: CoreOption, - opts: cabc.Sequence[str], - dest: str | None, - action: str | None = None, - nargs: int = 1, - const: t.Any | None = None, - ): - self._short_opts = [] - self._long_opts = [] - self.prefixes: set[str] = set() - - for opt in opts: - prefix, value = _split_opt(opt) - if not prefix: - raise ValueError(f"Invalid start character for option ({opt})") - self.prefixes.add(prefix[0]) - if len(prefix) == 1 and len(value) == 1: - self._short_opts.append(opt) - else: - self._long_opts.append(opt) - self.prefixes.add(prefix) - - if action is None: - action = "store" - - self.dest = dest - self.action = action - self.nargs = nargs - self.const = const - self.obj = obj - - @property - def takes_value(self) -> bool: - return self.action in ("store", "append") - - def process(self, value: t.Any, state: _ParsingState) -> None: - if self.action == "store": - state.opts[self.dest] = value # type: ignore - elif self.action == "store_const": - state.opts[self.dest] = self.const # type: ignore - elif self.action == "append": - state.opts.setdefault(self.dest, []).append(value) # type: ignore - elif self.action == "append_const": - state.opts.setdefault(self.dest, []).append(self.const) # type: ignore - elif self.action == "count": - state.opts[self.dest] = state.opts.get(self.dest, 0) + 1 # type: ignore - else: - raise ValueError(f"unknown action '{self.action}'") - state.order.append(self.obj) - - -class _Argument: - def __init__(self, obj: CoreArgument, dest: str | None, nargs: int = 1): - self.dest = dest - self.nargs = nargs - self.obj = obj - - def process( - self, - value: str | cabc.Sequence[str | None] | None | T_UNSET, - state: _ParsingState, - ) -> None: - if self.nargs > 1: - assert isinstance(value, cabc.Sequence) - holes = sum(1 for x in value if x is UNSET) - if holes == len(value): - value = UNSET - elif holes != 0: - raise BadArgumentUsage( - _("Argument {name!r} takes {nargs} values.").format( - name=self.dest, nargs=self.nargs - ) - ) - - # We failed to collect any argument value so we consider the argument as unset. - if value == (): - value = UNSET - - state.opts[self.dest] = value # type: ignore - state.order.append(self.obj) - - -class _ParsingState: - def __init__(self, rargs: list[str]) -> None: - self.opts: dict[str, t.Any] = {} - self.largs: list[str] = [] - self.rargs = rargs - self.order: list[CoreParameter] = [] - - -class _OptionParser: - """The option parser is an internal class that is ultimately used to - parse options and arguments. It's modelled after optparse and brings - a similar but vastly simplified API. It should generally not be used - directly as the high level Click classes wrap it for you. - - It's not nearly as extensible as optparse or argparse as it does not - implement features that are implemented on a higher level (such as - types or defaults). - - :param ctx: optionally the :class:`~click.Context` where this parser - should go with. - - .. deprecated:: 8.2 - Will be removed in Click 9.0. - """ - - def __init__(self, ctx: Context | None = None) -> None: - #: The :class:`~click.Context` for this parser. This might be - #: `None` for some advanced use cases. - self.ctx = ctx - #: This controls how the parser deals with interspersed arguments. - #: If this is set to `False`, the parser will stop on the first - #: non-option. Click uses this to implement nested subcommands - #: safely. - self.allow_interspersed_args: bool = True - #: This tells the parser how to deal with unknown options. By - #: default it will error out (which is sensible), but there is a - #: second mode where it will ignore it and continue processing - #: after shifting all the unknown options into the resulting args. - self.ignore_unknown_options: bool = False - - if ctx is not None: - self.allow_interspersed_args = ctx.allow_interspersed_args - self.ignore_unknown_options = ctx.ignore_unknown_options - - self._short_opt: dict[str, _Option] = {} - self._long_opt: dict[str, _Option] = {} - self._opt_prefixes = {"-", "--"} - self._args: list[_Argument] = [] - - def add_option( - self, - obj: CoreOption, - opts: cabc.Sequence[str], - dest: str | None, - action: str | None = None, - nargs: int = 1, - const: t.Any | None = None, - ) -> None: - """Adds a new option named `dest` to the parser. The destination - is not inferred (unlike with optparse) and needs to be explicitly - provided. Action can be any of ``store``, ``store_const``, - ``append``, ``append_const`` or ``count``. - - The `obj` can be used to identify the option in the order list - that is returned from the parser. - """ - opts = [_normalize_opt(opt, self.ctx) for opt in opts] - option = _Option(obj, opts, dest, action=action, nargs=nargs, const=const) - self._opt_prefixes.update(option.prefixes) - for opt in option._short_opts: - self._short_opt[opt] = option - for opt in option._long_opts: - self._long_opt[opt] = option - - def add_argument(self, obj: CoreArgument, dest: str | None, nargs: int = 1) -> None: - """Adds a positional argument named `dest` to the parser. - - The `obj` can be used to identify the option in the order list - that is returned from the parser. - """ - self._args.append(_Argument(obj, dest=dest, nargs=nargs)) - - def parse_args( - self, args: list[str] - ) -> tuple[dict[str, t.Any], list[str], list[CoreParameter]]: - """Parses positional arguments and returns ``(values, args, order)`` - for the parsed options and arguments as well as the leftover - arguments if there are any. The order is a list of objects as they - appear on the command line. If arguments appear multiple times they - will be memorized multiple times as well. - """ - state = _ParsingState(args) - try: - self._process_args_for_options(state) - self._process_args_for_args(state) - except UsageError: - if self.ctx is None or not self.ctx.resilient_parsing: - raise - return state.opts, state.largs, state.order - - def _process_args_for_args(self, state: _ParsingState) -> None: - pargs, args = _unpack_args( - state.largs + state.rargs, [x.nargs for x in self._args] - ) - - for idx, arg in enumerate(self._args): - arg.process(pargs[idx], state) - - state.largs = args - state.rargs = [] - - def _process_args_for_options(self, state: _ParsingState) -> None: - while state.rargs: - arg = state.rargs.pop(0) - arglen = len(arg) - # Double dashes always handled explicitly regardless of what - # prefixes are valid. - if arg == "--": - return - elif arg[:1] in self._opt_prefixes and arglen > 1: - self._process_opts(arg, state) - elif self.allow_interspersed_args: - state.largs.append(arg) - else: - state.rargs.insert(0, arg) - return - - # Say this is the original argument list: - # [arg0, arg1, ..., arg(i-1), arg(i), arg(i+1), ..., arg(N-1)] - # ^ - # (we are about to process arg(i)). - # - # Then rargs is [arg(i), ..., arg(N-1)] and largs is a *subset* of - # [arg0, ..., arg(i-1)] (any options and their arguments will have - # been removed from largs). - # - # The while loop will usually consume 1 or more arguments per pass. - # If it consumes 1 (eg. arg is an option that takes no arguments), - # then after _process_arg() is done the situation is: - # - # largs = subset of [arg0, ..., arg(i)] - # rargs = [arg(i+1), ..., arg(N-1)] - # - # If allow_interspersed_args is false, largs will always be - # *empty* -- still a subset of [arg0, ..., arg(i-1)], but - # not a very interesting subset! - - def _match_long_opt( - self, opt: str, explicit_value: str | None, state: _ParsingState - ) -> None: - if opt not in self._long_opt: - from difflib import get_close_matches - - possibilities = get_close_matches(opt, self._long_opt) - raise NoSuchOption(opt, possibilities=possibilities, ctx=self.ctx) - - option = self._long_opt[opt] - if option.takes_value: - # At this point it's safe to modify rargs by injecting the - # explicit value, because no exception is raised in this - # branch. This means that the inserted value will be fully - # consumed. - if explicit_value is not None: - state.rargs.insert(0, explicit_value) - - value = self._get_value_from_state(opt, option, state) - - elif explicit_value is not None: - raise BadOptionUsage( - opt, _("Option {name!r} does not take a value.").format(name=opt) - ) - - else: - value = UNSET - - option.process(value, state) - - def _match_short_opt(self, arg: str, state: _ParsingState) -> None: - stop = False - i = 1 - prefix = arg[0] - unknown_options = [] - - for ch in arg[1:]: - opt = _normalize_opt(f"{prefix}{ch}", self.ctx) - option = self._short_opt.get(opt) - i += 1 - - if not option: - if self.ignore_unknown_options: - unknown_options.append(ch) - continue - raise NoSuchOption(opt, ctx=self.ctx) - if option.takes_value: - # Any characters left in arg? Pretend they're the - # next arg, and stop consuming characters of arg. - if i < len(arg): - state.rargs.insert(0, arg[i:]) - stop = True - - value = self._get_value_from_state(opt, option, state) - - else: - value = UNSET - - option.process(value, state) - - if stop: - break - - # If we got any unknown options we recombine the string of the - # remaining options and re-attach the prefix, then report that - # to the state as new larg. This way there is basic combinatorics - # that can be achieved while still ignoring unknown arguments. - if self.ignore_unknown_options and unknown_options: - state.largs.append(f"{prefix}{''.join(unknown_options)}") - - def _get_value_from_state( - self, option_name: str, option: _Option, state: _ParsingState - ) -> str | cabc.Sequence[str] | T_FLAG_NEEDS_VALUE: - nargs = option.nargs - - value: str | cabc.Sequence[str] | T_FLAG_NEEDS_VALUE - - if len(state.rargs) < nargs: - if option.obj._flag_needs_value: - # Option allows omitting the value. - value = FLAG_NEEDS_VALUE - else: - raise BadOptionUsage( - option_name, - ngettext( - "Option {name!r} requires an argument.", - "Option {name!r} requires {nargs} arguments.", - nargs, - ).format(name=option_name, nargs=nargs), - ) - elif nargs == 1: - next_rarg = state.rargs[0] - - if ( - option.obj._flag_needs_value - and isinstance(next_rarg, str) - and next_rarg[:1] in self._opt_prefixes - and len(next_rarg) > 1 - ): - # The next arg looks like the start of an option, don't - # use it as the value if omitting the value is allowed. - value = FLAG_NEEDS_VALUE - else: - value = state.rargs.pop(0) - else: - value = tuple(state.rargs[:nargs]) - del state.rargs[:nargs] - - return value - - def _process_opts(self, arg: str, state: _ParsingState) -> None: - explicit_value = None - # Long option handling happens in two parts. The first part is - # supporting explicitly attached values. In any case, we will try - # to long match the option first. - if "=" in arg: - long_opt, explicit_value = arg.split("=", 1) - else: - long_opt = arg - norm_long_opt = _normalize_opt(long_opt, self.ctx) - - # At this point we will match the (assumed) long option through - # the long option matching code. Note that this allows options - # like "-foo" to be matched as long options. - try: - self._match_long_opt(norm_long_opt, explicit_value, state) - except NoSuchOption: - # At this point the long option matching failed, and we need - # to try with short options. However there is a special rule - # which says, that if we have a two character options prefix - # (applies to "--foo" for instance), we do not dispatch to the - # short option code and will instead raise the no option - # error. - if arg[:2] not in self._opt_prefixes: - self._match_short_opt(arg, state) - return - - if not self.ignore_unknown_options: - raise - - state.largs.append(arg) - - -def __getattr__(name: str) -> object: - import warnings - - if name in { - "OptionParser", - "Argument", - "Option", - "split_opt", - "normalize_opt", - "ParsingState", - }: - warnings.warn( - f"'parser.{name}' is deprecated and will be removed in Click 9.0." - " The old parser is available in 'optparse'.", - DeprecationWarning, - stacklevel=2, - ) - return globals()[f"_{name}"] - - if name == "split_arg_string": - from .shell_completion import split_arg_string - - warnings.warn( - "Importing 'parser.split_arg_string' is deprecated, it will only be" - " available in 'shell_completion' in Click 9.0.", - DeprecationWarning, - stacklevel=2, - ) - return split_arg_string - - raise AttributeError(name) diff --git a/.env/Lib/site-packages/click/py.typed b/.env/Lib/site-packages/click/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/click/shell_completion.py b/.env/Lib/site-packages/click/shell_completion.py deleted file mode 100644 index 8f1564c..0000000 --- a/.env/Lib/site-packages/click/shell_completion.py +++ /dev/null @@ -1,667 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import os -import re -import typing as t -from gettext import gettext as _ - -from .core import Argument -from .core import Command -from .core import Context -from .core import Group -from .core import Option -from .core import Parameter -from .core import ParameterSource -from .utils import echo - - -def shell_complete( - cli: Command, - ctx_args: cabc.MutableMapping[str, t.Any], - prog_name: str, - complete_var: str, - instruction: str, -) -> int: - """Perform shell completion for the given CLI program. - - :param cli: Command being called. - :param ctx_args: Extra arguments to pass to - ``cli.make_context``. - :param prog_name: Name of the executable in the shell. - :param complete_var: Name of the environment variable that holds - the completion instruction. - :param instruction: Value of ``complete_var`` with the completion - instruction and shell, in the form ``instruction_shell``. - :return: Status code to exit with. - """ - shell, _, instruction = instruction.partition("_") - comp_cls = get_completion_class(shell) - - if comp_cls is None: - return 1 - - comp = comp_cls(cli, ctx_args, prog_name, complete_var) - - if instruction == "source": - echo(comp.source()) - return 0 - - if instruction == "complete": - echo(comp.complete()) - return 0 - - return 1 - - -class CompletionItem: - """Represents a completion value and metadata about the value. The - default metadata is ``type`` to indicate special shell handling, - and ``help`` if a shell supports showing a help string next to the - value. - - Arbitrary parameters can be passed when creating the object, and - accessed using ``item.attr``. If an attribute wasn't passed, - accessing it returns ``None``. - - :param value: The completion suggestion. - :param type: Tells the shell script to provide special completion - support for the type. Click uses ``"dir"`` and ``"file"``. - :param help: String shown next to the value if supported. - :param kwargs: Arbitrary metadata. The built-in implementations - don't use this, but custom type completions paired with custom - shell support could use it. - """ - - __slots__ = ("value", "type", "help", "_info") - - def __init__( - self, - value: t.Any, - type: str = "plain", - help: str | None = None, - **kwargs: t.Any, - ) -> None: - self.value: t.Any = value - self.type: str = type - self.help: str | None = help - self._info = kwargs - - def __getattr__(self, name: str) -> t.Any: - return self._info.get(name) - - -# Only Bash >= 4.4 has the nosort option. -_SOURCE_BASH = """\ -%(complete_func)s() { - local IFS=$'\\n' - local response - - response=$(env COMP_WORDS="${COMP_WORDS[*]}" COMP_CWORD=$COMP_CWORD \ -%(complete_var)s=bash_complete $1) - - for completion in $response; do - IFS=',' read type value <<< "$completion" - - if [[ $type == 'dir' ]]; then - COMPREPLY=() - compopt -o dirnames - elif [[ $type == 'file' ]]; then - COMPREPLY=() - compopt -o default - elif [[ $type == 'plain' ]]; then - COMPREPLY+=($value) - fi - done - - return 0 -} - -%(complete_func)s_setup() { - complete -o nosort -F %(complete_func)s %(prog_name)s -} - -%(complete_func)s_setup; -""" - -# See ZshComplete.format_completion below, and issue #2703, before -# changing this script. -# -# (TL;DR: _describe is picky about the format, but this Zsh script snippet -# is already widely deployed. So freeze this script, and use clever-ish -# handling of colons in ZshComplet.format_completion.) -_SOURCE_ZSH = """\ -#compdef %(prog_name)s - -%(complete_func)s() { - local -a completions - local -a completions_with_descriptions - local -a response - (( ! $+commands[%(prog_name)s] )) && return 1 - - response=("${(@f)$(env COMP_WORDS="${words[*]}" COMP_CWORD=$((CURRENT-1)) \ -%(complete_var)s=zsh_complete %(prog_name)s)}") - - for type key descr in ${response}; do - if [[ "$type" == "plain" ]]; then - if [[ "$descr" == "_" ]]; then - completions+=("$key") - else - completions_with_descriptions+=("$key":"$descr") - fi - elif [[ "$type" == "dir" ]]; then - _path_files -/ - elif [[ "$type" == "file" ]]; then - _path_files -f - fi - done - - if [ -n "$completions_with_descriptions" ]; then - _describe -V unsorted completions_with_descriptions -U - fi - - if [ -n "$completions" ]; then - compadd -U -V unsorted -a completions - fi -} - -if [[ $zsh_eval_context[-1] == loadautofunc ]]; then - # autoload from fpath, call function directly - %(complete_func)s "$@" -else - # eval/source/. command, register function for later - compdef %(complete_func)s %(prog_name)s -fi -""" - -_SOURCE_FISH = """\ -function %(complete_func)s; - set -l response (env %(complete_var)s=fish_complete COMP_WORDS=(commandline -cp) \ -COMP_CWORD=(commandline -t) %(prog_name)s); - - for completion in $response; - set -l metadata (string split "," $completion); - - if test $metadata[1] = "dir"; - __fish_complete_directories $metadata[2]; - else if test $metadata[1] = "file"; - __fish_complete_path $metadata[2]; - else if test $metadata[1] = "plain"; - echo $metadata[2]; - end; - end; -end; - -complete --no-files --command %(prog_name)s --arguments \ -"(%(complete_func)s)"; -""" - - -class ShellComplete: - """Base class for providing shell completion support. A subclass for - a given shell will override attributes and methods to implement the - completion instructions (``source`` and ``complete``). - - :param cli: Command being called. - :param prog_name: Name of the executable in the shell. - :param complete_var: Name of the environment variable that holds - the completion instruction. - - .. versionadded:: 8.0 - """ - - name: t.ClassVar[str] - """Name to register the shell as with :func:`add_completion_class`. - This is used in completion instructions (``{name}_source`` and - ``{name}_complete``). - """ - - source_template: t.ClassVar[str] - """Completion script template formatted by :meth:`source`. This must - be provided by subclasses. - """ - - def __init__( - self, - cli: Command, - ctx_args: cabc.MutableMapping[str, t.Any], - prog_name: str, - complete_var: str, - ) -> None: - self.cli = cli - self.ctx_args = ctx_args - self.prog_name = prog_name - self.complete_var = complete_var - - @property - def func_name(self) -> str: - """The name of the shell function defined by the completion - script. - """ - safe_name = re.sub(r"\W*", "", self.prog_name.replace("-", "_"), flags=re.ASCII) - return f"_{safe_name}_completion" - - def source_vars(self) -> dict[str, t.Any]: - """Vars for formatting :attr:`source_template`. - - By default this provides ``complete_func``, ``complete_var``, - and ``prog_name``. - """ - return { - "complete_func": self.func_name, - "complete_var": self.complete_var, - "prog_name": self.prog_name, - } - - def source(self) -> str: - """Produce the shell script that defines the completion - function. By default this ``%``-style formats - :attr:`source_template` with the dict returned by - :meth:`source_vars`. - """ - return self.source_template % self.source_vars() - - def get_completion_args(self) -> tuple[list[str], str]: - """Use the env vars defined by the shell script to return a - tuple of ``args, incomplete``. This must be implemented by - subclasses. - """ - raise NotImplementedError - - def get_completions(self, args: list[str], incomplete: str) -> list[CompletionItem]: - """Determine the context and last complete command or parameter - from the complete args. Call that object's ``shell_complete`` - method to get the completions for the incomplete value. - - :param args: List of complete args before the incomplete value. - :param incomplete: Value being completed. May be empty. - """ - ctx = _resolve_context(self.cli, self.ctx_args, self.prog_name, args) - obj, incomplete = _resolve_incomplete(ctx, args, incomplete) - return obj.shell_complete(ctx, incomplete) - - def format_completion(self, item: CompletionItem) -> str: - """Format a completion item into the form recognized by the - shell script. This must be implemented by subclasses. - - :param item: Completion item to format. - """ - raise NotImplementedError - - def complete(self) -> str: - """Produce the completion data to send back to the shell. - - By default this calls :meth:`get_completion_args`, gets the - completions, then calls :meth:`format_completion` for each - completion. - """ - args, incomplete = self.get_completion_args() - completions = self.get_completions(args, incomplete) - out = [self.format_completion(item) for item in completions] - return "\n".join(out) - - -class BashComplete(ShellComplete): - """Shell completion for Bash.""" - - name = "bash" - source_template = _SOURCE_BASH - - @staticmethod - def _check_version() -> None: - import shutil - import subprocess - - bash_exe = shutil.which("bash") - - if bash_exe is None: - match = None - else: - output = subprocess.run( - [bash_exe, "--norc", "-c", 'echo "${BASH_VERSION}"'], - stdout=subprocess.PIPE, - ) - match = re.search(r"^(\d+)\.(\d+)\.\d+", output.stdout.decode()) - - if match is not None: - major, minor = match.groups() - - if major < "4" or major == "4" and minor < "4": - echo( - _( - "Shell completion is not supported for Bash" - " versions older than 4.4." - ), - err=True, - ) - else: - echo( - _("Couldn't detect Bash version, shell completion is not supported."), - err=True, - ) - - def source(self) -> str: - self._check_version() - return super().source() - - def get_completion_args(self) -> tuple[list[str], str]: - cwords = split_arg_string(os.environ["COMP_WORDS"]) - cword = int(os.environ["COMP_CWORD"]) - args = cwords[1:cword] - - try: - incomplete = cwords[cword] - except IndexError: - incomplete = "" - - return args, incomplete - - def format_completion(self, item: CompletionItem) -> str: - return f"{item.type},{item.value}" - - -class ZshComplete(ShellComplete): - """Shell completion for Zsh.""" - - name = "zsh" - source_template = _SOURCE_ZSH - - def get_completion_args(self) -> tuple[list[str], str]: - cwords = split_arg_string(os.environ["COMP_WORDS"]) - cword = int(os.environ["COMP_CWORD"]) - args = cwords[1:cword] - - try: - incomplete = cwords[cword] - except IndexError: - incomplete = "" - - return args, incomplete - - def format_completion(self, item: CompletionItem) -> str: - help_ = item.help or "_" - # The zsh completion script uses `_describe` on items with help - # texts (which splits the item help from the item value at the - # first unescaped colon) and `compadd` on items without help - # text (which uses the item value as-is and does not support - # colon escaping). So escape colons in the item value if and - # only if the item help is not the sentinel "_" value, as used - # by the completion script. - # - # (The zsh completion script is potentially widely deployed, and - # thus harder to fix than this method.) - # - # See issue #1812 and issue #2703 for further context. - value = item.value.replace(":", r"\:") if help_ != "_" else item.value - return f"{item.type}\n{value}\n{help_}" - - -class FishComplete(ShellComplete): - """Shell completion for Fish.""" - - name = "fish" - source_template = _SOURCE_FISH - - def get_completion_args(self) -> tuple[list[str], str]: - cwords = split_arg_string(os.environ["COMP_WORDS"]) - incomplete = os.environ["COMP_CWORD"] - if incomplete: - incomplete = split_arg_string(incomplete)[0] - args = cwords[1:] - - # Fish stores the partial word in both COMP_WORDS and - # COMP_CWORD, remove it from complete args. - if incomplete and args and args[-1] == incomplete: - args.pop() - - return args, incomplete - - def format_completion(self, item: CompletionItem) -> str: - if item.help: - return f"{item.type},{item.value}\t{item.help}" - - return f"{item.type},{item.value}" - - -ShellCompleteType = t.TypeVar("ShellCompleteType", bound="type[ShellComplete]") - - -_available_shells: dict[str, type[ShellComplete]] = { - "bash": BashComplete, - "fish": FishComplete, - "zsh": ZshComplete, -} - - -def add_completion_class( - cls: ShellCompleteType, name: str | None = None -) -> ShellCompleteType: - """Register a :class:`ShellComplete` subclass under the given name. - The name will be provided by the completion instruction environment - variable during completion. - - :param cls: The completion class that will handle completion for the - shell. - :param name: Name to register the class under. Defaults to the - class's ``name`` attribute. - """ - if name is None: - name = cls.name - - _available_shells[name] = cls - - return cls - - -def get_completion_class(shell: str) -> type[ShellComplete] | None: - """Look up a registered :class:`ShellComplete` subclass by the name - provided by the completion instruction environment variable. If the - name isn't registered, returns ``None``. - - :param shell: Name the class is registered under. - """ - return _available_shells.get(shell) - - -def split_arg_string(string: str) -> list[str]: - """Split an argument string as with :func:`shlex.split`, but don't - fail if the string is incomplete. Ignores a missing closing quote or - incomplete escape sequence and uses the partial token as-is. - - .. code-block:: python - - split_arg_string("example 'my file") - ["example", "my file"] - - split_arg_string("example my\\") - ["example", "my"] - - :param string: String to split. - - .. versionchanged:: 8.2 - Moved to ``shell_completion`` from ``parser``. - """ - import shlex - - lex = shlex.shlex(string, posix=True) - lex.whitespace_split = True - lex.commenters = "" - out = [] - - try: - for token in lex: - out.append(token) - except ValueError: - # Raised when end-of-string is reached in an invalid state. Use - # the partial token as-is. The quote or escape character is in - # lex.state, not lex.token. - out.append(lex.token) - - return out - - -def _is_incomplete_argument(ctx: Context, param: Parameter) -> bool: - """Determine if the given parameter is an argument that can still - accept values. - - :param ctx: Invocation context for the command represented by the - parsed complete args. - :param param: Argument object being checked. - """ - if not isinstance(param, Argument): - return False - - assert param.name is not None - # Will be None if expose_value is False. - value = ctx.params.get(param.name) - return ( - param.nargs == -1 - or ctx.get_parameter_source(param.name) is not ParameterSource.COMMANDLINE - or ( - param.nargs > 1 - and isinstance(value, (tuple, list)) - and len(value) < param.nargs - ) - ) - - -def _start_of_option(ctx: Context, value: str) -> bool: - """Check if the value looks like the start of an option.""" - if not value: - return False - - c = value[0] - return c in ctx._opt_prefixes - - -def _is_incomplete_option(ctx: Context, args: list[str], param: Parameter) -> bool: - """Determine if the given parameter is an option that needs a value. - - :param args: List of complete args before the incomplete value. - :param param: Option object being checked. - """ - if not isinstance(param, Option): - return False - - if param.is_flag or param.count: - return False - - last_option = None - - for index, arg in enumerate(reversed(args)): - if index + 1 > param.nargs: - break - - if _start_of_option(ctx, arg): - last_option = arg - break - - return last_option is not None and last_option in param.opts - - -def _resolve_context( - cli: Command, - ctx_args: cabc.MutableMapping[str, t.Any], - prog_name: str, - args: list[str], -) -> Context: - """Produce the context hierarchy starting with the command and - traversing the complete arguments. This only follows the commands, - it doesn't trigger input prompts or callbacks. - - :param cli: Command being called. - :param prog_name: Name of the executable in the shell. - :param args: List of complete args before the incomplete value. - """ - ctx_args["resilient_parsing"] = True - with cli.make_context(prog_name, args.copy(), **ctx_args) as ctx: - args = ctx._protected_args + ctx.args - - while args: - command = ctx.command - - if isinstance(command, Group): - if not command.chain: - name, cmd, args = command.resolve_command(ctx, args) - - if cmd is None: - return ctx - - with cmd.make_context( - name, args, parent=ctx, resilient_parsing=True - ) as sub_ctx: - ctx = sub_ctx - args = ctx._protected_args + ctx.args - else: - sub_ctx = ctx - - while args: - name, cmd, args = command.resolve_command(ctx, args) - - if cmd is None: - return ctx - - with cmd.make_context( - name, - args, - parent=ctx, - allow_extra_args=True, - allow_interspersed_args=False, - resilient_parsing=True, - ) as sub_sub_ctx: - sub_ctx = sub_sub_ctx - args = sub_ctx.args - - ctx = sub_ctx - args = [*sub_ctx._protected_args, *sub_ctx.args] - else: - break - - return ctx - - -def _resolve_incomplete( - ctx: Context, args: list[str], incomplete: str -) -> tuple[Command | Parameter, str]: - """Find the Click object that will handle the completion of the - incomplete value. Return the object and the incomplete value. - - :param ctx: Invocation context for the command represented by - the parsed complete args. - :param args: List of complete args before the incomplete value. - :param incomplete: Value being completed. May be empty. - """ - # Different shells treat an "=" between a long option name and - # value differently. Might keep the value joined, return the "=" - # as a separate item, or return the split name and value. Always - # split and discard the "=" to make completion easier. - if incomplete == "=": - incomplete = "" - elif "=" in incomplete and _start_of_option(ctx, incomplete): - name, _, incomplete = incomplete.partition("=") - args.append(name) - - # The "--" marker tells Click to stop treating values as options - # even if they start with the option character. If it hasn't been - # given and the incomplete arg looks like an option, the current - # command will provide option name completions. - if "--" not in args and _start_of_option(ctx, incomplete): - return ctx.command, incomplete - - params = ctx.command.get_params(ctx) - - # If the last complete arg is an option name with an incomplete - # value, the option will provide value completions. - for param in params: - if _is_incomplete_option(ctx, args, param): - return param, incomplete - - # It's not an option name or value. The first argument without a - # parsed value will provide value completions. - for param in params: - if _is_incomplete_argument(ctx, param): - return param, incomplete - - # There were no unparsed arguments, the command may be a group that - # will provide command name completions. - return ctx.command, incomplete diff --git a/.env/Lib/site-packages/click/termui.py b/.env/Lib/site-packages/click/termui.py deleted file mode 100644 index 2e98a07..0000000 --- a/.env/Lib/site-packages/click/termui.py +++ /dev/null @@ -1,883 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import inspect -import io -import itertools -import sys -import typing as t -from contextlib import AbstractContextManager -from gettext import gettext as _ - -from ._compat import isatty -from ._compat import strip_ansi -from .exceptions import Abort -from .exceptions import UsageError -from .globals import resolve_color_default -from .types import Choice -from .types import convert_type -from .types import ParamType -from .utils import echo -from .utils import LazyFile - -if t.TYPE_CHECKING: - from ._termui_impl import ProgressBar - -V = t.TypeVar("V") - -# The prompt functions to use. The doc tools currently override these -# functions to customize how they work. -visible_prompt_func: t.Callable[[str], str] = input - -_ansi_colors = { - "black": 30, - "red": 31, - "green": 32, - "yellow": 33, - "blue": 34, - "magenta": 35, - "cyan": 36, - "white": 37, - "reset": 39, - "bright_black": 90, - "bright_red": 91, - "bright_green": 92, - "bright_yellow": 93, - "bright_blue": 94, - "bright_magenta": 95, - "bright_cyan": 96, - "bright_white": 97, -} -_ansi_reset_all = "\033[0m" - - -def hidden_prompt_func(prompt: str) -> str: - import getpass - - return getpass.getpass(prompt) - - -def _build_prompt( - text: str, - suffix: str, - show_default: bool = False, - default: t.Any | None = None, - show_choices: bool = True, - type: ParamType | None = None, -) -> str: - prompt = text - if type is not None and show_choices and isinstance(type, Choice): - prompt += f" ({', '.join(map(str, type.choices))})" - if default is not None and show_default: - prompt = f"{prompt} [{_format_default(default)}]" - return f"{prompt}{suffix}" - - -def _format_default(default: t.Any) -> t.Any: - if isinstance(default, (io.IOBase, LazyFile)) and hasattr(default, "name"): - return default.name - - return default - - -def prompt( - text: str, - default: t.Any | None = None, - hide_input: bool = False, - confirmation_prompt: bool | str = False, - type: ParamType | t.Any | None = None, - value_proc: t.Callable[[str], t.Any] | None = None, - prompt_suffix: str = ": ", - show_default: bool = True, - err: bool = False, - show_choices: bool = True, -) -> t.Any: - """Prompts a user for input. This is a convenience function that can - be used to prompt a user for input later. - - If the user aborts the input by sending an interrupt signal, this - function will catch it and raise a :exc:`Abort` exception. - - :param text: the text to show for the prompt. - :param default: the default value to use if no input happens. If this - is not given it will prompt until it's aborted. - :param hide_input: if this is set to true then the input value will - be hidden. - :param confirmation_prompt: Prompt a second time to confirm the - value. Can be set to a string instead of ``True`` to customize - the message. - :param type: the type to use to check the value against. - :param value_proc: if this parameter is provided it's a function that - is invoked instead of the type conversion to - convert a value. - :param prompt_suffix: a suffix that should be added to the prompt. - :param show_default: shows or hides the default value in the prompt. - :param err: if set to true the file defaults to ``stderr`` instead of - ``stdout``, the same as with echo. - :param show_choices: Show or hide choices if the passed type is a Choice. - For example if type is a Choice of either day or week, - show_choices is true and text is "Group by" then the - prompt will be "Group by (day, week): ". - - .. versionchanged:: 8.3.1 - A space is no longer appended to the prompt. - - .. versionadded:: 8.0 - ``confirmation_prompt`` can be a custom string. - - .. versionadded:: 7.0 - Added the ``show_choices`` parameter. - - .. versionadded:: 6.0 - Added unicode support for cmd.exe on Windows. - - .. versionadded:: 4.0 - Added the `err` parameter. - - """ - - def prompt_func(text: str) -> str: - f = hidden_prompt_func if hide_input else visible_prompt_func - try: - # Write the prompt separately so that we get nice - # coloring through colorama on Windows - echo(text[:-1], nl=False, err=err) - # Echo the last character to stdout to work around an issue where - # readline causes backspace to clear the whole line. - return f(text[-1:]) - except (KeyboardInterrupt, EOFError): - # getpass doesn't print a newline if the user aborts input with ^C. - # Allegedly this behavior is inherited from getpass(3). - # A doc bug has been filed at https://bugs.python.org/issue24711 - if hide_input: - echo(None, err=err) - raise Abort() from None - - if value_proc is None: - value_proc = convert_type(type, default) - - prompt = _build_prompt( - text, prompt_suffix, show_default, default, show_choices, type - ) - - if confirmation_prompt: - if confirmation_prompt is True: - confirmation_prompt = _("Repeat for confirmation") - - confirmation_prompt = _build_prompt(confirmation_prompt, prompt_suffix) - - while True: - while True: - value = prompt_func(prompt) - if value: - break - elif default is not None: - value = default - break - try: - result = value_proc(value) - except UsageError as e: - if hide_input: - echo(_("Error: The value you entered was invalid."), err=err) - else: - echo(_("Error: {e.message}").format(e=e), err=err) - continue - if not confirmation_prompt: - return result - while True: - value2 = prompt_func(confirmation_prompt) - is_empty = not value and not value2 - if value2 or is_empty: - break - if value == value2: - return result - echo(_("Error: The two entered values do not match."), err=err) - - -def confirm( - text: str, - default: bool | None = False, - abort: bool = False, - prompt_suffix: str = ": ", - show_default: bool = True, - err: bool = False, -) -> bool: - """Prompts for confirmation (yes/no question). - - If the user aborts the input by sending a interrupt signal this - function will catch it and raise a :exc:`Abort` exception. - - :param text: the question to ask. - :param default: The default value to use when no input is given. If - ``None``, repeat until input is given. - :param abort: if this is set to `True` a negative answer aborts the - exception by raising :exc:`Abort`. - :param prompt_suffix: a suffix that should be added to the prompt. - :param show_default: shows or hides the default value in the prompt. - :param err: if set to true the file defaults to ``stderr`` instead of - ``stdout``, the same as with echo. - - .. versionchanged:: 8.3.1 - A space is no longer appended to the prompt. - - .. versionchanged:: 8.0 - Repeat until input is given if ``default`` is ``None``. - - .. versionadded:: 4.0 - Added the ``err`` parameter. - """ - prompt = _build_prompt( - text, - prompt_suffix, - show_default, - "y/n" if default is None else ("Y/n" if default else "y/N"), - ) - - while True: - try: - # Write the prompt separately so that we get nice - # coloring through colorama on Windows - echo(prompt[:-1], nl=False, err=err) - # Echo the last character to stdout to work around an issue where - # readline causes backspace to clear the whole line. - value = visible_prompt_func(prompt[-1:]).lower().strip() - except (KeyboardInterrupt, EOFError): - raise Abort() from None - if value in ("y", "yes"): - rv = True - elif value in ("n", "no"): - rv = False - elif default is not None and value == "": - rv = default - else: - echo(_("Error: invalid input"), err=err) - continue - break - if abort and not rv: - raise Abort() - return rv - - -def echo_via_pager( - text_or_generator: cabc.Iterable[str] | t.Callable[[], cabc.Iterable[str]] | str, - color: bool | None = None, -) -> None: - """This function takes a text and shows it via an environment specific - pager on stdout. - - .. versionchanged:: 3.0 - Added the `color` flag. - - :param text_or_generator: the text to page, or alternatively, a - generator emitting the text to page. - :param color: controls if the pager supports ANSI colors or not. The - default is autodetection. - """ - color = resolve_color_default(color) - - if inspect.isgeneratorfunction(text_or_generator): - i = t.cast("t.Callable[[], cabc.Iterable[str]]", text_or_generator)() - elif isinstance(text_or_generator, str): - i = [text_or_generator] - else: - i = iter(t.cast("cabc.Iterable[str]", text_or_generator)) - - # convert every element of i to a text type if necessary - text_generator = (el if isinstance(el, str) else str(el) for el in i) - - from ._termui_impl import pager - - return pager(itertools.chain(text_generator, "\n"), color) - - -@t.overload -def progressbar( - *, - length: int, - label: str | None = None, - hidden: bool = False, - show_eta: bool = True, - show_percent: bool | None = None, - show_pos: bool = False, - fill_char: str = "#", - empty_char: str = "-", - bar_template: str = "%(label)s [%(bar)s] %(info)s", - info_sep: str = " ", - width: int = 36, - file: t.TextIO | None = None, - color: bool | None = None, - update_min_steps: int = 1, -) -> ProgressBar[int]: ... - - -@t.overload -def progressbar( - iterable: cabc.Iterable[V] | None = None, - length: int | None = None, - label: str | None = None, - hidden: bool = False, - show_eta: bool = True, - show_percent: bool | None = None, - show_pos: bool = False, - item_show_func: t.Callable[[V | None], str | None] | None = None, - fill_char: str = "#", - empty_char: str = "-", - bar_template: str = "%(label)s [%(bar)s] %(info)s", - info_sep: str = " ", - width: int = 36, - file: t.TextIO | None = None, - color: bool | None = None, - update_min_steps: int = 1, -) -> ProgressBar[V]: ... - - -def progressbar( - iterable: cabc.Iterable[V] | None = None, - length: int | None = None, - label: str | None = None, - hidden: bool = False, - show_eta: bool = True, - show_percent: bool | None = None, - show_pos: bool = False, - item_show_func: t.Callable[[V | None], str | None] | None = None, - fill_char: str = "#", - empty_char: str = "-", - bar_template: str = "%(label)s [%(bar)s] %(info)s", - info_sep: str = " ", - width: int = 36, - file: t.TextIO | None = None, - color: bool | None = None, - update_min_steps: int = 1, -) -> ProgressBar[V]: - """This function creates an iterable context manager that can be used - to iterate over something while showing a progress bar. It will - either iterate over the `iterable` or `length` items (that are counted - up). While iteration happens, this function will print a rendered - progress bar to the given `file` (defaults to stdout) and will attempt - to calculate remaining time and more. By default, this progress bar - will not be rendered if the file is not a terminal. - - The context manager creates the progress bar. When the context - manager is entered the progress bar is already created. With every - iteration over the progress bar, the iterable passed to the bar is - advanced and the bar is updated. When the context manager exits, - a newline is printed and the progress bar is finalized on screen. - - Note: The progress bar is currently designed for use cases where the - total progress can be expected to take at least several seconds. - Because of this, the ProgressBar class object won't display - progress that is considered too fast, and progress where the time - between steps is less than a second. - - No printing must happen or the progress bar will be unintentionally - destroyed. - - Example usage:: - - with progressbar(items) as bar: - for item in bar: - do_something_with(item) - - Alternatively, if no iterable is specified, one can manually update the - progress bar through the `update()` method instead of directly - iterating over the progress bar. The update method accepts the number - of steps to increment the bar with:: - - with progressbar(length=chunks.total_bytes) as bar: - for chunk in chunks: - process_chunk(chunk) - bar.update(chunks.bytes) - - The ``update()`` method also takes an optional value specifying the - ``current_item`` at the new position. This is useful when used - together with ``item_show_func`` to customize the output for each - manual step:: - - with click.progressbar( - length=total_size, - label='Unzipping archive', - item_show_func=lambda a: a.filename - ) as bar: - for archive in zip_file: - archive.extract() - bar.update(archive.size, archive) - - :param iterable: an iterable to iterate over. If not provided the length - is required. - :param length: the number of items to iterate over. By default the - progressbar will attempt to ask the iterator about its - length, which might or might not work. If an iterable is - also provided this parameter can be used to override the - length. If an iterable is not provided the progress bar - will iterate over a range of that length. - :param label: the label to show next to the progress bar. - :param hidden: hide the progressbar. Defaults to ``False``. When no tty is - detected, it will only print the progressbar label. Setting this to - ``False`` also disables that. - :param show_eta: enables or disables the estimated time display. This is - automatically disabled if the length cannot be - determined. - :param show_percent: enables or disables the percentage display. The - default is `True` if the iterable has a length or - `False` if not. - :param show_pos: enables or disables the absolute position display. The - default is `False`. - :param item_show_func: A function called with the current item which - can return a string to show next to the progress bar. If the - function returns ``None`` nothing is shown. The current item can - be ``None``, such as when entering and exiting the bar. - :param fill_char: the character to use to show the filled part of the - progress bar. - :param empty_char: the character to use to show the non-filled part of - the progress bar. - :param bar_template: the format string to use as template for the bar. - The parameters in it are ``label`` for the label, - ``bar`` for the progress bar and ``info`` for the - info section. - :param info_sep: the separator between multiple info items (eta etc.) - :param width: the width of the progress bar in characters, 0 means full - terminal width - :param file: The file to write to. If this is not a terminal then - only the label is printed. - :param color: controls if the terminal supports ANSI colors or not. The - default is autodetection. This is only needed if ANSI - codes are included anywhere in the progress bar output - which is not the case by default. - :param update_min_steps: Render only when this many updates have - completed. This allows tuning for very fast iterators. - - .. versionadded:: 8.2 - The ``hidden`` argument. - - .. versionchanged:: 8.0 - Output is shown even if execution time is less than 0.5 seconds. - - .. versionchanged:: 8.0 - ``item_show_func`` shows the current item, not the previous one. - - .. versionchanged:: 8.0 - Labels are echoed if the output is not a TTY. Reverts a change - in 7.0 that removed all output. - - .. versionadded:: 8.0 - The ``update_min_steps`` parameter. - - .. versionadded:: 4.0 - The ``color`` parameter and ``update`` method. - - .. versionadded:: 2.0 - """ - from ._termui_impl import ProgressBar - - color = resolve_color_default(color) - return ProgressBar( - iterable=iterable, - length=length, - hidden=hidden, - show_eta=show_eta, - show_percent=show_percent, - show_pos=show_pos, - item_show_func=item_show_func, - fill_char=fill_char, - empty_char=empty_char, - bar_template=bar_template, - info_sep=info_sep, - file=file, - label=label, - width=width, - color=color, - update_min_steps=update_min_steps, - ) - - -def clear() -> None: - """Clears the terminal screen. This will have the effect of clearing - the whole visible space of the terminal and moving the cursor to the - top left. This does not do anything if not connected to a terminal. - - .. versionadded:: 2.0 - """ - if not isatty(sys.stdout): - return - - # ANSI escape \033[2J clears the screen, \033[1;1H moves the cursor - echo("\033[2J\033[1;1H", nl=False) - - -def _interpret_color(color: int | tuple[int, int, int] | str, offset: int = 0) -> str: - if isinstance(color, int): - return f"{38 + offset};5;{color:d}" - - if isinstance(color, (tuple, list)): - r, g, b = color - return f"{38 + offset};2;{r:d};{g:d};{b:d}" - - return str(_ansi_colors[color] + offset) - - -def style( - text: t.Any, - fg: int | tuple[int, int, int] | str | None = None, - bg: int | tuple[int, int, int] | str | None = None, - bold: bool | None = None, - dim: bool | None = None, - underline: bool | None = None, - overline: bool | None = None, - italic: bool | None = None, - blink: bool | None = None, - reverse: bool | None = None, - strikethrough: bool | None = None, - reset: bool = True, -) -> str: - """Styles a text with ANSI styles and returns the new string. By - default the styling is self contained which means that at the end - of the string a reset code is issued. This can be prevented by - passing ``reset=False``. - - Examples:: - - click.echo(click.style('Hello World!', fg='green')) - click.echo(click.style('ATTENTION!', blink=True)) - click.echo(click.style('Some things', reverse=True, fg='cyan')) - click.echo(click.style('More colors', fg=(255, 12, 128), bg=117)) - - Supported color names: - - * ``black`` (might be a gray) - * ``red`` - * ``green`` - * ``yellow`` (might be an orange) - * ``blue`` - * ``magenta`` - * ``cyan`` - * ``white`` (might be light gray) - * ``bright_black`` - * ``bright_red`` - * ``bright_green`` - * ``bright_yellow`` - * ``bright_blue`` - * ``bright_magenta`` - * ``bright_cyan`` - * ``bright_white`` - * ``reset`` (reset the color code only) - - If the terminal supports it, color may also be specified as: - - - An integer in the interval [0, 255]. The terminal must support - 8-bit/256-color mode. - - An RGB tuple of three integers in [0, 255]. The terminal must - support 24-bit/true-color mode. - - See https://en.wikipedia.org/wiki/ANSI_color and - https://gist.github.com/XVilka/8346728 for more information. - - :param text: the string to style with ansi codes. - :param fg: if provided this will become the foreground color. - :param bg: if provided this will become the background color. - :param bold: if provided this will enable or disable bold mode. - :param dim: if provided this will enable or disable dim mode. This is - badly supported. - :param underline: if provided this will enable or disable underline. - :param overline: if provided this will enable or disable overline. - :param italic: if provided this will enable or disable italic. - :param blink: if provided this will enable or disable blinking. - :param reverse: if provided this will enable or disable inverse - rendering (foreground becomes background and the - other way round). - :param strikethrough: if provided this will enable or disable - striking through text. - :param reset: by default a reset-all code is added at the end of the - string which means that styles do not carry over. This - can be disabled to compose styles. - - .. versionchanged:: 8.0 - A non-string ``message`` is converted to a string. - - .. versionchanged:: 8.0 - Added support for 256 and RGB color codes. - - .. versionchanged:: 8.0 - Added the ``strikethrough``, ``italic``, and ``overline`` - parameters. - - .. versionchanged:: 7.0 - Added support for bright colors. - - .. versionadded:: 2.0 - """ - if not isinstance(text, str): - text = str(text) - - bits = [] - - if fg: - try: - bits.append(f"\033[{_interpret_color(fg)}m") - except KeyError: - raise TypeError(f"Unknown color {fg!r}") from None - - if bg: - try: - bits.append(f"\033[{_interpret_color(bg, 10)}m") - except KeyError: - raise TypeError(f"Unknown color {bg!r}") from None - - if bold is not None: - bits.append(f"\033[{1 if bold else 22}m") - if dim is not None: - bits.append(f"\033[{2 if dim else 22}m") - if underline is not None: - bits.append(f"\033[{4 if underline else 24}m") - if overline is not None: - bits.append(f"\033[{53 if overline else 55}m") - if italic is not None: - bits.append(f"\033[{3 if italic else 23}m") - if blink is not None: - bits.append(f"\033[{5 if blink else 25}m") - if reverse is not None: - bits.append(f"\033[{7 if reverse else 27}m") - if strikethrough is not None: - bits.append(f"\033[{9 if strikethrough else 29}m") - bits.append(text) - if reset: - bits.append(_ansi_reset_all) - return "".join(bits) - - -def unstyle(text: str) -> str: - """Removes ANSI styling information from a string. Usually it's not - necessary to use this function as Click's echo function will - automatically remove styling if necessary. - - .. versionadded:: 2.0 - - :param text: the text to remove style information from. - """ - return strip_ansi(text) - - -def secho( - message: t.Any | None = None, - file: t.IO[t.AnyStr] | None = None, - nl: bool = True, - err: bool = False, - color: bool | None = None, - **styles: t.Any, -) -> None: - """This function combines :func:`echo` and :func:`style` into one - call. As such the following two calls are the same:: - - click.secho('Hello World!', fg='green') - click.echo(click.style('Hello World!', fg='green')) - - All keyword arguments are forwarded to the underlying functions - depending on which one they go with. - - Non-string types will be converted to :class:`str`. However, - :class:`bytes` are passed directly to :meth:`echo` without applying - style. If you want to style bytes that represent text, call - :meth:`bytes.decode` first. - - .. versionchanged:: 8.0 - A non-string ``message`` is converted to a string. Bytes are - passed through without style applied. - - .. versionadded:: 2.0 - """ - if message is not None and not isinstance(message, (bytes, bytearray)): - message = style(message, **styles) - - return echo(message, file=file, nl=nl, err=err, color=color) - - -@t.overload -def edit( - text: bytes | bytearray, - editor: str | None = None, - env: cabc.Mapping[str, str] | None = None, - require_save: bool = False, - extension: str = ".txt", -) -> bytes | None: ... - - -@t.overload -def edit( - text: str, - editor: str | None = None, - env: cabc.Mapping[str, str] | None = None, - require_save: bool = True, - extension: str = ".txt", -) -> str | None: ... - - -@t.overload -def edit( - text: None = None, - editor: str | None = None, - env: cabc.Mapping[str, str] | None = None, - require_save: bool = True, - extension: str = ".txt", - filename: str | cabc.Iterable[str] | None = None, -) -> None: ... - - -def edit( - text: str | bytes | bytearray | None = None, - editor: str | None = None, - env: cabc.Mapping[str, str] | None = None, - require_save: bool = True, - extension: str = ".txt", - filename: str | cabc.Iterable[str] | None = None, -) -> str | bytes | bytearray | None: - r"""Edits the given text in the defined editor. If an editor is given - (should be the full path to the executable but the regular operating - system search path is used for finding the executable) it overrides - the detected editor. Optionally, some environment variables can be - used. If the editor is closed without changes, `None` is returned. In - case a file is edited directly the return value is always `None` and - `require_save` and `extension` are ignored. - - If the editor cannot be opened a :exc:`UsageError` is raised. - - Note for Windows: to simplify cross-platform usage, the newlines are - automatically converted from POSIX to Windows and vice versa. As such, - the message here will have ``\n`` as newline markers. - - :param text: the text to edit. - :param editor: optionally the editor to use. Defaults to automatic - detection. - :param env: environment variables to forward to the editor. - :param require_save: if this is true, then not saving in the editor - will make the return value become `None`. - :param extension: the extension to tell the editor about. This defaults - to `.txt` but changing this might change syntax - highlighting. - :param filename: if provided it will edit this file instead of the - provided text contents. It will not use a temporary - file as an indirection in that case. If the editor supports - editing multiple files at once, a sequence of files may be - passed as well. Invoke `click.file` once per file instead - if multiple files cannot be managed at once or editing the - files serially is desired. - - .. versionchanged:: 8.2.0 - ``filename`` now accepts any ``Iterable[str]`` in addition to a ``str`` - if the ``editor`` supports editing multiple files at once. - - """ - from ._termui_impl import Editor - - ed = Editor(editor=editor, env=env, require_save=require_save, extension=extension) - - if filename is None: - return ed.edit(text) - - if isinstance(filename, str): - filename = (filename,) - - ed.edit_files(filenames=filename) - return None - - -def launch(url: str, wait: bool = False, locate: bool = False) -> int: - """This function launches the given URL (or filename) in the default - viewer application for this file type. If this is an executable, it - might launch the executable in a new session. The return value is - the exit code of the launched application. Usually, ``0`` indicates - success. - - Examples:: - - click.launch('https://click.palletsprojects.com/') - click.launch('/my/downloaded/file', locate=True) - - .. versionadded:: 2.0 - - :param url: URL or filename of the thing to launch. - :param wait: Wait for the program to exit before returning. This - only works if the launched program blocks. In particular, - ``xdg-open`` on Linux does not block. - :param locate: if this is set to `True` then instead of launching the - application associated with the URL it will attempt to - launch a file manager with the file located. This - might have weird effects if the URL does not point to - the filesystem. - """ - from ._termui_impl import open_url - - return open_url(url, wait=wait, locate=locate) - - -# If this is provided, getchar() calls into this instead. This is used -# for unittesting purposes. -_getchar: t.Callable[[bool], str] | None = None - - -def getchar(echo: bool = False) -> str: - """Fetches a single character from the terminal and returns it. This - will always return a unicode character and under certain rare - circumstances this might return more than one character. The - situations which more than one character is returned is when for - whatever reason multiple characters end up in the terminal buffer or - standard input was not actually a terminal. - - Note that this will always read from the terminal, even if something - is piped into the standard input. - - Note for Windows: in rare cases when typing non-ASCII characters, this - function might wait for a second character and then return both at once. - This is because certain Unicode characters look like special-key markers. - - .. versionadded:: 2.0 - - :param echo: if set to `True`, the character read will also show up on - the terminal. The default is to not show it. - """ - global _getchar - - if _getchar is None: - from ._termui_impl import getchar as f - - _getchar = f - - return _getchar(echo) - - -def raw_terminal() -> AbstractContextManager[int]: - from ._termui_impl import raw_terminal as f - - return f() - - -def pause(info: str | None = None, err: bool = False) -> None: - """This command stops execution and waits for the user to press any - key to continue. This is similar to the Windows batch "pause" - command. If the program is not run through a terminal, this command - will instead do nothing. - - .. versionadded:: 2.0 - - .. versionadded:: 4.0 - Added the `err` parameter. - - :param info: The message to print before pausing. Defaults to - ``"Press any key to continue..."``. - :param err: if set to message goes to ``stderr`` instead of - ``stdout``, the same as with echo. - """ - if not isatty(sys.stdin) or not isatty(sys.stdout): - return - - if info is None: - info = _("Press any key to continue...") - - try: - if info: - echo(info, nl=False, err=err) - try: - getchar() - except (KeyboardInterrupt, EOFError): - pass - finally: - if info: - echo(err=err) diff --git a/.env/Lib/site-packages/click/testing.py b/.env/Lib/site-packages/click/testing.py deleted file mode 100644 index f6f60b8..0000000 --- a/.env/Lib/site-packages/click/testing.py +++ /dev/null @@ -1,577 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import contextlib -import io -import os -import shlex -import sys -import tempfile -import typing as t -from types import TracebackType - -from . import _compat -from . import formatting -from . import termui -from . import utils -from ._compat import _find_binary_reader - -if t.TYPE_CHECKING: - from _typeshed import ReadableBuffer - - from .core import Command - - -class EchoingStdin: - def __init__(self, input: t.BinaryIO, output: t.BinaryIO) -> None: - self._input = input - self._output = output - self._paused = False - - def __getattr__(self, x: str) -> t.Any: - return getattr(self._input, x) - - def _echo(self, rv: bytes) -> bytes: - if not self._paused: - self._output.write(rv) - - return rv - - def read(self, n: int = -1) -> bytes: - return self._echo(self._input.read(n)) - - def read1(self, n: int = -1) -> bytes: - return self._echo(self._input.read1(n)) # type: ignore - - def readline(self, n: int = -1) -> bytes: - return self._echo(self._input.readline(n)) - - def readlines(self) -> list[bytes]: - return [self._echo(x) for x in self._input.readlines()] - - def __iter__(self) -> cabc.Iterator[bytes]: - return iter(self._echo(x) for x in self._input) - - def __repr__(self) -> str: - return repr(self._input) - - -@contextlib.contextmanager -def _pause_echo(stream: EchoingStdin | None) -> cabc.Iterator[None]: - if stream is None: - yield - else: - stream._paused = True - yield - stream._paused = False - - -class BytesIOCopy(io.BytesIO): - """Patch ``io.BytesIO`` to let the written stream be copied to another. - - .. versionadded:: 8.2 - """ - - def __init__(self, copy_to: io.BytesIO) -> None: - super().__init__() - self.copy_to = copy_to - - def flush(self) -> None: - super().flush() - self.copy_to.flush() - - def write(self, b: ReadableBuffer) -> int: - self.copy_to.write(b) - return super().write(b) - - -class StreamMixer: - """Mixes `` and `` streams. - - The result is available in the ``output`` attribute. - - .. versionadded:: 8.2 - """ - - def __init__(self) -> None: - self.output: io.BytesIO = io.BytesIO() - self.stdout: io.BytesIO = BytesIOCopy(copy_to=self.output) - self.stderr: io.BytesIO = BytesIOCopy(copy_to=self.output) - - def __del__(self) -> None: - """ - Guarantee that embedded file-like objects are closed in a - predictable order, protecting against races between - self.output being closed and other streams being flushed on close - - .. versionadded:: 8.2.2 - """ - self.stderr.close() - self.stdout.close() - self.output.close() - - -class _NamedTextIOWrapper(io.TextIOWrapper): - def __init__( - self, buffer: t.BinaryIO, name: str, mode: str, **kwargs: t.Any - ) -> None: - super().__init__(buffer, **kwargs) - self._name = name - self._mode = mode - - @property - def name(self) -> str: - return self._name - - @property - def mode(self) -> str: - return self._mode - - -def make_input_stream( - input: str | bytes | t.IO[t.Any] | None, charset: str -) -> t.BinaryIO: - # Is already an input stream. - if hasattr(input, "read"): - rv = _find_binary_reader(t.cast("t.IO[t.Any]", input)) - - if rv is not None: - return rv - - raise TypeError("Could not find binary reader for input stream.") - - if input is None: - input = b"" - elif isinstance(input, str): - input = input.encode(charset) - - return io.BytesIO(input) - - -class Result: - """Holds the captured result of an invoked CLI script. - - :param runner: The runner that created the result - :param stdout_bytes: The standard output as bytes. - :param stderr_bytes: The standard error as bytes. - :param output_bytes: A mix of ``stdout_bytes`` and ``stderr_bytes``, as the - user would see it in its terminal. - :param return_value: The value returned from the invoked command. - :param exit_code: The exit code as integer. - :param exception: The exception that happened if one did. - :param exc_info: Exception information (exception type, exception instance, - traceback type). - - .. versionchanged:: 8.2 - ``stderr_bytes`` no longer optional, ``output_bytes`` introduced and - ``mix_stderr`` has been removed. - - .. versionadded:: 8.0 - Added ``return_value``. - """ - - def __init__( - self, - runner: CliRunner, - stdout_bytes: bytes, - stderr_bytes: bytes, - output_bytes: bytes, - return_value: t.Any, - exit_code: int, - exception: BaseException | None, - exc_info: tuple[type[BaseException], BaseException, TracebackType] - | None = None, - ): - self.runner = runner - self.stdout_bytes = stdout_bytes - self.stderr_bytes = stderr_bytes - self.output_bytes = output_bytes - self.return_value = return_value - self.exit_code = exit_code - self.exception = exception - self.exc_info = exc_info - - @property - def output(self) -> str: - """The terminal output as unicode string, as the user would see it. - - .. versionchanged:: 8.2 - No longer a proxy for ``self.stdout``. Now has its own independent stream - that is mixing `` and ``, in the order they were written. - """ - return self.output_bytes.decode(self.runner.charset, "replace").replace( - "\r\n", "\n" - ) - - @property - def stdout(self) -> str: - """The standard output as unicode string.""" - return self.stdout_bytes.decode(self.runner.charset, "replace").replace( - "\r\n", "\n" - ) - - @property - def stderr(self) -> str: - """The standard error as unicode string. - - .. versionchanged:: 8.2 - No longer raise an exception, always returns the `` string. - """ - return self.stderr_bytes.decode(self.runner.charset, "replace").replace( - "\r\n", "\n" - ) - - def __repr__(self) -> str: - exc_str = repr(self.exception) if self.exception else "okay" - return f"<{type(self).__name__} {exc_str}>" - - -class CliRunner: - """The CLI runner provides functionality to invoke a Click command line - script for unittesting purposes in a isolated environment. This only - works in single-threaded systems without any concurrency as it changes the - global interpreter state. - - :param charset: the character set for the input and output data. - :param env: a dictionary with environment variables for overriding. - :param echo_stdin: if this is set to `True`, then reading from `` writes - to ``. This is useful for showing examples in - some circumstances. Note that regular prompts - will automatically echo the input. - :param catch_exceptions: Whether to catch any exceptions other than - ``SystemExit`` when running :meth:`~CliRunner.invoke`. - - .. versionchanged:: 8.2 - Added the ``catch_exceptions`` parameter. - - .. versionchanged:: 8.2 - ``mix_stderr`` parameter has been removed. - """ - - def __init__( - self, - charset: str = "utf-8", - env: cabc.Mapping[str, str | None] | None = None, - echo_stdin: bool = False, - catch_exceptions: bool = True, - ) -> None: - self.charset = charset - self.env: cabc.Mapping[str, str | None] = env or {} - self.echo_stdin = echo_stdin - self.catch_exceptions = catch_exceptions - - def get_default_prog_name(self, cli: Command) -> str: - """Given a command object it will return the default program name - for it. The default is the `name` attribute or ``"root"`` if not - set. - """ - return cli.name or "root" - - def make_env( - self, overrides: cabc.Mapping[str, str | None] | None = None - ) -> cabc.Mapping[str, str | None]: - """Returns the environment overrides for invoking a script.""" - rv = dict(self.env) - if overrides: - rv.update(overrides) - return rv - - @contextlib.contextmanager - def isolation( - self, - input: str | bytes | t.IO[t.Any] | None = None, - env: cabc.Mapping[str, str | None] | None = None, - color: bool = False, - ) -> cabc.Iterator[tuple[io.BytesIO, io.BytesIO, io.BytesIO]]: - """A context manager that sets up the isolation for invoking of a - command line tool. This sets up `` with the given input data - and `os.environ` with the overrides from the given dictionary. - This also rebinds some internals in Click to be mocked (like the - prompt functionality). - - This is automatically done in the :meth:`invoke` method. - - :param input: the input stream to put into `sys.stdin`. - :param env: the environment overrides as dictionary. - :param color: whether the output should contain color codes. The - application can still override this explicitly. - - .. versionadded:: 8.2 - An additional output stream is returned, which is a mix of - `` and `` streams. - - .. versionchanged:: 8.2 - Always returns the `` stream. - - .. versionchanged:: 8.0 - `` is opened with ``errors="backslashreplace"`` - instead of the default ``"strict"``. - - .. versionchanged:: 4.0 - Added the ``color`` parameter. - """ - bytes_input = make_input_stream(input, self.charset) - echo_input = None - - old_stdin = sys.stdin - old_stdout = sys.stdout - old_stderr = sys.stderr - old_forced_width = formatting.FORCED_WIDTH - formatting.FORCED_WIDTH = 80 - - env = self.make_env(env) - - stream_mixer = StreamMixer() - - if self.echo_stdin: - bytes_input = echo_input = t.cast( - t.BinaryIO, EchoingStdin(bytes_input, stream_mixer.stdout) - ) - - sys.stdin = text_input = _NamedTextIOWrapper( - bytes_input, encoding=self.charset, name="", mode="r" - ) - - if self.echo_stdin: - # Force unbuffered reads, otherwise TextIOWrapper reads a - # large chunk which is echoed early. - text_input._CHUNK_SIZE = 1 # type: ignore - - sys.stdout = _NamedTextIOWrapper( - stream_mixer.stdout, encoding=self.charset, name="", mode="w" - ) - - sys.stderr = _NamedTextIOWrapper( - stream_mixer.stderr, - encoding=self.charset, - name="", - mode="w", - errors="backslashreplace", - ) - - @_pause_echo(echo_input) # type: ignore - def visible_input(prompt: str | None = None) -> str: - sys.stdout.write(prompt or "") - try: - val = next(text_input).rstrip("\r\n") - except StopIteration as e: - raise EOFError() from e - sys.stdout.write(f"{val}\n") - sys.stdout.flush() - return val - - @_pause_echo(echo_input) # type: ignore - def hidden_input(prompt: str | None = None) -> str: - sys.stdout.write(f"{prompt or ''}\n") - sys.stdout.flush() - try: - return next(text_input).rstrip("\r\n") - except StopIteration as e: - raise EOFError() from e - - @_pause_echo(echo_input) # type: ignore - def _getchar(echo: bool) -> str: - char = sys.stdin.read(1) - - if echo: - sys.stdout.write(char) - - sys.stdout.flush() - return char - - default_color = color - - def should_strip_ansi( - stream: t.IO[t.Any] | None = None, color: bool | None = None - ) -> bool: - if color is None: - return not default_color - return not color - - old_visible_prompt_func = termui.visible_prompt_func - old_hidden_prompt_func = termui.hidden_prompt_func - old__getchar_func = termui._getchar - old_should_strip_ansi = utils.should_strip_ansi # type: ignore - old__compat_should_strip_ansi = _compat.should_strip_ansi - termui.visible_prompt_func = visible_input - termui.hidden_prompt_func = hidden_input - termui._getchar = _getchar - utils.should_strip_ansi = should_strip_ansi # type: ignore - _compat.should_strip_ansi = should_strip_ansi - - old_env = {} - try: - for key, value in env.items(): - old_env[key] = os.environ.get(key) - if value is None: - try: - del os.environ[key] - except Exception: - pass - else: - os.environ[key] = value - yield (stream_mixer.stdout, stream_mixer.stderr, stream_mixer.output) - finally: - for key, value in old_env.items(): - if value is None: - try: - del os.environ[key] - except Exception: - pass - else: - os.environ[key] = value - sys.stdout = old_stdout - sys.stderr = old_stderr - sys.stdin = old_stdin - termui.visible_prompt_func = old_visible_prompt_func - termui.hidden_prompt_func = old_hidden_prompt_func - termui._getchar = old__getchar_func - utils.should_strip_ansi = old_should_strip_ansi # type: ignore - _compat.should_strip_ansi = old__compat_should_strip_ansi - formatting.FORCED_WIDTH = old_forced_width - - def invoke( - self, - cli: Command, - args: str | cabc.Sequence[str] | None = None, - input: str | bytes | t.IO[t.Any] | None = None, - env: cabc.Mapping[str, str | None] | None = None, - catch_exceptions: bool | None = None, - color: bool = False, - **extra: t.Any, - ) -> Result: - """Invokes a command in an isolated environment. The arguments are - forwarded directly to the command line script, the `extra` keyword - arguments are passed to the :meth:`~clickpkg.Command.main` function of - the command. - - This returns a :class:`Result` object. - - :param cli: the command to invoke - :param args: the arguments to invoke. It may be given as an iterable - or a string. When given as string it will be interpreted - as a Unix shell command. More details at - :func:`shlex.split`. - :param input: the input data for `sys.stdin`. - :param env: the environment overrides. - :param catch_exceptions: Whether to catch any other exceptions than - ``SystemExit``. If :data:`None`, the value - from :class:`CliRunner` is used. - :param extra: the keyword arguments to pass to :meth:`main`. - :param color: whether the output should contain color codes. The - application can still override this explicitly. - - .. versionadded:: 8.2 - The result object has the ``output_bytes`` attribute with - the mix of ``stdout_bytes`` and ``stderr_bytes``, as the user would - see it in its terminal. - - .. versionchanged:: 8.2 - The result object always returns the ``stderr_bytes`` stream. - - .. versionchanged:: 8.0 - The result object has the ``return_value`` attribute with - the value returned from the invoked command. - - .. versionchanged:: 4.0 - Added the ``color`` parameter. - - .. versionchanged:: 3.0 - Added the ``catch_exceptions`` parameter. - - .. versionchanged:: 3.0 - The result object has the ``exc_info`` attribute with the - traceback if available. - """ - exc_info = None - if catch_exceptions is None: - catch_exceptions = self.catch_exceptions - - with self.isolation(input=input, env=env, color=color) as outstreams: - return_value = None - exception: BaseException | None = None - exit_code = 0 - - if isinstance(args, str): - args = shlex.split(args) - - try: - prog_name = extra.pop("prog_name") - except KeyError: - prog_name = self.get_default_prog_name(cli) - - try: - return_value = cli.main(args=args or (), prog_name=prog_name, **extra) - except SystemExit as e: - exc_info = sys.exc_info() - e_code = t.cast("int | t.Any | None", e.code) - - if e_code is None: - e_code = 0 - - if e_code != 0: - exception = e - - if not isinstance(e_code, int): - sys.stdout.write(str(e_code)) - sys.stdout.write("\n") - e_code = 1 - - exit_code = e_code - - except Exception as e: - if not catch_exceptions: - raise - exception = e - exit_code = 1 - exc_info = sys.exc_info() - finally: - sys.stdout.flush() - sys.stderr.flush() - stdout = outstreams[0].getvalue() - stderr = outstreams[1].getvalue() - output = outstreams[2].getvalue() - - return Result( - runner=self, - stdout_bytes=stdout, - stderr_bytes=stderr, - output_bytes=output, - return_value=return_value, - exit_code=exit_code, - exception=exception, - exc_info=exc_info, # type: ignore - ) - - @contextlib.contextmanager - def isolated_filesystem( - self, temp_dir: str | os.PathLike[str] | None = None - ) -> cabc.Iterator[str]: - """A context manager that creates a temporary directory and - changes the current working directory to it. This isolates tests - that affect the contents of the CWD to prevent them from - interfering with each other. - - :param temp_dir: Create the temporary directory under this - directory. If given, the created directory is not removed - when exiting. - - .. versionchanged:: 8.0 - Added the ``temp_dir`` parameter. - """ - cwd = os.getcwd() - dt = tempfile.mkdtemp(dir=temp_dir) - os.chdir(dt) - - try: - yield dt - finally: - os.chdir(cwd) - - if temp_dir is None: - import shutil - - try: - shutil.rmtree(dt) - except OSError: - pass diff --git a/.env/Lib/site-packages/click/types.py b/.env/Lib/site-packages/click/types.py deleted file mode 100644 index e71c1c2..0000000 --- a/.env/Lib/site-packages/click/types.py +++ /dev/null @@ -1,1209 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import enum -import os -import stat -import sys -import typing as t -from datetime import datetime -from gettext import gettext as _ -from gettext import ngettext - -from ._compat import _get_argv_encoding -from ._compat import open_stream -from .exceptions import BadParameter -from .utils import format_filename -from .utils import LazyFile -from .utils import safecall - -if t.TYPE_CHECKING: - import typing_extensions as te - - from .core import Context - from .core import Parameter - from .shell_completion import CompletionItem - -ParamTypeValue = t.TypeVar("ParamTypeValue") - - -class ParamType: - """Represents the type of a parameter. Validates and converts values - from the command line or Python into the correct type. - - To implement a custom type, subclass and implement at least the - following: - - - The :attr:`name` class attribute must be set. - - Calling an instance of the type with ``None`` must return - ``None``. This is already implemented by default. - - :meth:`convert` must convert string values to the correct type. - - :meth:`convert` must accept values that are already the correct - type. - - It must be able to convert a value if the ``ctx`` and ``param`` - arguments are ``None``. This can occur when converting prompt - input. - """ - - is_composite: t.ClassVar[bool] = False - arity: t.ClassVar[int] = 1 - - #: the descriptive name of this type - name: str - - #: if a list of this type is expected and the value is pulled from a - #: string environment variable, this is what splits it up. `None` - #: means any whitespace. For all parameters the general rule is that - #: whitespace splits them up. The exception are paths and files which - #: are split by ``os.path.pathsep`` by default (":" on Unix and ";" on - #: Windows). - envvar_list_splitter: t.ClassVar[str | None] = None - - def to_info_dict(self) -> dict[str, t.Any]: - """Gather information that could be useful for a tool generating - user-facing documentation. - - Use :meth:`click.Context.to_info_dict` to traverse the entire - CLI structure. - - .. versionadded:: 8.0 - """ - # The class name without the "ParamType" suffix. - param_type = type(self).__name__.partition("ParamType")[0] - param_type = param_type.partition("ParameterType")[0] - - # Custom subclasses might not remember to set a name. - if hasattr(self, "name"): - name = self.name - else: - name = param_type - - return {"param_type": param_type, "name": name} - - def __call__( - self, - value: t.Any, - param: Parameter | None = None, - ctx: Context | None = None, - ) -> t.Any: - if value is not None: - return self.convert(value, param, ctx) - - def get_metavar(self, param: Parameter, ctx: Context) -> str | None: - """Returns the metavar default for this param if it provides one.""" - - def get_missing_message(self, param: Parameter, ctx: Context | None) -> str | None: - """Optionally might return extra information about a missing - parameter. - - .. versionadded:: 2.0 - """ - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - """Convert the value to the correct type. This is not called if - the value is ``None`` (the missing value). - - This must accept string values from the command line, as well as - values that are already the correct type. It may also convert - other compatible types. - - The ``param`` and ``ctx`` arguments may be ``None`` in certain - situations, such as when converting prompt input. - - If the value cannot be converted, call :meth:`fail` with a - descriptive message. - - :param value: The value to convert. - :param param: The parameter that is using this type to convert - its value. May be ``None``. - :param ctx: The current context that arrived at this value. May - be ``None``. - """ - return value - - def split_envvar_value(self, rv: str) -> cabc.Sequence[str]: - """Given a value from an environment variable this splits it up - into small chunks depending on the defined envvar list splitter. - - If the splitter is set to `None`, which means that whitespace splits, - then leading and trailing whitespace is ignored. Otherwise, leading - and trailing splitters usually lead to empty items being included. - """ - return (rv or "").split(self.envvar_list_splitter) - - def fail( - self, - message: str, - param: Parameter | None = None, - ctx: Context | None = None, - ) -> t.NoReturn: - """Helper method to fail with an invalid value message.""" - raise BadParameter(message, ctx=ctx, param=param) - - def shell_complete( - self, ctx: Context, param: Parameter, incomplete: str - ) -> list[CompletionItem]: - """Return a list of - :class:`~click.shell_completion.CompletionItem` objects for the - incomplete value. Most types do not provide completions, but - some do, and this allows custom types to provide custom - completions as well. - - :param ctx: Invocation context for this command. - :param param: The parameter that is requesting completion. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - return [] - - -class CompositeParamType(ParamType): - is_composite = True - - @property - def arity(self) -> int: # type: ignore - raise NotImplementedError() - - -class FuncParamType(ParamType): - def __init__(self, func: t.Callable[[t.Any], t.Any]) -> None: - self.name: str = func.__name__ - self.func = func - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict["func"] = self.func - return info_dict - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - try: - return self.func(value) - except ValueError: - try: - value = str(value) - except UnicodeError: - value = value.decode("utf-8", "replace") - - self.fail(value, param, ctx) - - -class UnprocessedParamType(ParamType): - name = "text" - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - return value - - def __repr__(self) -> str: - return "UNPROCESSED" - - -class StringParamType(ParamType): - name = "text" - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - if isinstance(value, bytes): - enc = _get_argv_encoding() - try: - value = value.decode(enc) - except UnicodeError: - fs_enc = sys.getfilesystemencoding() - if fs_enc != enc: - try: - value = value.decode(fs_enc) - except UnicodeError: - value = value.decode("utf-8", "replace") - else: - value = value.decode("utf-8", "replace") - return value - return str(value) - - def __repr__(self) -> str: - return "STRING" - - -class Choice(ParamType, t.Generic[ParamTypeValue]): - """The choice type allows a value to be checked against a fixed set - of supported values. - - You may pass any iterable value which will be converted to a tuple - and thus will only be iterated once. - - The resulting value will always be one of the originally passed choices. - See :meth:`normalize_choice` for more info on the mapping of strings - to choices. See :ref:`choice-opts` for an example. - - :param case_sensitive: Set to false to make choices case - insensitive. Defaults to true. - - .. versionchanged:: 8.2.0 - Non-``str`` ``choices`` are now supported. It can additionally be any - iterable. Before you were not recommended to pass anything but a list or - tuple. - - .. versionadded:: 8.2.0 - Choice normalization can be overridden via :meth:`normalize_choice`. - """ - - name = "choice" - - def __init__( - self, choices: cabc.Iterable[ParamTypeValue], case_sensitive: bool = True - ) -> None: - self.choices: cabc.Sequence[ParamTypeValue] = tuple(choices) - self.case_sensitive = case_sensitive - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict["choices"] = self.choices - info_dict["case_sensitive"] = self.case_sensitive - return info_dict - - def _normalized_mapping( - self, ctx: Context | None = None - ) -> cabc.Mapping[ParamTypeValue, str]: - """ - Returns mapping where keys are the original choices and the values are - the normalized values that are accepted via the command line. - - This is a simple wrapper around :meth:`normalize_choice`, use that - instead which is supported. - """ - return { - choice: self.normalize_choice( - choice=choice, - ctx=ctx, - ) - for choice in self.choices - } - - def normalize_choice(self, choice: ParamTypeValue, ctx: Context | None) -> str: - """ - Normalize a choice value, used to map a passed string to a choice. - Each choice must have a unique normalized value. - - By default uses :meth:`Context.token_normalize_func` and if not case - sensitive, convert it to a casefolded value. - - .. versionadded:: 8.2.0 - """ - normed_value = choice.name if isinstance(choice, enum.Enum) else str(choice) - - if ctx is not None and ctx.token_normalize_func is not None: - normed_value = ctx.token_normalize_func(normed_value) - - if not self.case_sensitive: - normed_value = normed_value.casefold() - - return normed_value - - def get_metavar(self, param: Parameter, ctx: Context) -> str | None: - if param.param_type_name == "option" and not param.show_choices: # type: ignore - choice_metavars = [ - convert_type(type(choice)).name.upper() for choice in self.choices - ] - choices_str = "|".join([*dict.fromkeys(choice_metavars)]) - else: - choices_str = "|".join( - [str(i) for i in self._normalized_mapping(ctx=ctx).values()] - ) - - # Use curly braces to indicate a required argument. - if param.required and param.param_type_name == "argument": - return f"{{{choices_str}}}" - - # Use square braces to indicate an option or optional argument. - return f"[{choices_str}]" - - def get_missing_message(self, param: Parameter, ctx: Context | None) -> str: - """ - Message shown when no choice is passed. - - .. versionchanged:: 8.2.0 Added ``ctx`` argument. - """ - return _("Choose from:\n\t{choices}").format( - choices=",\n\t".join(self._normalized_mapping(ctx=ctx).values()) - ) - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> ParamTypeValue: - """ - For a given value from the parser, normalize it and find its - matching normalized value in the list of choices. Then return the - matched "original" choice. - """ - normed_value = self.normalize_choice(choice=value, ctx=ctx) - normalized_mapping = self._normalized_mapping(ctx=ctx) - - try: - return next( - original - for original, normalized in normalized_mapping.items() - if normalized == normed_value - ) - except StopIteration: - self.fail( - self.get_invalid_choice_message(value=value, ctx=ctx), - param=param, - ctx=ctx, - ) - - def get_invalid_choice_message(self, value: t.Any, ctx: Context | None) -> str: - """Get the error message when the given choice is invalid. - - :param value: The invalid value. - - .. versionadded:: 8.2 - """ - choices_str = ", ".join(map(repr, self._normalized_mapping(ctx=ctx).values())) - return ngettext( - "{value!r} is not {choice}.", - "{value!r} is not one of {choices}.", - len(self.choices), - ).format(value=value, choice=choices_str, choices=choices_str) - - def __repr__(self) -> str: - return f"Choice({list(self.choices)})" - - def shell_complete( - self, ctx: Context, param: Parameter, incomplete: str - ) -> list[CompletionItem]: - """Complete choices that start with the incomplete value. - - :param ctx: Invocation context for this command. - :param param: The parameter that is requesting completion. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - from click.shell_completion import CompletionItem - - str_choices = map(str, self.choices) - - if self.case_sensitive: - matched = (c for c in str_choices if c.startswith(incomplete)) - else: - incomplete = incomplete.lower() - matched = (c for c in str_choices if c.lower().startswith(incomplete)) - - return [CompletionItem(c) for c in matched] - - -class DateTime(ParamType): - """The DateTime type converts date strings into `datetime` objects. - - The format strings which are checked are configurable, but default to some - common (non-timezone aware) ISO 8601 formats. - - When specifying *DateTime* formats, you should only pass a list or a tuple. - Other iterables, like generators, may lead to surprising results. - - The format strings are processed using ``datetime.strptime``, and this - consequently defines the format strings which are allowed. - - Parsing is tried using each format, in order, and the first format which - parses successfully is used. - - :param formats: A list or tuple of date format strings, in the order in - which they should be tried. Defaults to - ``'%Y-%m-%d'``, ``'%Y-%m-%dT%H:%M:%S'``, - ``'%Y-%m-%d %H:%M:%S'``. - """ - - name = "datetime" - - def __init__(self, formats: cabc.Sequence[str] | None = None): - self.formats: cabc.Sequence[str] = formats or [ - "%Y-%m-%d", - "%Y-%m-%dT%H:%M:%S", - "%Y-%m-%d %H:%M:%S", - ] - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict["formats"] = self.formats - return info_dict - - def get_metavar(self, param: Parameter, ctx: Context) -> str | None: - return f"[{'|'.join(self.formats)}]" - - def _try_to_convert_date(self, value: t.Any, format: str) -> datetime | None: - try: - return datetime.strptime(value, format) - except ValueError: - return None - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - if isinstance(value, datetime): - return value - - for format in self.formats: - converted = self._try_to_convert_date(value, format) - - if converted is not None: - return converted - - formats_str = ", ".join(map(repr, self.formats)) - self.fail( - ngettext( - "{value!r} does not match the format {format}.", - "{value!r} does not match the formats {formats}.", - len(self.formats), - ).format(value=value, format=formats_str, formats=formats_str), - param, - ctx, - ) - - def __repr__(self) -> str: - return "DateTime" - - -class _NumberParamTypeBase(ParamType): - _number_class: t.ClassVar[type[t.Any]] - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - try: - return self._number_class(value) - except ValueError: - self.fail( - _("{value!r} is not a valid {number_type}.").format( - value=value, number_type=self.name - ), - param, - ctx, - ) - - -class _NumberRangeBase(_NumberParamTypeBase): - def __init__( - self, - min: float | None = None, - max: float | None = None, - min_open: bool = False, - max_open: bool = False, - clamp: bool = False, - ) -> None: - self.min = min - self.max = max - self.min_open = min_open - self.max_open = max_open - self.clamp = clamp - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict.update( - min=self.min, - max=self.max, - min_open=self.min_open, - max_open=self.max_open, - clamp=self.clamp, - ) - return info_dict - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - import operator - - rv = super().convert(value, param, ctx) - lt_min: bool = self.min is not None and ( - operator.le if self.min_open else operator.lt - )(rv, self.min) - gt_max: bool = self.max is not None and ( - operator.ge if self.max_open else operator.gt - )(rv, self.max) - - if self.clamp: - if lt_min: - return self._clamp(self.min, 1, self.min_open) # type: ignore - - if gt_max: - return self._clamp(self.max, -1, self.max_open) # type: ignore - - if lt_min or gt_max: - self.fail( - _("{value} is not in the range {range}.").format( - value=rv, range=self._describe_range() - ), - param, - ctx, - ) - - return rv - - def _clamp(self, bound: float, dir: t.Literal[1, -1], open: bool) -> float: - """Find the valid value to clamp to bound in the given - direction. - - :param bound: The boundary value. - :param dir: 1 or -1 indicating the direction to move. - :param open: If true, the range does not include the bound. - """ - raise NotImplementedError - - def _describe_range(self) -> str: - """Describe the range for use in help text.""" - if self.min is None: - op = "<" if self.max_open else "<=" - return f"x{op}{self.max}" - - if self.max is None: - op = ">" if self.min_open else ">=" - return f"x{op}{self.min}" - - lop = "<" if self.min_open else "<=" - rop = "<" if self.max_open else "<=" - return f"{self.min}{lop}x{rop}{self.max}" - - def __repr__(self) -> str: - clamp = " clamped" if self.clamp else "" - return f"<{type(self).__name__} {self._describe_range()}{clamp}>" - - -class IntParamType(_NumberParamTypeBase): - name = "integer" - _number_class = int - - def __repr__(self) -> str: - return "INT" - - -class IntRange(_NumberRangeBase, IntParamType): - """Restrict an :data:`click.INT` value to a range of accepted - values. See :ref:`ranges`. - - If ``min`` or ``max`` are not passed, any value is accepted in that - direction. If ``min_open`` or ``max_open`` are enabled, the - corresponding boundary is not included in the range. - - If ``clamp`` is enabled, a value outside the range is clamped to the - boundary instead of failing. - - .. versionchanged:: 8.0 - Added the ``min_open`` and ``max_open`` parameters. - """ - - name = "integer range" - - def _clamp( # type: ignore - self, bound: int, dir: t.Literal[1, -1], open: bool - ) -> int: - if not open: - return bound - - return bound + dir - - -class FloatParamType(_NumberParamTypeBase): - name = "float" - _number_class = float - - def __repr__(self) -> str: - return "FLOAT" - - -class FloatRange(_NumberRangeBase, FloatParamType): - """Restrict a :data:`click.FLOAT` value to a range of accepted - values. See :ref:`ranges`. - - If ``min`` or ``max`` are not passed, any value is accepted in that - direction. If ``min_open`` or ``max_open`` are enabled, the - corresponding boundary is not included in the range. - - If ``clamp`` is enabled, a value outside the range is clamped to the - boundary instead of failing. This is not supported if either - boundary is marked ``open``. - - .. versionchanged:: 8.0 - Added the ``min_open`` and ``max_open`` parameters. - """ - - name = "float range" - - def __init__( - self, - min: float | None = None, - max: float | None = None, - min_open: bool = False, - max_open: bool = False, - clamp: bool = False, - ) -> None: - super().__init__( - min=min, max=max, min_open=min_open, max_open=max_open, clamp=clamp - ) - - if (min_open or max_open) and clamp: - raise TypeError("Clamping is not supported for open bounds.") - - def _clamp(self, bound: float, dir: t.Literal[1, -1], open: bool) -> float: - if not open: - return bound - - # Could use math.nextafter here, but clamping an - # open float range doesn't seem to be particularly useful. It's - # left up to the user to write a callback to do it if needed. - raise RuntimeError("Clamping is not supported for open bounds.") - - -class BoolParamType(ParamType): - name = "boolean" - - bool_states: dict[str, bool] = { - "1": True, - "0": False, - "yes": True, - "no": False, - "true": True, - "false": False, - "on": True, - "off": False, - "t": True, - "f": False, - "y": True, - "n": False, - # Absence of value is considered False. - "": False, - } - """A mapping of string values to boolean states. - - Mapping is inspired by :py:attr:`configparser.ConfigParser.BOOLEAN_STATES` - and extends it. - - .. caution:: - String values are lower-cased, as the ``str_to_bool`` comparison function - below is case-insensitive. - - .. warning:: - The mapping is not exhaustive, and does not cover all possible boolean strings - representations. It will remains as it is to avoid endless bikeshedding. - - Future work my be considered to make this mapping user-configurable from public - API. - """ - - @staticmethod - def str_to_bool(value: str | bool) -> bool | None: - """Convert a string to a boolean value. - - If the value is already a boolean, it is returned as-is. If the value is a - string, it is stripped of whitespaces and lower-cased, then checked against - the known boolean states pre-defined in the `BoolParamType.bool_states` mapping - above. - - Returns `None` if the value does not match any known boolean state. - """ - if isinstance(value, bool): - return value - return BoolParamType.bool_states.get(value.strip().lower()) - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> bool: - normalized = self.str_to_bool(value) - if normalized is None: - self.fail( - _( - "{value!r} is not a valid boolean. Recognized values: {states}" - ).format(value=value, states=", ".join(sorted(self.bool_states))), - param, - ctx, - ) - return normalized - - def __repr__(self) -> str: - return "BOOL" - - -class UUIDParameterType(ParamType): - name = "uuid" - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - import uuid - - if isinstance(value, uuid.UUID): - return value - - value = value.strip() - - try: - return uuid.UUID(value) - except ValueError: - self.fail( - _("{value!r} is not a valid UUID.").format(value=value), param, ctx - ) - - def __repr__(self) -> str: - return "UUID" - - -class File(ParamType): - """Declares a parameter to be a file for reading or writing. The file - is automatically closed once the context tears down (after the command - finished working). - - Files can be opened for reading or writing. The special value ``-`` - indicates stdin or stdout depending on the mode. - - By default, the file is opened for reading text data, but it can also be - opened in binary mode or for writing. The encoding parameter can be used - to force a specific encoding. - - The `lazy` flag controls if the file should be opened immediately or upon - first IO. The default is to be non-lazy for standard input and output - streams as well as files opened for reading, `lazy` otherwise. When opening a - file lazily for reading, it is still opened temporarily for validation, but - will not be held open until first IO. lazy is mainly useful when opening - for writing to avoid creating the file until it is needed. - - Files can also be opened atomically in which case all writes go into a - separate file in the same folder and upon completion the file will - be moved over to the original location. This is useful if a file - regularly read by other users is modified. - - See :ref:`file-args` for more information. - - .. versionchanged:: 2.0 - Added the ``atomic`` parameter. - """ - - name = "filename" - envvar_list_splitter: t.ClassVar[str] = os.path.pathsep - - def __init__( - self, - mode: str = "r", - encoding: str | None = None, - errors: str | None = "strict", - lazy: bool | None = None, - atomic: bool = False, - ) -> None: - self.mode = mode - self.encoding = encoding - self.errors = errors - self.lazy = lazy - self.atomic = atomic - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict.update(mode=self.mode, encoding=self.encoding) - return info_dict - - def resolve_lazy_flag(self, value: str | os.PathLike[str]) -> bool: - if self.lazy is not None: - return self.lazy - if os.fspath(value) == "-": - return False - elif "w" in self.mode: - return True - return False - - def convert( - self, - value: str | os.PathLike[str] | t.IO[t.Any], - param: Parameter | None, - ctx: Context | None, - ) -> t.IO[t.Any]: - if _is_file_like(value): - return value - - value = t.cast("str | os.PathLike[str]", value) - - try: - lazy = self.resolve_lazy_flag(value) - - if lazy: - lf = LazyFile( - value, self.mode, self.encoding, self.errors, atomic=self.atomic - ) - - if ctx is not None: - ctx.call_on_close(lf.close_intelligently) - - return t.cast("t.IO[t.Any]", lf) - - f, should_close = open_stream( - value, self.mode, self.encoding, self.errors, atomic=self.atomic - ) - - # If a context is provided, we automatically close the file - # at the end of the context execution (or flush out). If a - # context does not exist, it's the caller's responsibility to - # properly close the file. This for instance happens when the - # type is used with prompts. - if ctx is not None: - if should_close: - ctx.call_on_close(safecall(f.close)) - else: - ctx.call_on_close(safecall(f.flush)) - - return f - except OSError as e: - self.fail(f"'{format_filename(value)}': {e.strerror}", param, ctx) - - def shell_complete( - self, ctx: Context, param: Parameter, incomplete: str - ) -> list[CompletionItem]: - """Return a special completion marker that tells the completion - system to use the shell to provide file path completions. - - :param ctx: Invocation context for this command. - :param param: The parameter that is requesting completion. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - from click.shell_completion import CompletionItem - - return [CompletionItem(incomplete, type="file")] - - -def _is_file_like(value: t.Any) -> te.TypeGuard[t.IO[t.Any]]: - return hasattr(value, "read") or hasattr(value, "write") - - -class Path(ParamType): - """The ``Path`` type is similar to the :class:`File` type, but - returns the filename instead of an open file. Various checks can be - enabled to validate the type of file and permissions. - - :param exists: The file or directory needs to exist for the value to - be valid. If this is not set to ``True``, and the file does not - exist, then all further checks are silently skipped. - :param file_okay: Allow a file as a value. - :param dir_okay: Allow a directory as a value. - :param readable: if true, a readable check is performed. - :param writable: if true, a writable check is performed. - :param executable: if true, an executable check is performed. - :param resolve_path: Make the value absolute and resolve any - symlinks. A ``~`` is not expanded, as this is supposed to be - done by the shell only. - :param allow_dash: Allow a single dash as a value, which indicates - a standard stream (but does not open it). Use - :func:`~click.open_file` to handle opening this value. - :param path_type: Convert the incoming path value to this type. If - ``None``, keep Python's default, which is ``str``. Useful to - convert to :class:`pathlib.Path`. - - .. versionchanged:: 8.1 - Added the ``executable`` parameter. - - .. versionchanged:: 8.0 - Allow passing ``path_type=pathlib.Path``. - - .. versionchanged:: 6.0 - Added the ``allow_dash`` parameter. - """ - - envvar_list_splitter: t.ClassVar[str] = os.path.pathsep - - def __init__( - self, - exists: bool = False, - file_okay: bool = True, - dir_okay: bool = True, - writable: bool = False, - readable: bool = True, - resolve_path: bool = False, - allow_dash: bool = False, - path_type: type[t.Any] | None = None, - executable: bool = False, - ): - self.exists = exists - self.file_okay = file_okay - self.dir_okay = dir_okay - self.readable = readable - self.writable = writable - self.executable = executable - self.resolve_path = resolve_path - self.allow_dash = allow_dash - self.type = path_type - - if self.file_okay and not self.dir_okay: - self.name: str = _("file") - elif self.dir_okay and not self.file_okay: - self.name = _("directory") - else: - self.name = _("path") - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict.update( - exists=self.exists, - file_okay=self.file_okay, - dir_okay=self.dir_okay, - writable=self.writable, - readable=self.readable, - allow_dash=self.allow_dash, - ) - return info_dict - - def coerce_path_result( - self, value: str | os.PathLike[str] - ) -> str | bytes | os.PathLike[str]: - if self.type is not None and not isinstance(value, self.type): - if self.type is str: - return os.fsdecode(value) - elif self.type is bytes: - return os.fsencode(value) - else: - return t.cast("os.PathLike[str]", self.type(value)) - - return value - - def convert( - self, - value: str | os.PathLike[str], - param: Parameter | None, - ctx: Context | None, - ) -> str | bytes | os.PathLike[str]: - rv = value - - is_dash = self.file_okay and self.allow_dash and rv in (b"-", "-") - - if not is_dash: - if self.resolve_path: - rv = os.path.realpath(rv) - - try: - st = os.stat(rv) - except OSError: - if not self.exists: - return self.coerce_path_result(rv) - self.fail( - _("{name} {filename!r} does not exist.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - - if not self.file_okay and stat.S_ISREG(st.st_mode): - self.fail( - _("{name} {filename!r} is a file.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - if not self.dir_okay and stat.S_ISDIR(st.st_mode): - self.fail( - _("{name} {filename!r} is a directory.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - - if self.readable and not os.access(rv, os.R_OK): - self.fail( - _("{name} {filename!r} is not readable.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - - if self.writable and not os.access(rv, os.W_OK): - self.fail( - _("{name} {filename!r} is not writable.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - - if self.executable and not os.access(value, os.X_OK): - self.fail( - _("{name} {filename!r} is not executable.").format( - name=self.name.title(), filename=format_filename(value) - ), - param, - ctx, - ) - - return self.coerce_path_result(rv) - - def shell_complete( - self, ctx: Context, param: Parameter, incomplete: str - ) -> list[CompletionItem]: - """Return a special completion marker that tells the completion - system to use the shell to provide path completions for only - directories or any paths. - - :param ctx: Invocation context for this command. - :param param: The parameter that is requesting completion. - :param incomplete: Value being completed. May be empty. - - .. versionadded:: 8.0 - """ - from click.shell_completion import CompletionItem - - type = "dir" if self.dir_okay and not self.file_okay else "file" - return [CompletionItem(incomplete, type=type)] - - -class Tuple(CompositeParamType): - """The default behavior of Click is to apply a type on a value directly. - This works well in most cases, except for when `nargs` is set to a fixed - count and different types should be used for different items. In this - case the :class:`Tuple` type can be used. This type can only be used - if `nargs` is set to a fixed number. - - For more information see :ref:`tuple-type`. - - This can be selected by using a Python tuple literal as a type. - - :param types: a list of types that should be used for the tuple items. - """ - - def __init__(self, types: cabc.Sequence[type[t.Any] | ParamType]) -> None: - self.types: cabc.Sequence[ParamType] = [convert_type(ty) for ty in types] - - def to_info_dict(self) -> dict[str, t.Any]: - info_dict = super().to_info_dict() - info_dict["types"] = [t.to_info_dict() for t in self.types] - return info_dict - - @property - def name(self) -> str: # type: ignore - return f"<{' '.join(ty.name for ty in self.types)}>" - - @property - def arity(self) -> int: # type: ignore - return len(self.types) - - def convert( - self, value: t.Any, param: Parameter | None, ctx: Context | None - ) -> t.Any: - len_type = len(self.types) - len_value = len(value) - - if len_value != len_type: - self.fail( - ngettext( - "{len_type} values are required, but {len_value} was given.", - "{len_type} values are required, but {len_value} were given.", - len_value, - ).format(len_type=len_type, len_value=len_value), - param=param, - ctx=ctx, - ) - - return tuple( - ty(x, param, ctx) for ty, x in zip(self.types, value, strict=False) - ) - - -def convert_type(ty: t.Any | None, default: t.Any | None = None) -> ParamType: - """Find the most appropriate :class:`ParamType` for the given Python - type. If the type isn't provided, it can be inferred from a default - value. - """ - guessed_type = False - - if ty is None and default is not None: - if isinstance(default, (tuple, list)): - # If the default is empty, ty will remain None and will - # return STRING. - if default: - item = default[0] - - # A tuple of tuples needs to detect the inner types. - # Can't call convert recursively because that would - # incorrectly unwind the tuple to a single type. - if isinstance(item, (tuple, list)): - ty = tuple(map(type, item)) - else: - ty = type(item) - else: - ty = type(default) - - guessed_type = True - - if isinstance(ty, tuple): - return Tuple(ty) - - if isinstance(ty, ParamType): - return ty - - if ty is str or ty is None: - return STRING - - if ty is int: - return INT - - if ty is float: - return FLOAT - - if ty is bool: - return BOOL - - if guessed_type: - return STRING - - if __debug__: - try: - if issubclass(ty, ParamType): - raise AssertionError( - f"Attempted to use an uninstantiated parameter type ({ty})." - ) - except TypeError: - # ty is an instance (correct), so issubclass fails. - pass - - return FuncParamType(ty) - - -#: A dummy parameter type that just does nothing. From a user's -#: perspective this appears to just be the same as `STRING` but -#: internally no string conversion takes place if the input was bytes. -#: This is usually useful when working with file paths as they can -#: appear in bytes and unicode. -#: -#: For path related uses the :class:`Path` type is a better choice but -#: there are situations where an unprocessed type is useful which is why -#: it is is provided. -#: -#: .. versionadded:: 4.0 -UNPROCESSED = UnprocessedParamType() - -#: A unicode string parameter type which is the implicit default. This -#: can also be selected by using ``str`` as type. -STRING = StringParamType() - -#: An integer parameter. This can also be selected by using ``int`` as -#: type. -INT = IntParamType() - -#: A floating point value parameter. This can also be selected by using -#: ``float`` as type. -FLOAT = FloatParamType() - -#: A boolean parameter. This is the default for boolean flags. This can -#: also be selected by using ``bool`` as a type. -BOOL = BoolParamType() - -#: A UUID parameter. -UUID = UUIDParameterType() - - -class OptionHelpExtra(t.TypedDict, total=False): - envvars: tuple[str, ...] - default: str - range: str - required: str diff --git a/.env/Lib/site-packages/click/utils.py b/.env/Lib/site-packages/click/utils.py deleted file mode 100644 index beae26f..0000000 --- a/.env/Lib/site-packages/click/utils.py +++ /dev/null @@ -1,627 +0,0 @@ -from __future__ import annotations - -import collections.abc as cabc -import os -import re -import sys -import typing as t -from functools import update_wrapper -from types import ModuleType -from types import TracebackType - -from ._compat import _default_text_stderr -from ._compat import _default_text_stdout -from ._compat import _find_binary_writer -from ._compat import auto_wrap_for_ansi -from ._compat import binary_streams -from ._compat import open_stream -from ._compat import should_strip_ansi -from ._compat import strip_ansi -from ._compat import text_streams -from ._compat import WIN -from .globals import resolve_color_default - -if t.TYPE_CHECKING: - import typing_extensions as te - - P = te.ParamSpec("P") - -R = t.TypeVar("R") - - -def _posixify(name: str) -> str: - return "-".join(name.split()).lower() - - -def safecall(func: t.Callable[P, R]) -> t.Callable[P, R | None]: - """Wraps a function so that it swallows exceptions.""" - - def wrapper(*args: P.args, **kwargs: P.kwargs) -> R | None: - try: - return func(*args, **kwargs) - except Exception: - pass - return None - - return update_wrapper(wrapper, func) - - -def make_str(value: t.Any) -> str: - """Converts a value into a valid string.""" - if isinstance(value, bytes): - try: - return value.decode(sys.getfilesystemencoding()) - except UnicodeError: - return value.decode("utf-8", "replace") - return str(value) - - -def make_default_short_help(help: str, max_length: int = 45) -> str: - """Returns a condensed version of help string.""" - # Consider only the first paragraph. - paragraph_end = help.find("\n\n") - - if paragraph_end != -1: - help = help[:paragraph_end] - - # Collapse newlines, tabs, and spaces. - words = help.split() - - if not words: - return "" - - # The first paragraph started with a "no rewrap" marker, ignore it. - if words[0] == "\b": - words = words[1:] - - total_length = 0 - last_index = len(words) - 1 - - for i, word in enumerate(words): - total_length += len(word) + (i > 0) - - if total_length > max_length: # too long, truncate - break - - if word[-1] == ".": # sentence end, truncate without "..." - return " ".join(words[: i + 1]) - - if total_length == max_length and i != last_index: - break # not at sentence end, truncate with "..." - else: - return " ".join(words) # no truncation needed - - # Account for the length of the suffix. - total_length += len("...") - - # remove words until the length is short enough - while i > 0: - total_length -= len(words[i]) + (i > 0) - - if total_length <= max_length: - break - - i -= 1 - - return " ".join(words[:i]) + "..." - - -class LazyFile: - """A lazy file works like a regular file but it does not fully open - the file but it does perform some basic checks early to see if the - filename parameter does make sense. This is useful for safely opening - files for writing. - """ - - def __init__( - self, - filename: str | os.PathLike[str], - mode: str = "r", - encoding: str | None = None, - errors: str | None = "strict", - atomic: bool = False, - ): - self.name: str = os.fspath(filename) - self.mode = mode - self.encoding = encoding - self.errors = errors - self.atomic = atomic - self._f: t.IO[t.Any] | None - self.should_close: bool - - if self.name == "-": - self._f, self.should_close = open_stream(filename, mode, encoding, errors) - else: - if "r" in mode: - # Open and close the file in case we're opening it for - # reading so that we can catch at least some errors in - # some cases early. - open(filename, mode).close() - self._f = None - self.should_close = True - - def __getattr__(self, name: str) -> t.Any: - return getattr(self.open(), name) - - def __repr__(self) -> str: - if self._f is not None: - return repr(self._f) - return f"" - - def open(self) -> t.IO[t.Any]: - """Opens the file if it's not yet open. This call might fail with - a :exc:`FileError`. Not handling this error will produce an error - that Click shows. - """ - if self._f is not None: - return self._f - try: - rv, self.should_close = open_stream( - self.name, self.mode, self.encoding, self.errors, atomic=self.atomic - ) - except OSError as e: - from .exceptions import FileError - - raise FileError(self.name, hint=e.strerror) from e - self._f = rv - return rv - - def close(self) -> None: - """Closes the underlying file, no matter what.""" - if self._f is not None: - self._f.close() - - def close_intelligently(self) -> None: - """This function only closes the file if it was opened by the lazy - file wrapper. For instance this will never close stdin. - """ - if self.should_close: - self.close() - - def __enter__(self) -> LazyFile: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> None: - self.close_intelligently() - - def __iter__(self) -> cabc.Iterator[t.AnyStr]: - self.open() - return iter(self._f) # type: ignore - - -class KeepOpenFile: - def __init__(self, file: t.IO[t.Any]) -> None: - self._file: t.IO[t.Any] = file - - def __getattr__(self, name: str) -> t.Any: - return getattr(self._file, name) - - def __enter__(self) -> KeepOpenFile: - return self - - def __exit__( - self, - exc_type: type[BaseException] | None, - exc_value: BaseException | None, - tb: TracebackType | None, - ) -> None: - pass - - def __repr__(self) -> str: - return repr(self._file) - - def __iter__(self) -> cabc.Iterator[t.AnyStr]: - return iter(self._file) - - -def echo( - message: t.Any | None = None, - file: t.IO[t.Any] | None = None, - nl: bool = True, - err: bool = False, - color: bool | None = None, -) -> None: - """Print a message and newline to stdout or a file. This should be - used instead of :func:`print` because it provides better support - for different data, files, and environments. - - Compared to :func:`print`, this does the following: - - - Ensures that the output encoding is not misconfigured on Linux. - - Supports Unicode in the Windows console. - - Supports writing to binary outputs, and supports writing bytes - to text outputs. - - Supports colors and styles on Windows. - - Removes ANSI color and style codes if the output does not look - like an interactive terminal. - - Always flushes the output. - - :param message: The string or bytes to output. Other objects are - converted to strings. - :param file: The file to write to. Defaults to ``stdout``. - :param err: Write to ``stderr`` instead of ``stdout``. - :param nl: Print a newline after the message. Enabled by default. - :param color: Force showing or hiding colors and other styles. By - default Click will remove color if the output does not look like - an interactive terminal. - - .. versionchanged:: 6.0 - Support Unicode output on the Windows console. Click does not - modify ``sys.stdout``, so ``sys.stdout.write()`` and ``print()`` - will still not support Unicode. - - .. versionchanged:: 4.0 - Added the ``color`` parameter. - - .. versionadded:: 3.0 - Added the ``err`` parameter. - - .. versionchanged:: 2.0 - Support colors on Windows if colorama is installed. - """ - if file is None: - if err: - file = _default_text_stderr() - else: - file = _default_text_stdout() - - # There are no standard streams attached to write to. For example, - # pythonw on Windows. - if file is None: - return - - # Convert non bytes/text into the native string type. - if message is not None and not isinstance(message, (str, bytes, bytearray)): - out: str | bytes | bytearray | None = str(message) - else: - out = message - - if nl: - out = out or "" - if isinstance(out, str): - out += "\n" - else: - out += b"\n" - - if not out: - file.flush() - return - - # If there is a message and the value looks like bytes, we manually - # need to find the binary stream and write the message in there. - # This is done separately so that most stream types will work as you - # would expect. Eg: you can write to StringIO for other cases. - if isinstance(out, (bytes, bytearray)): - binary_file = _find_binary_writer(file) - - if binary_file is not None: - file.flush() - binary_file.write(out) - binary_file.flush() - return - - # ANSI style code support. For no message or bytes, nothing happens. - # When outputting to a file instead of a terminal, strip codes. - else: - color = resolve_color_default(color) - - if should_strip_ansi(file, color): - out = strip_ansi(out) - elif WIN: - if auto_wrap_for_ansi is not None: - file = auto_wrap_for_ansi(file, color) # type: ignore - elif not color: - out = strip_ansi(out) - - file.write(out) # type: ignore - file.flush() - - -def get_binary_stream(name: t.Literal["stdin", "stdout", "stderr"]) -> t.BinaryIO: - """Returns a system stream for byte processing. - - :param name: the name of the stream to open. Valid names are ``'stdin'``, - ``'stdout'`` and ``'stderr'`` - """ - opener = binary_streams.get(name) - if opener is None: - raise TypeError(f"Unknown standard stream '{name}'") - return opener() - - -def get_text_stream( - name: t.Literal["stdin", "stdout", "stderr"], - encoding: str | None = None, - errors: str | None = "strict", -) -> t.TextIO: - """Returns a system stream for text processing. This usually returns - a wrapped stream around a binary stream returned from - :func:`get_binary_stream` but it also can take shortcuts for already - correctly configured streams. - - :param name: the name of the stream to open. Valid names are ``'stdin'``, - ``'stdout'`` and ``'stderr'`` - :param encoding: overrides the detected default encoding. - :param errors: overrides the default error mode. - """ - opener = text_streams.get(name) - if opener is None: - raise TypeError(f"Unknown standard stream '{name}'") - return opener(encoding, errors) - - -def open_file( - filename: str | os.PathLike[str], - mode: str = "r", - encoding: str | None = None, - errors: str | None = "strict", - lazy: bool = False, - atomic: bool = False, -) -> t.IO[t.Any]: - """Open a file, with extra behavior to handle ``'-'`` to indicate - a standard stream, lazy open on write, and atomic write. Similar to - the behavior of the :class:`~click.File` param type. - - If ``'-'`` is given to open ``stdout`` or ``stdin``, the stream is - wrapped so that using it in a context manager will not close it. - This makes it possible to use the function without accidentally - closing a standard stream: - - .. code-block:: python - - with open_file(filename) as f: - ... - - :param filename: The name or Path of the file to open, or ``'-'`` for - ``stdin``/``stdout``. - :param mode: The mode in which to open the file. - :param encoding: The encoding to decode or encode a file opened in - text mode. - :param errors: The error handling mode. - :param lazy: Wait to open the file until it is accessed. For read - mode, the file is temporarily opened to raise access errors - early, then closed until it is read again. - :param atomic: Write to a temporary file and replace the given file - on close. - - .. versionadded:: 3.0 - """ - if lazy: - return t.cast( - "t.IO[t.Any]", LazyFile(filename, mode, encoding, errors, atomic=atomic) - ) - - f, should_close = open_stream(filename, mode, encoding, errors, atomic=atomic) - - if not should_close: - f = t.cast("t.IO[t.Any]", KeepOpenFile(f)) - - return f - - -def format_filename( - filename: str | bytes | os.PathLike[str] | os.PathLike[bytes], - shorten: bool = False, -) -> str: - """Format a filename as a string for display. Ensures the filename can be - displayed by replacing any invalid bytes or surrogate escapes in the name - with the replacement character ``�``. - - Invalid bytes or surrogate escapes will raise an error when written to a - stream with ``errors="strict"``. This will typically happen with ``stdout`` - when the locale is something like ``en_GB.UTF-8``. - - Many scenarios *are* safe to write surrogates though, due to PEP 538 and - PEP 540, including: - - - Writing to ``stderr``, which uses ``errors="backslashreplace"``. - - The system has ``LANG=C.UTF-8``, ``C``, or ``POSIX``. Python opens - stdout and stderr with ``errors="surrogateescape"``. - - None of ``LANG/LC_*`` are set. Python assumes ``LANG=C.UTF-8``. - - Python is started in UTF-8 mode with ``PYTHONUTF8=1`` or ``-X utf8``. - Python opens stdout and stderr with ``errors="surrogateescape"``. - - :param filename: formats a filename for UI display. This will also convert - the filename into unicode without failing. - :param shorten: this optionally shortens the filename to strip of the - path that leads up to it. - """ - if shorten: - filename = os.path.basename(filename) - else: - filename = os.fspath(filename) - - if isinstance(filename, bytes): - filename = filename.decode(sys.getfilesystemencoding(), "replace") - else: - filename = filename.encode("utf-8", "surrogateescape").decode( - "utf-8", "replace" - ) - - return filename - - -def get_app_dir(app_name: str, roaming: bool = True, force_posix: bool = False) -> str: - r"""Returns the config folder for the application. The default behavior - is to return whatever is most appropriate for the operating system. - - To give you an idea, for an app called ``"Foo Bar"``, something like - the following folders could be returned: - - Mac OS X: - ``~/Library/Application Support/Foo Bar`` - Mac OS X (POSIX): - ``~/.foo-bar`` - Unix: - ``~/.config/foo-bar`` - Unix (POSIX): - ``~/.foo-bar`` - Windows (roaming): - ``C:\Users\\AppData\Roaming\Foo Bar`` - Windows (not roaming): - ``C:\Users\\AppData\Local\Foo Bar`` - - .. versionadded:: 2.0 - - :param app_name: the application name. This should be properly capitalized - and can contain whitespace. - :param roaming: controls if the folder should be roaming or not on Windows. - Has no effect otherwise. - :param force_posix: if this is set to `True` then on any POSIX system the - folder will be stored in the home folder with a leading - dot instead of the XDG config home or darwin's - application support folder. - """ - if WIN: - key = "APPDATA" if roaming else "LOCALAPPDATA" - folder = os.environ.get(key) - if folder is None: - folder = os.path.expanduser("~") - return os.path.join(folder, app_name) - if force_posix: - return os.path.join(os.path.expanduser(f"~/.{_posixify(app_name)}")) - if sys.platform == "darwin": - return os.path.join( - os.path.expanduser("~/Library/Application Support"), app_name - ) - return os.path.join( - os.environ.get("XDG_CONFIG_HOME", os.path.expanduser("~/.config")), - _posixify(app_name), - ) - - -class PacifyFlushWrapper: - """This wrapper is used to catch and suppress BrokenPipeErrors resulting - from ``.flush()`` being called on broken pipe during the shutdown/final-GC - of the Python interpreter. Notably ``.flush()`` is always called on - ``sys.stdout`` and ``sys.stderr``. So as to have minimal impact on any - other cleanup code, and the case where the underlying file is not a broken - pipe, all calls and attributes are proxied. - """ - - def __init__(self, wrapped: t.IO[t.Any]) -> None: - self.wrapped = wrapped - - def flush(self) -> None: - try: - self.wrapped.flush() - except OSError as e: - import errno - - if e.errno != errno.EPIPE: - raise - - def __getattr__(self, attr: str) -> t.Any: - return getattr(self.wrapped, attr) - - -def _detect_program_name( - path: str | None = None, _main: ModuleType | None = None -) -> str: - """Determine the command used to run the program, for use in help - text. If a file or entry point was executed, the file name is - returned. If ``python -m`` was used to execute a module or package, - ``python -m name`` is returned. - - This doesn't try to be too precise, the goal is to give a concise - name for help text. Files are only shown as their name without the - path. ``python`` is only shown for modules, and the full path to - ``sys.executable`` is not shown. - - :param path: The Python file being executed. Python puts this in - ``sys.argv[0]``, which is used by default. - :param _main: The ``__main__`` module. This should only be passed - during internal testing. - - .. versionadded:: 8.0 - Based on command args detection in the Werkzeug reloader. - - :meta private: - """ - if _main is None: - _main = sys.modules["__main__"] - - if not path: - path = sys.argv[0] - - # The value of __package__ indicates how Python was called. It may - # not exist if a setuptools script is installed as an egg. It may be - # set incorrectly for entry points created with pip on Windows. - # It is set to "" inside a Shiv or PEX zipapp. - if getattr(_main, "__package__", None) in {None, ""} or ( - os.name == "nt" - and _main.__package__ == "" - and not os.path.exists(path) - and os.path.exists(f"{path}.exe") - ): - # Executed a file, like "python app.py". - return os.path.basename(path) - - # Executed a module, like "python -m example". - # Rewritten by Python from "-m script" to "/path/to/script.py". - # Need to look at main module to determine how it was executed. - py_module = t.cast(str, _main.__package__) - name = os.path.splitext(os.path.basename(path))[0] - - # A submodule like "example.cli". - if name != "__main__": - py_module = f"{py_module}.{name}" - - return f"python -m {py_module.lstrip('.')}" - - -def _expand_args( - args: cabc.Iterable[str], - *, - user: bool = True, - env: bool = True, - glob_recursive: bool = True, -) -> list[str]: - """Simulate Unix shell expansion with Python functions. - - See :func:`glob.glob`, :func:`os.path.expanduser`, and - :func:`os.path.expandvars`. - - This is intended for use on Windows, where the shell does not do any - expansion. It may not exactly match what a Unix shell would do. - - :param args: List of command line arguments to expand. - :param user: Expand user home directory. - :param env: Expand environment variables. - :param glob_recursive: ``**`` matches directories recursively. - - .. versionchanged:: 8.1 - Invalid glob patterns are treated as empty expansions rather - than raising an error. - - .. versionadded:: 8.0 - - :meta private: - """ - from glob import glob - - out = [] - - for arg in args: - if user: - arg = os.path.expanduser(arg) - - if env: - arg = os.path.expandvars(arg) - - try: - matches = glob(arg, recursive=glob_recursive) - except re.error: - matches = [] - - if not matches: - out.append(arg) - else: - out.extend(matches) - - return out diff --git a/.env/Lib/site-packages/colorama-0.4.6.dist-info/INSTALLER b/.env/Lib/site-packages/colorama-0.4.6.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/colorama-0.4.6.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/colorama-0.4.6.dist-info/METADATA b/.env/Lib/site-packages/colorama-0.4.6.dist-info/METADATA deleted file mode 100644 index a1b5c57..0000000 --- a/.env/Lib/site-packages/colorama-0.4.6.dist-info/METADATA +++ /dev/null @@ -1,441 +0,0 @@ -Metadata-Version: 2.1 -Name: colorama -Version: 0.4.6 -Summary: Cross-platform colored terminal text. -Project-URL: Homepage, https://github.com/tartley/colorama -Author-email: Jonathan Hartley -License-File: LICENSE.txt -Keywords: ansi,color,colour,crossplatform,terminal,text,windows,xplatform -Classifier: Development Status :: 5 - Production/Stable -Classifier: Environment :: Console -Classifier: Intended Audience :: Developers -Classifier: License :: OSI Approved :: BSD License -Classifier: Operating System :: OS Independent -Classifier: Programming Language :: Python -Classifier: Programming Language :: Python :: 2 -Classifier: Programming Language :: Python :: 2.7 -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3.7 -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: Implementation :: CPython -Classifier: Programming Language :: Python :: Implementation :: PyPy -Classifier: Topic :: Terminals -Requires-Python: !=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7 -Description-Content-Type: text/x-rst - -.. image:: https://img.shields.io/pypi/v/colorama.svg - :target: https://pypi.org/project/colorama/ - :alt: Latest Version - -.. image:: https://img.shields.io/pypi/pyversions/colorama.svg - :target: https://pypi.org/project/colorama/ - :alt: Supported Python versions - -.. image:: https://github.com/tartley/colorama/actions/workflows/test.yml/badge.svg - :target: https://github.com/tartley/colorama/actions/workflows/test.yml - :alt: Build Status - -Colorama -======== - -Makes ANSI escape character sequences (for producing colored terminal text and -cursor positioning) work under MS Windows. - -.. |donate| image:: https://www.paypalobjects.com/en_US/i/btn/btn_donate_SM.gif - :target: https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=2MZ9D2GMLYCUJ&item_name=Colorama¤cy_code=USD - :alt: Donate with Paypal - -`PyPI for releases `_ | -`Github for source `_ | -`Colorama for enterprise on Tidelift `_ - -If you find Colorama useful, please |donate| to the authors. Thank you! - -Installation ------------- - -Tested on CPython 2.7, 3.7, 3.8, 3.9 and 3.10 and Pypy 2.7 and 3.8. - -No requirements other than the standard library. - -.. code-block:: bash - - pip install colorama - # or - conda install -c anaconda colorama - -Description ------------ - -ANSI escape character sequences have long been used to produce colored terminal -text and cursor positioning on Unix and Macs. Colorama makes this work on -Windows, too, by wrapping ``stdout``, stripping ANSI sequences it finds (which -would appear as gobbledygook in the output), and converting them into the -appropriate win32 calls to modify the state of the terminal. On other platforms, -Colorama does nothing. - -This has the upshot of providing a simple cross-platform API for printing -colored terminal text from Python, and has the happy side-effect that existing -applications or libraries which use ANSI sequences to produce colored output on -Linux or Macs can now also work on Windows, simply by calling -``colorama.just_fix_windows_console()`` (since v0.4.6) or ``colorama.init()`` -(all versions, but may have other side-effects – see below). - -An alternative approach is to install ``ansi.sys`` on Windows machines, which -provides the same behaviour for all applications running in terminals. Colorama -is intended for situations where that isn't easy (e.g., maybe your app doesn't -have an installer.) - -Demo scripts in the source code repository print some colored text using -ANSI sequences. Compare their output under Gnome-terminal's built in ANSI -handling, versus on Windows Command-Prompt using Colorama: - -.. image:: https://github.com/tartley/colorama/raw/master/screenshots/ubuntu-demo.png - :width: 661 - :height: 357 - :alt: ANSI sequences on Ubuntu under gnome-terminal. - -.. image:: https://github.com/tartley/colorama/raw/master/screenshots/windows-demo.png - :width: 668 - :height: 325 - :alt: Same ANSI sequences on Windows, using Colorama. - -These screenshots show that, on Windows, Colorama does not support ANSI 'dim -text'; it looks the same as 'normal text'. - -Usage ------ - -Initialisation -.............. - -If the only thing you want from Colorama is to get ANSI escapes to work on -Windows, then run: - -.. code-block:: python - - from colorama import just_fix_windows_console - just_fix_windows_console() - -If you're on a recent version of Windows 10 or better, and your stdout/stderr -are pointing to a Windows console, then this will flip the magic configuration -switch to enable Windows' built-in ANSI support. - -If you're on an older version of Windows, and your stdout/stderr are pointing to -a Windows console, then this will wrap ``sys.stdout`` and/or ``sys.stderr`` in a -magic file object that intercepts ANSI escape sequences and issues the -appropriate Win32 calls to emulate them. - -In all other circumstances, it does nothing whatsoever. Basically the idea is -that this makes Windows act like Unix with respect to ANSI escape handling. - -It's safe to call this function multiple times. It's safe to call this function -on non-Windows platforms, but it won't do anything. It's safe to call this -function when one or both of your stdout/stderr are redirected to a file – it -won't do anything to those streams. - -Alternatively, you can use the older interface with more features (but also more -potential footguns): - -.. code-block:: python - - from colorama import init - init() - -This does the same thing as ``just_fix_windows_console``, except for the -following differences: - -- It's not safe to call ``init`` multiple times; you can end up with multiple - layers of wrapping and broken ANSI support. - -- Colorama will apply a heuristic to guess whether stdout/stderr support ANSI, - and if it thinks they don't, then it will wrap ``sys.stdout`` and - ``sys.stderr`` in a magic file object that strips out ANSI escape sequences - before printing them. This happens on all platforms, and can be convenient if - you want to write your code to emit ANSI escape sequences unconditionally, and - let Colorama decide whether they should actually be output. But note that - Colorama's heuristic is not particularly clever. - -- ``init`` also accepts explicit keyword args to enable/disable various - functionality – see below. - -To stop using Colorama before your program exits, simply call ``deinit()``. -This will restore ``stdout`` and ``stderr`` to their original values, so that -Colorama is disabled. To resume using Colorama again, call ``reinit()``; it is -cheaper than calling ``init()`` again (but does the same thing). - -Most users should depend on ``colorama >= 0.4.6``, and use -``just_fix_windows_console``. The old ``init`` interface will be supported -indefinitely for backwards compatibility, but we don't plan to fix any issues -with it, also for backwards compatibility. - -Colored Output -.............. - -Cross-platform printing of colored text can then be done using Colorama's -constant shorthand for ANSI escape sequences. These are deliberately -rudimentary, see below. - -.. code-block:: python - - from colorama import Fore, Back, Style - print(Fore.RED + 'some red text') - print(Back.GREEN + 'and with a green background') - print(Style.DIM + 'and in dim text') - print(Style.RESET_ALL) - print('back to normal now') - -...or simply by manually printing ANSI sequences from your own code: - -.. code-block:: python - - print('\033[31m' + 'some red text') - print('\033[39m') # and reset to default color - -...or, Colorama can be used in conjunction with existing ANSI libraries -such as the venerable `Termcolor `_ -the fabulous `Blessings `_, -or the incredible `_Rich `_. - -If you wish Colorama's Fore, Back and Style constants were more capable, -then consider using one of the above highly capable libraries to generate -colors, etc, and use Colorama just for its primary purpose: to convert -those ANSI sequences to also work on Windows: - -SIMILARLY, do not send PRs adding the generation of new ANSI types to Colorama. -We are only interested in converting ANSI codes to win32 API calls, not -shortcuts like the above to generate ANSI characters. - -.. code-block:: python - - from colorama import just_fix_windows_console - from termcolor import colored - - # use Colorama to make Termcolor work on Windows too - just_fix_windows_console() - - # then use Termcolor for all colored text output - print(colored('Hello, World!', 'green', 'on_red')) - -Available formatting constants are:: - - Fore: BLACK, RED, GREEN, YELLOW, BLUE, MAGENTA, CYAN, WHITE, RESET. - Back: BLACK, RED, GREEN, YELLOW, BLUE, MAGENTA, CYAN, WHITE, RESET. - Style: DIM, NORMAL, BRIGHT, RESET_ALL - -``Style.RESET_ALL`` resets foreground, background, and brightness. Colorama will -perform this reset automatically on program exit. - -These are fairly well supported, but not part of the standard:: - - Fore: LIGHTBLACK_EX, LIGHTRED_EX, LIGHTGREEN_EX, LIGHTYELLOW_EX, LIGHTBLUE_EX, LIGHTMAGENTA_EX, LIGHTCYAN_EX, LIGHTWHITE_EX - Back: LIGHTBLACK_EX, LIGHTRED_EX, LIGHTGREEN_EX, LIGHTYELLOW_EX, LIGHTBLUE_EX, LIGHTMAGENTA_EX, LIGHTCYAN_EX, LIGHTWHITE_EX - -Cursor Positioning -.................. - -ANSI codes to reposition the cursor are supported. See ``demos/demo06.py`` for -an example of how to generate them. - -Init Keyword Args -................. - -``init()`` accepts some ``**kwargs`` to override default behaviour. - -init(autoreset=False): - If you find yourself repeatedly sending reset sequences to turn off color - changes at the end of every print, then ``init(autoreset=True)`` will - automate that: - - .. code-block:: python - - from colorama import init - init(autoreset=True) - print(Fore.RED + 'some red text') - print('automatically back to default color again') - -init(strip=None): - Pass ``True`` or ``False`` to override whether ANSI codes should be - stripped from the output. The default behaviour is to strip if on Windows - or if output is redirected (not a tty). - -init(convert=None): - Pass ``True`` or ``False`` to override whether to convert ANSI codes in the - output into win32 calls. The default behaviour is to convert if on Windows - and output is to a tty (terminal). - -init(wrap=True): - On Windows, Colorama works by replacing ``sys.stdout`` and ``sys.stderr`` - with proxy objects, which override the ``.write()`` method to do their work. - If this wrapping causes you problems, then this can be disabled by passing - ``init(wrap=False)``. The default behaviour is to wrap if ``autoreset`` or - ``strip`` or ``convert`` are True. - - When wrapping is disabled, colored printing on non-Windows platforms will - continue to work as normal. To do cross-platform colored output, you can - use Colorama's ``AnsiToWin32`` proxy directly: - - .. code-block:: python - - import sys - from colorama import init, AnsiToWin32 - init(wrap=False) - stream = AnsiToWin32(sys.stderr).stream - - # Python 2 - print >>stream, Fore.BLUE + 'blue text on stderr' - - # Python 3 - print(Fore.BLUE + 'blue text on stderr', file=stream) - -Recognised ANSI Sequences -......................... - -ANSI sequences generally take the form:: - - ESC [ ; ... - -Where ```` is an integer, and ```` is a single letter. Zero or -more params are passed to a ````. If no params are passed, it is -generally synonymous with passing a single zero. No spaces exist in the -sequence; they have been inserted here simply to read more easily. - -The only ANSI sequences that Colorama converts into win32 calls are:: - - ESC [ 0 m # reset all (colors and brightness) - ESC [ 1 m # bright - ESC [ 2 m # dim (looks same as normal brightness) - ESC [ 22 m # normal brightness - - # FOREGROUND: - ESC [ 30 m # black - ESC [ 31 m # red - ESC [ 32 m # green - ESC [ 33 m # yellow - ESC [ 34 m # blue - ESC [ 35 m # magenta - ESC [ 36 m # cyan - ESC [ 37 m # white - ESC [ 39 m # reset - - # BACKGROUND - ESC [ 40 m # black - ESC [ 41 m # red - ESC [ 42 m # green - ESC [ 43 m # yellow - ESC [ 44 m # blue - ESC [ 45 m # magenta - ESC [ 46 m # cyan - ESC [ 47 m # white - ESC [ 49 m # reset - - # cursor positioning - ESC [ y;x H # position cursor at x across, y down - ESC [ y;x f # position cursor at x across, y down - ESC [ n A # move cursor n lines up - ESC [ n B # move cursor n lines down - ESC [ n C # move cursor n characters forward - ESC [ n D # move cursor n characters backward - - # clear the screen - ESC [ mode J # clear the screen - - # clear the line - ESC [ mode K # clear the line - -Multiple numeric params to the ``'m'`` command can be combined into a single -sequence:: - - ESC [ 36 ; 45 ; 1 m # bright cyan text on magenta background - -All other ANSI sequences of the form ``ESC [ ; ... `` -are silently stripped from the output on Windows. - -Any other form of ANSI sequence, such as single-character codes or alternative -initial characters, are not recognised or stripped. It would be cool to add -them though. Let me know if it would be useful for you, via the Issues on -GitHub. - -Status & Known Problems ------------------------ - -I've personally only tested it on Windows XP (CMD, Console2), Ubuntu -(gnome-terminal, xterm), and OS X. - -Some valid ANSI sequences aren't recognised. - -If you're hacking on the code, see `README-hacking.md`_. ESPECIALLY, see the -explanation there of why we do not want PRs that allow Colorama to generate new -types of ANSI codes. - -See outstanding issues and wish-list: -https://github.com/tartley/colorama/issues - -If anything doesn't work for you, or doesn't do what you expected or hoped for, -I'd love to hear about it on that issues list, would be delighted by patches, -and would be happy to grant commit access to anyone who submits a working patch -or two. - -.. _README-hacking.md: README-hacking.md - -License -------- - -Copyright Jonathan Hartley & Arnon Yaari, 2013-2020. BSD 3-Clause license; see -LICENSE file. - -Professional support --------------------- - -.. |tideliftlogo| image:: https://cdn2.hubspot.net/hubfs/4008838/website/logos/logos_for_download/Tidelift_primary-shorthand-logo.png - :alt: Tidelift - :target: https://tidelift.com/subscription/pkg/pypi-colorama?utm_source=pypi-colorama&utm_medium=referral&utm_campaign=readme - -.. list-table:: - :widths: 10 100 - - * - |tideliftlogo| - - Professional support for colorama is available as part of the - `Tidelift Subscription`_. - Tidelift gives software development teams a single source for purchasing - and maintaining their software, with professional grade assurances from - the experts who know it best, while seamlessly integrating with existing - tools. - -.. _Tidelift Subscription: https://tidelift.com/subscription/pkg/pypi-colorama?utm_source=pypi-colorama&utm_medium=referral&utm_campaign=readme - -Thanks ------- - -See the CHANGELOG for more thanks! - -* Marc Schlaich (schlamar) for a ``setup.py`` fix for Python2.5. -* Marc Abramowitz, reported & fixed a crash on exit with closed ``stdout``, - providing a solution to issue #7's setuptools/distutils debate, - and other fixes. -* User 'eryksun', for guidance on correctly instantiating ``ctypes.windll``. -* Matthew McCormick for politely pointing out a longstanding crash on non-Win. -* Ben Hoyt, for a magnificent fix under 64-bit Windows. -* Jesse at Empty Square for submitting a fix for examples in the README. -* User 'jamessp', an observant documentation fix for cursor positioning. -* User 'vaal1239', Dave Mckee & Lackner Kristof for a tiny but much-needed Win7 - fix. -* Julien Stuyck, for wisely suggesting Python3 compatible updates to README. -* Daniel Griffith for multiple fabulous patches. -* Oscar Lesta for a valuable fix to stop ANSI chars being sent to non-tty - output. -* Roger Binns, for many suggestions, valuable feedback, & bug reports. -* Tim Golden for thought and much appreciated feedback on the initial idea. -* User 'Zearin' for updates to the README file. -* John Szakmeister for adding support for light colors -* Charles Merriam for adding documentation to demos -* Jurko for a fix on 64-bit Windows CPython2.5 w/o ctypes -* Florian Bruhin for a fix when stdout or stderr are None -* Thomas Weininger for fixing ValueError on Windows -* Remi Rampin for better Github integration and fixes to the README file -* Simeon Visser for closing a file handle using 'with' and updating classifiers - to include Python 3.3 and 3.4 -* Andy Neff for fixing RESET of LIGHT_EX colors. -* Jonathan Hartley for the initial idea and implementation. diff --git a/.env/Lib/site-packages/colorama-0.4.6.dist-info/RECORD b/.env/Lib/site-packages/colorama-0.4.6.dist-info/RECORD deleted file mode 100644 index cd6b130..0000000 --- a/.env/Lib/site-packages/colorama-0.4.6.dist-info/RECORD +++ /dev/null @@ -1,31 +0,0 @@ -colorama-0.4.6.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -colorama-0.4.6.dist-info/METADATA,sha256=e67SnrUMOym9sz_4TjF3vxvAV4T3aF7NyqRHHH3YEMw,17158 -colorama-0.4.6.dist-info/RECORD,, -colorama-0.4.6.dist-info/WHEEL,sha256=cdcF4Fbd0FPtw2EMIOwH-3rSOTUdTCeOSXRMD1iLUb8,105 -colorama-0.4.6.dist-info/licenses/LICENSE.txt,sha256=ysNcAmhuXQSlpxQL-zs25zrtSWZW6JEQLkKIhteTAxg,1491 -colorama/__init__.py,sha256=wePQA4U20tKgYARySLEC047ucNX-g8pRLpYBuiHlLb8,266 -colorama/__pycache__/__init__.cpython-312.pyc,, -colorama/__pycache__/ansi.cpython-312.pyc,, -colorama/__pycache__/ansitowin32.cpython-312.pyc,, -colorama/__pycache__/initialise.cpython-312.pyc,, -colorama/__pycache__/win32.cpython-312.pyc,, -colorama/__pycache__/winterm.cpython-312.pyc,, -colorama/ansi.py,sha256=Top4EeEuaQdBWdteKMEcGOTeKeF19Q-Wo_6_Cj5kOzQ,2522 -colorama/ansitowin32.py,sha256=vPNYa3OZbxjbuFyaVo0Tmhmy1FZ1lKMWCnT7odXpItk,11128 -colorama/initialise.py,sha256=-hIny86ClXo39ixh5iSCfUIa2f_h_bgKRDW7gqs-KLU,3325 -colorama/tests/__init__.py,sha256=MkgPAEzGQd-Rq0w0PZXSX2LadRWhUECcisJY8lSrm4Q,75 -colorama/tests/__pycache__/__init__.cpython-312.pyc,, -colorama/tests/__pycache__/ansi_test.cpython-312.pyc,, -colorama/tests/__pycache__/ansitowin32_test.cpython-312.pyc,, -colorama/tests/__pycache__/initialise_test.cpython-312.pyc,, -colorama/tests/__pycache__/isatty_test.cpython-312.pyc,, -colorama/tests/__pycache__/utils.cpython-312.pyc,, -colorama/tests/__pycache__/winterm_test.cpython-312.pyc,, -colorama/tests/ansi_test.py,sha256=FeViDrUINIZcr505PAxvU4AjXz1asEiALs9GXMhwRaE,2839 -colorama/tests/ansitowin32_test.py,sha256=RN7AIhMJ5EqDsYaCjVo-o4u8JzDD4ukJbmevWKS70rY,10678 -colorama/tests/initialise_test.py,sha256=BbPy-XfyHwJ6zKozuQOvNvQZzsx9vdb_0bYXn7hsBTc,6741 -colorama/tests/isatty_test.py,sha256=Pg26LRpv0yQDB5Ac-sxgVXG7hsA1NYvapFgApZfYzZg,1866 -colorama/tests/utils.py,sha256=1IIRylG39z5-dzq09R_ngufxyPZxgldNbrxKxUGwGKE,1079 -colorama/tests/winterm_test.py,sha256=qoWFPEjym5gm2RuMwpf3pOis3a5r_PJZFCzK254JL8A,3709 -colorama/win32.py,sha256=YQOKwMTwtGBbsY4dL5HYTvwTeP9wIQra5MvPNddpxZs,6181 -colorama/winterm.py,sha256=XCQFDHjPi6AHYNdZwy0tA02H-Jh48Jp-HvCjeLeLp3U,7134 diff --git a/.env/Lib/site-packages/colorama-0.4.6.dist-info/WHEEL b/.env/Lib/site-packages/colorama-0.4.6.dist-info/WHEEL deleted file mode 100644 index d79189f..0000000 --- a/.env/Lib/site-packages/colorama-0.4.6.dist-info/WHEEL +++ /dev/null @@ -1,5 +0,0 @@ -Wheel-Version: 1.0 -Generator: hatchling 1.11.1 -Root-Is-Purelib: true -Tag: py2-none-any -Tag: py3-none-any diff --git a/.env/Lib/site-packages/colorama-0.4.6.dist-info/licenses/LICENSE.txt b/.env/Lib/site-packages/colorama-0.4.6.dist-info/licenses/LICENSE.txt deleted file mode 100644 index 3105888..0000000 --- a/.env/Lib/site-packages/colorama-0.4.6.dist-info/licenses/LICENSE.txt +++ /dev/null @@ -1,27 +0,0 @@ -Copyright (c) 2010 Jonathan Hartley -All rights reserved. - -Redistribution and use in source and binary forms, with or without -modification, are permitted provided that the following conditions are met: - -* Redistributions of source code must retain the above copyright notice, this - list of conditions and the following disclaimer. - -* Redistributions in binary form must reproduce the above copyright notice, - this list of conditions and the following disclaimer in the documentation - and/or other materials provided with the distribution. - -* Neither the name of the copyright holders, nor those of its contributors - may be used to endorse or promote products derived from this software without - specific prior written permission. - -THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND -ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED -WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE -DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE -FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL -DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR -SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER -CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, -OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/.env/Lib/site-packages/colorama/__init__.py b/.env/Lib/site-packages/colorama/__init__.py deleted file mode 100644 index 383101c..0000000 --- a/.env/Lib/site-packages/colorama/__init__.py +++ /dev/null @@ -1,7 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -from .initialise import init, deinit, reinit, colorama_text, just_fix_windows_console -from .ansi import Fore, Back, Style, Cursor -from .ansitowin32 import AnsiToWin32 - -__version__ = '0.4.6' - diff --git a/.env/Lib/site-packages/colorama/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/colorama/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 081bb39c7cdf7fa75aad4cc964879db8c2a6edb8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 481 zcmXv~Jxeq(5KeBhdmnpeAqv`A*ev2HDi$Jw3Syxa>LG-MuF zga5(8-coD=qF^C*f}NEUKc;wQ=6T3GleufP8VKU%_EI!3LNBiQi^^~Du?fW$%20-L zBr(OZLMzgvo+Ok=pZeHol{}DDT9q|glXY5`4GIH+_wrCSX%lKfTLV7|-oOV%z9pg% zr42Ivll4z-r7BlCkqImLS!u_Pd!}SZ;*6b&B2%Y^rK&J0=Mk}FLuqcwT9Tevf3FkizBbm)}ODW!s4=I3>7Olg+BM!m*Nf>n0QxZYGOM{|MNne@VGmO zw?(QIXnm1#6SsNMkGI5O3{&}Z7k-lb1N!d$IT2=y?iU!@;QKP5e{AJ9YZy;ulRI0N5$+M8YS3vF9X#Wj_yTl?Iu>?)9L`}3LO|s-2qREy5 zs8|_G-4Qeu>JZc+k<1b+yhE&r9eyYU<7g3Xje<2cV2yHX9IOdD((fDN)+AU{1J*dV zj(~M^z?u+=9edbyR!f2w?`tX05`Aq1v}j)&ZKiXFwp!8Ub;r$aG_312JG*ArZEAS7 zn=MqTb6IoUpoZz$G;202FnW#bCUxqbQ(ps1twFg7+BclKQFHUrs%=kZ*FA4@a$>@+ z=Ra|-Ih(fS82JWWn?P*>$JK4uG&XG=2aRrt$3KK}<+%_th3+b_lA&Pe+NS~T5)Ux= zyly}V*(MLWe{4aMB_LKwPOLCt&`WMb+frT@0hd!Z8sd#c31Ou{l;q9`DoR3Jo~*(!@Op!ftfGe*f1uHy6faO zn=FB&>s~|0fVmVo6^GJ34A3G^V#zNvyUSmU-8#Ql-H&A-$=NnEkfG&@VBpLB9teZW z4VJY{aB&{?;$MHMTjKZ6=Jg7Oq`!Q(49UGV*wEov4XB&OIOKbvX0fDnXfD=G!iaCiz z3c!aVu5l=OVd&4d_74Bi>kTx(-)VB?M}&MTYztTW?Cwtt%~By@Lt<-xeNr zDe&Ax7lp99vy6vLF*=#moT3FYey5s5kR=OXIe9txxJ>7EvEz${- z!^igIKg@kL{Wx>ius< ze)23t4!_QMQOU^+W-ul58?~T?z6EnV2&lx}qsPI(0r%R)fbEdB z7*Zzru)M_tWB=>rTl6j-yoi%j(|f z7r(f*y=U&HPCN>q2$Jp9IX4v~TOKAI@=KmxxK!Cn4S0rzx1k$7i@JJm2Xi7VK*b>!rr$-G^8v3xdIkYgkNy~865;0ve3y_dL0a=@ zA)JEZVv=K83(A{qc?de7o()nRdk>|0?^JFtLd!s^W15SQ;u55{45_VXAxqW5R!ECj zVJ&J!w3rpu;#N#cSaB_BCA5^4)JCk7HfoJ%X=_x=z(Xs;!Wglc2DSayVFLdRge<}d z1k7R|1L)%td`wp#x1PnB0nEHq!3Emf8aOfFt7RG4zPmn_Cr z7z&l3jSq=xlD7pzqaN@6x>7m)Pk}2)Sw>R*di>ABH;1hUIy49kB=N{ zl@N-3X$GZfgxS8tyOf@)-sTWn8>t9FpHE4NxvNe^Q8oa8iUbANNQ^Lpl!N<3v4~0~6 zp@YO%rN*g zZAt%71>pq3sWwU+I^nP|xvO>vRGlMH;q2~_4uPsOo)N}AYjy}!ozod%Jdnpbr)43% k8|e_Jp2bNteS5qme;XMMs`TySE#=!t=JsnhRUWH<0t_oOwEzGB diff --git a/.env/Lib/site-packages/colorama/__pycache__/ansitowin32.cpython-312.pyc b/.env/Lib/site-packages/colorama/__pycache__/ansitowin32.cpython-312.pyc deleted file mode 100644 index d3277bfe4c70d7de8cb353a01a9c9dd7bab6b194..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16401 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 diff --git a/.env/Lib/site-packages/colorama/ansi.py b/.env/Lib/site-packages/colorama/ansi.py deleted file mode 100644 index 11ec695..0000000 --- a/.env/Lib/site-packages/colorama/ansi.py +++ /dev/null @@ -1,102 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -''' -This module generates ANSI character codes to printing colors to terminals. -See: http://en.wikipedia.org/wiki/ANSI_escape_code -''' - -CSI = '\033[' -OSC = '\033]' -BEL = '\a' - - -def code_to_chars(code): - return CSI + str(code) + 'm' - -def set_title(title): - return OSC + '2;' + title + BEL - -def clear_screen(mode=2): - return CSI + str(mode) + 'J' - -def clear_line(mode=2): - return CSI + str(mode) + 'K' - - -class AnsiCodes(object): - def __init__(self): - # the subclasses declare class attributes which are numbers. - # Upon instantiation we define instance attributes, which are the same - # as the class attributes but wrapped with the ANSI escape sequence - for name in dir(self): - if not name.startswith('_'): - value = getattr(self, name) - setattr(self, name, code_to_chars(value)) - - -class AnsiCursor(object): - def UP(self, n=1): - return CSI + str(n) + 'A' - def DOWN(self, n=1): - return CSI + str(n) + 'B' - def FORWARD(self, n=1): - return CSI + str(n) + 'C' - def BACK(self, n=1): - return CSI + str(n) + 'D' - def POS(self, x=1, y=1): - return CSI + str(y) + ';' + str(x) + 'H' - - -class AnsiFore(AnsiCodes): - BLACK = 30 - RED = 31 - GREEN = 32 - YELLOW = 33 - BLUE = 34 - MAGENTA = 35 - CYAN = 36 - WHITE = 37 - RESET = 39 - - # These are fairly well supported, but not part of the standard. - LIGHTBLACK_EX = 90 - LIGHTRED_EX = 91 - LIGHTGREEN_EX = 92 - LIGHTYELLOW_EX = 93 - LIGHTBLUE_EX = 94 - LIGHTMAGENTA_EX = 95 - LIGHTCYAN_EX = 96 - LIGHTWHITE_EX = 97 - - -class AnsiBack(AnsiCodes): - BLACK = 40 - RED = 41 - GREEN = 42 - YELLOW = 43 - BLUE = 44 - MAGENTA = 45 - CYAN = 46 - WHITE = 47 - RESET = 49 - - # These are fairly well supported, but not part of the standard. - LIGHTBLACK_EX = 100 - LIGHTRED_EX = 101 - LIGHTGREEN_EX = 102 - LIGHTYELLOW_EX = 103 - LIGHTBLUE_EX = 104 - LIGHTMAGENTA_EX = 105 - LIGHTCYAN_EX = 106 - LIGHTWHITE_EX = 107 - - -class AnsiStyle(AnsiCodes): - BRIGHT = 1 - DIM = 2 - NORMAL = 22 - RESET_ALL = 0 - -Fore = AnsiFore() -Back = AnsiBack() -Style = AnsiStyle() -Cursor = AnsiCursor() diff --git a/.env/Lib/site-packages/colorama/ansitowin32.py b/.env/Lib/site-packages/colorama/ansitowin32.py deleted file mode 100644 index abf209e..0000000 --- a/.env/Lib/site-packages/colorama/ansitowin32.py +++ /dev/null @@ -1,277 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import re -import sys -import os - -from .ansi import AnsiFore, AnsiBack, AnsiStyle, Style, BEL -from .winterm import enable_vt_processing, WinTerm, WinColor, WinStyle -from .win32 import windll, winapi_test - - -winterm = None -if windll is not None: - winterm = WinTerm() - - -class StreamWrapper(object): - ''' - Wraps a stream (such as stdout), acting as a transparent proxy for all - attribute access apart from method 'write()', which is delegated to our - Converter instance. - ''' - def __init__(self, wrapped, converter): - # double-underscore everything to prevent clashes with names of - # attributes on the wrapped stream object. - self.__wrapped = wrapped - self.__convertor = converter - - def __getattr__(self, name): - return getattr(self.__wrapped, name) - - def __enter__(self, *args, **kwargs): - # special method lookup bypasses __getattr__/__getattribute__, see - # https://stackoverflow.com/questions/12632894/why-doesnt-getattr-work-with-exit - # thus, contextlib magic methods are not proxied via __getattr__ - return self.__wrapped.__enter__(*args, **kwargs) - - def __exit__(self, *args, **kwargs): - return self.__wrapped.__exit__(*args, **kwargs) - - def __setstate__(self, state): - self.__dict__ = state - - def __getstate__(self): - return self.__dict__ - - def write(self, text): - self.__convertor.write(text) - - def isatty(self): - stream = self.__wrapped - if 'PYCHARM_HOSTED' in os.environ: - if stream is not None and (stream is sys.__stdout__ or stream is sys.__stderr__): - return True - try: - stream_isatty = stream.isatty - except AttributeError: - return False - else: - return stream_isatty() - - @property - def closed(self): - stream = self.__wrapped - try: - return stream.closed - # AttributeError in the case that the stream doesn't support being closed - # ValueError for the case that the stream has already been detached when atexit runs - except (AttributeError, ValueError): - return True - - -class AnsiToWin32(object): - ''' - Implements a 'write()' method which, on Windows, will strip ANSI character - sequences from the text, and if outputting to a tty, will convert them into - win32 function calls. - ''' - ANSI_CSI_RE = re.compile('\001?\033\\[((?:\\d|;)*)([a-zA-Z])\002?') # Control Sequence Introducer - ANSI_OSC_RE = re.compile('\001?\033\\]([^\a]*)(\a)\002?') # Operating System Command - - def __init__(self, wrapped, convert=None, strip=None, autoreset=False): - # The wrapped stream (normally sys.stdout or sys.stderr) - self.wrapped = wrapped - - # should we reset colors to defaults after every .write() - self.autoreset = autoreset - - # create the proxy wrapping our output stream - self.stream = StreamWrapper(wrapped, self) - - on_windows = os.name == 'nt' - # We test if the WinAPI works, because even if we are on Windows - # we may be using a terminal that doesn't support the WinAPI - # (e.g. Cygwin Terminal). In this case it's up to the terminal - # to support the ANSI codes. - conversion_supported = on_windows and winapi_test() - try: - fd = wrapped.fileno() - except Exception: - fd = -1 - system_has_native_ansi = not on_windows or enable_vt_processing(fd) - have_tty = not self.stream.closed and self.stream.isatty() - need_conversion = conversion_supported and not system_has_native_ansi - - # should we strip ANSI sequences from our output? - if strip is None: - strip = need_conversion or not have_tty - self.strip = strip - - # should we should convert ANSI sequences into win32 calls? - if convert is None: - convert = need_conversion and have_tty - self.convert = convert - - # dict of ansi codes to win32 functions and parameters - self.win32_calls = self.get_win32_calls() - - # are we wrapping stderr? - self.on_stderr = self.wrapped is sys.stderr - - def should_wrap(self): - ''' - True if this class is actually needed. If false, then the output - stream will not be affected, nor will win32 calls be issued, so - wrapping stdout is not actually required. This will generally be - False on non-Windows platforms, unless optional functionality like - autoreset has been requested using kwargs to init() - ''' - return self.convert or self.strip or self.autoreset - - def get_win32_calls(self): - if self.convert and winterm: - return { - AnsiStyle.RESET_ALL: (winterm.reset_all, ), - AnsiStyle.BRIGHT: (winterm.style, WinStyle.BRIGHT), - AnsiStyle.DIM: (winterm.style, WinStyle.NORMAL), - AnsiStyle.NORMAL: (winterm.style, WinStyle.NORMAL), - AnsiFore.BLACK: (winterm.fore, WinColor.BLACK), - AnsiFore.RED: (winterm.fore, WinColor.RED), - AnsiFore.GREEN: (winterm.fore, WinColor.GREEN), - AnsiFore.YELLOW: (winterm.fore, WinColor.YELLOW), - AnsiFore.BLUE: (winterm.fore, WinColor.BLUE), - AnsiFore.MAGENTA: (winterm.fore, WinColor.MAGENTA), - AnsiFore.CYAN: (winterm.fore, WinColor.CYAN), - AnsiFore.WHITE: (winterm.fore, WinColor.GREY), - AnsiFore.RESET: (winterm.fore, ), - AnsiFore.LIGHTBLACK_EX: (winterm.fore, WinColor.BLACK, True), - AnsiFore.LIGHTRED_EX: (winterm.fore, WinColor.RED, True), - AnsiFore.LIGHTGREEN_EX: (winterm.fore, WinColor.GREEN, True), - AnsiFore.LIGHTYELLOW_EX: (winterm.fore, WinColor.YELLOW, True), - AnsiFore.LIGHTBLUE_EX: (winterm.fore, WinColor.BLUE, True), - AnsiFore.LIGHTMAGENTA_EX: (winterm.fore, WinColor.MAGENTA, True), - AnsiFore.LIGHTCYAN_EX: (winterm.fore, WinColor.CYAN, True), - AnsiFore.LIGHTWHITE_EX: (winterm.fore, WinColor.GREY, True), - AnsiBack.BLACK: (winterm.back, WinColor.BLACK), - AnsiBack.RED: (winterm.back, WinColor.RED), - AnsiBack.GREEN: (winterm.back, WinColor.GREEN), - AnsiBack.YELLOW: (winterm.back, WinColor.YELLOW), - AnsiBack.BLUE: (winterm.back, WinColor.BLUE), - AnsiBack.MAGENTA: (winterm.back, WinColor.MAGENTA), - AnsiBack.CYAN: (winterm.back, WinColor.CYAN), - AnsiBack.WHITE: (winterm.back, WinColor.GREY), - AnsiBack.RESET: (winterm.back, ), - AnsiBack.LIGHTBLACK_EX: (winterm.back, WinColor.BLACK, True), - AnsiBack.LIGHTRED_EX: (winterm.back, WinColor.RED, True), - AnsiBack.LIGHTGREEN_EX: (winterm.back, WinColor.GREEN, True), - AnsiBack.LIGHTYELLOW_EX: (winterm.back, WinColor.YELLOW, True), - AnsiBack.LIGHTBLUE_EX: (winterm.back, WinColor.BLUE, True), - AnsiBack.LIGHTMAGENTA_EX: (winterm.back, WinColor.MAGENTA, True), - AnsiBack.LIGHTCYAN_EX: (winterm.back, WinColor.CYAN, True), - AnsiBack.LIGHTWHITE_EX: (winterm.back, WinColor.GREY, True), - } - return dict() - - def write(self, text): - if self.strip or self.convert: - self.write_and_convert(text) - else: - self.wrapped.write(text) - self.wrapped.flush() - if self.autoreset: - self.reset_all() - - - def reset_all(self): - if self.convert: - self.call_win32('m', (0,)) - elif not self.strip and not self.stream.closed: - self.wrapped.write(Style.RESET_ALL) - - - def write_and_convert(self, text): - ''' - Write the given text to our wrapped stream, stripping any ANSI - sequences from the text, and optionally converting them into win32 - calls. - ''' - cursor = 0 - text = self.convert_osc(text) - for match in self.ANSI_CSI_RE.finditer(text): - start, end = match.span() - self.write_plain_text(text, cursor, start) - self.convert_ansi(*match.groups()) - cursor = end - self.write_plain_text(text, cursor, len(text)) - - - def write_plain_text(self, text, start, end): - if start < end: - self.wrapped.write(text[start:end]) - self.wrapped.flush() - - - def convert_ansi(self, paramstring, command): - if self.convert: - params = self.extract_params(command, paramstring) - self.call_win32(command, params) - - - def extract_params(self, command, paramstring): - if command in 'Hf': - params = tuple(int(p) if len(p) != 0 else 1 for p in paramstring.split(';')) - while len(params) < 2: - # defaults: - params = params + (1,) - else: - params = tuple(int(p) for p in paramstring.split(';') if len(p) != 0) - if len(params) == 0: - # defaults: - if command in 'JKm': - params = (0,) - elif command in 'ABCD': - params = (1,) - - return params - - - def call_win32(self, command, params): - if command == 'm': - for param in params: - if param in self.win32_calls: - func_args = self.win32_calls[param] - func = func_args[0] - args = func_args[1:] - kwargs = dict(on_stderr=self.on_stderr) - func(*args, **kwargs) - elif command in 'J': - winterm.erase_screen(params[0], on_stderr=self.on_stderr) - elif command in 'K': - winterm.erase_line(params[0], on_stderr=self.on_stderr) - elif command in 'Hf': # cursor position - absolute - winterm.set_cursor_position(params, on_stderr=self.on_stderr) - elif command in 'ABCD': # cursor position - relative - n = params[0] - # A - up, B - down, C - forward, D - back - x, y = {'A': (0, -n), 'B': (0, n), 'C': (n, 0), 'D': (-n, 0)}[command] - winterm.cursor_adjust(x, y, on_stderr=self.on_stderr) - - - def convert_osc(self, text): - for match in self.ANSI_OSC_RE.finditer(text): - start, end = match.span() - text = text[:start] + text[end:] - paramstring, command = match.groups() - if command == BEL: - if paramstring.count(";") == 1: - params = paramstring.split(";") - # 0 - change title and icon (we will only change title) - # 1 - change icon (we don't support this) - # 2 - change title - if params[0] in '02': - winterm.set_title(params[1]) - return text - - - def flush(self): - self.wrapped.flush() diff --git a/.env/Lib/site-packages/colorama/initialise.py b/.env/Lib/site-packages/colorama/initialise.py deleted file mode 100644 index d5fd4b7..0000000 --- a/.env/Lib/site-packages/colorama/initialise.py +++ /dev/null @@ -1,121 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import atexit -import contextlib -import sys - -from .ansitowin32 import AnsiToWin32 - - -def _wipe_internal_state_for_tests(): - global orig_stdout, orig_stderr - orig_stdout = None - orig_stderr = None - - global wrapped_stdout, wrapped_stderr - wrapped_stdout = None - wrapped_stderr = None - - global atexit_done - atexit_done = False - - global fixed_windows_console - fixed_windows_console = False - - try: - # no-op if it wasn't registered - atexit.unregister(reset_all) - except AttributeError: - # python 2: no atexit.unregister. Oh well, we did our best. - pass - - -def reset_all(): - if AnsiToWin32 is not None: # Issue #74: objects might become None at exit - AnsiToWin32(orig_stdout).reset_all() - - -def init(autoreset=False, convert=None, strip=None, wrap=True): - - if not wrap and any([autoreset, convert, strip]): - raise ValueError('wrap=False conflicts with any other arg=True') - - global wrapped_stdout, wrapped_stderr - global orig_stdout, orig_stderr - - orig_stdout = sys.stdout - orig_stderr = sys.stderr - - if sys.stdout is None: - wrapped_stdout = None - else: - sys.stdout = wrapped_stdout = \ - wrap_stream(orig_stdout, convert, strip, autoreset, wrap) - if sys.stderr is None: - wrapped_stderr = None - else: - sys.stderr = wrapped_stderr = \ - wrap_stream(orig_stderr, convert, strip, autoreset, wrap) - - global atexit_done - if not atexit_done: - atexit.register(reset_all) - atexit_done = True - - -def deinit(): - if orig_stdout is not None: - sys.stdout = orig_stdout - if orig_stderr is not None: - sys.stderr = orig_stderr - - -def just_fix_windows_console(): - global fixed_windows_console - - if sys.platform != "win32": - return - if fixed_windows_console: - return - if wrapped_stdout is not None or wrapped_stderr is not None: - # Someone already ran init() and it did stuff, so we won't second-guess them - return - - # On newer versions of Windows, AnsiToWin32.__init__ will implicitly enable the - # native ANSI support in the console as a side-effect. We only need to actually - # replace sys.stdout/stderr if we're in the old-style conversion mode. - new_stdout = AnsiToWin32(sys.stdout, convert=None, strip=None, autoreset=False) - if new_stdout.convert: - sys.stdout = new_stdout - new_stderr = AnsiToWin32(sys.stderr, convert=None, strip=None, autoreset=False) - if new_stderr.convert: - sys.stderr = new_stderr - - fixed_windows_console = True - -@contextlib.contextmanager -def colorama_text(*args, **kwargs): - init(*args, **kwargs) - try: - yield - finally: - deinit() - - -def reinit(): - if wrapped_stdout is not None: - sys.stdout = wrapped_stdout - if wrapped_stderr is not None: - sys.stderr = wrapped_stderr - - -def wrap_stream(stream, convert, strip, autoreset, wrap): - if wrap: - wrapper = AnsiToWin32(stream, - convert=convert, strip=strip, autoreset=autoreset) - if wrapper.should_wrap(): - stream = wrapper.stream - return stream - - -# Use this for initial setup as well, to reduce code duplication -_wipe_internal_state_for_tests() diff --git a/.env/Lib/site-packages/colorama/tests/__init__.py b/.env/Lib/site-packages/colorama/tests/__init__.py deleted file mode 100644 index 8c5661e..0000000 --- a/.env/Lib/site-packages/colorama/tests/__init__.py +++ /dev/null @@ -1 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 1e988f702404aff738babae2c059a86f519c99d5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 187 zcmX@j%ge<81bYs&WqJeY#~=gjEsy$ H%s>_ZqnR^~ diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/ansi_test.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/ansi_test.cpython-312.pyc deleted file mode 100644 index a2af89b11233cd0c91296d5584d87705923be6fd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5456 zcmeHL-A`Oa6ra2I-rWlew3K357M0po-Ik9+*V3g;mBN-{VVh7$?LxBIaee&F0o z>!LAzsHP?w9|=j5Vqzi<{sBJt;Dc{ItwLHS8cq6O`bMBnee#@{{pPmjK5CeUb7s!W z@66}Dzccf5eSIx~^uv#zB@TEAd5ncZQuCSZdyrWoI?<&hiK2ETrIah`O36`KBD$7MJA}oZ=x(m6IUdv-G ztzcWjNkWpGc;?$Ka0U4y&sp+7iUi!OA#R!mwIUfS{J}yz$rLL)Z!p=+n#^rldS=0* zHQK07^@n4uY{yP23V z%;_#8eRFy=ac$a6SVreO>|Xpjm}fG{42`Ga-4?8|*&T-sjA6EGKFd_ouqNjh8wy*{ z#mf!Qv3U~2ZL;YmHT4fQe@^q?8~Uc{+t#mJe{JjdZf;%cf27sfRBUMd+khk$n~96f zH}f6HED;N&NLGFu2#FPQR&fK%YKz;OCA1!}0hM{h+Ql+7ah>i38*K!^&9I8XybsMy zAjD$QW^AEZMw||1ZlxC+i}Nm+@T)V?*=_QBbNd(a7qwh{SxXOqv0_p*aqKN!e>cfUu04BPn|fP;;KieUZGA16%^_%p-u|bCsc^5 zkbs8-JS5;D0S^gyNWen^9_mX4nwVFR6lW;vfuS<>+~~mIMJA7jhnPA%9u7yC`)YV} z^wJdw4Mr!!%riDH9F9y3FlF%SK!mARE{sfsnK~Z69G+nHqa(u?Cb)}O_&w(3MR0`$ zQO8|kv4PhK*I2I0Lnp&n+QUovu1eL|C6#t zeJk&-Yu)9xw&k{!#dWQ>+-BF#u8)E(PuF794}HgSzGJH}?o(wxANm40UtkqRezL-9 z-?k&?>sW=+hbr10&H0Y5!ub0tterVu=PJxVLh@3d90xqt;DBrJ zz%{tw8hmgKPPhgyTqD2Imtl|;LwJgD41EJ6*wsocDIL6}@d}p%?JpPp}vN{re=uvr1%9@lS~ltrrv&dQt&Zd<%FT zW%AI-7;{H1jgJkCGWWT0j2U;F*0G83201<#nSPV5sR_fSS*#%7m}E-i^cFgN7PWfo>4yv{FZ1B3VcI( zK)4(J^g!XckZ^&QVG~>5AzO-~p5Cf+sRy@eo7B)&N0ZvW-65;J0*4SB06xvdVu8K< zse@??>4b&rLh%WLD%ePg!_$uV$6x?Y`3if{Xk~8UgB^cn;;$qW<6O#LNAOxF#?^

G@5WD6PDse*o4Cf_eY| diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/ansitowin32_test.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/ansitowin32_test.cpython-312.pyc deleted file mode 100644 index 9b3cc68eb8a46e4b5aee10cd850585bda791ac37..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 17995 zcmc&+dvH_dmA?;N=~}*$En|%3hibq)1h(-LOvEGRVIT!Tz=6b&s6tl;fn>S&$~?5D zVRqA2Zsy}ZX0uXo{nPz#g?{d)s9)oac5#`+JVR5| zbxNc}I!s0AemcVRLl)dKVK!pwH$}L9j;2a5hYgz}mVQg-&J?yrO8ZMALcb8P_1mBg zCz``$k@EiXh@;;@?k(Yp$ddjgkmkjbaAjm^|5BP_s6CWueS;E9gQj;`bsYUwBv*i3 zTTbq>V6|vJyOyFPo}q%=yO}}L)Wx!kWh<#5-$+R%WVCF!L>=2J)RjZ$sSE5Y)Ok8n z1II>*rm#JKyil`WtLZL(h<&>mg+CILLW8IF9aL<6!3(i{2aieqp`oCJMSVdzw$m>M z6;s3?8g$c2$?glG*pZk&aMsN#B?tVcLV*L(z*)saC^zJf1x`O?)Q(Gg2IWv+^jK)H zqg}DVQ-c0TrtiD7vSPQlXUG2CzN7mN_r27!-`BVM@PU0^NE|wRaOdtLNA`L5JOoZr z@KAG99`r|o0UWS)VG&;Qtw63*L8_k?seVSJ-=O+gkpXHFS)iO~0%{gHpcc^#l)q?k z^NJ&HJouggtxN3I1V6~B`ELNZPQ~;xjU$)nSca1nH3E-*7hf8#OUwv$Mw2x)i_j6~ zj5+5vlN_Oqa)+sRGtUFQQ+H{v+a#4h3&{pVvHE2>D8>4u;h<6~`3J>l#OII2Bw&b1 z4u%I5YXmr=zMf(^uQF4{!8SBBJUzMdxf3rzd-=rSs2FWH;18S*4u;MKPYhi=aUc|s zqEL1&6bQ;ET7rY;PV5hzJRyf-!R8@gx&IVo2cqGqRWLBQ9mr*BcB%92qY3M5#aC`y@9dhYXiHdTc;RZ< zXxZ4Zw6iJcY#O(wTYHnOy_3$~X@2iKw4(CZh7(WN_H#%qz_wymGP$fAVRTlqBdifC zMLNqKV73KEk$(ZRQGg1rGkie?+=@(y8qlAJ>~^zWg*zP#hof#<`U+HWb2-fC&P%`z z2`5b{3HZZdpI(?4xaKu z>KOJIZ#3poel2V*?*WqKRdwA>bHb*w?7&pTbI7u?tF}?wwNokHrLoVJu3n$4UZ1Y+ zN>+E>v)VWMaC;jW8p$#=tJ>60*>MLAfjh zq=bK38=3-^qOuiSz*4fd9#iYID(Tmc0NEi25xPyy>I^H>L5jsaxHSAPccGWD=qxfckqFT*#X^34|# z^sKFBtZgikvNeq}GlKo<-qF3+UQP+FPg%;|{2-%(^ez4dKg}=yl!MGkThmjy6uSCB zMb9#JVj0)dV98@7PC8*^73kYy83}M`pf^69;W3>tS=Ix)a2y3!ECa`s4y7mzF$7R@ zjnG*=hZ56>=}ehcmi_%KF)0)q0|BVe7Ug`wxR_b2rB(1GH!mS9DE1zJ@X*QOSa7!_ zMJ1)g=L-UJC7(|zl|y3C7aSM}24X~n6hU1A4*NrLP|nH{AcYPC5|JTG@IoLoI1p7) z0F?w$AOizVlqG`E-e^Wb4)_NF(Lud{G>>mJ=B{4r z+0EPFx!Qv5aK2rWFwgMztHP*o#g5B>`HpF-qM5ARX}&tiSC1{b<+|azQ#sAI6yv?l zm+wQbxqQhk$r=YVti417r*yoBW|>Q?2b|LRQDm9v0HU=5>&+0^LG?2bf}E1jKyjNT z9InbQvls*`__V5gotTY`lDdFswwAisHb0v)YBxdAr?NJ_{qypspo2{~03@J|4%a&u zZpNn^>r@uOqI{)nQ7rPhOBCMcL!%q;$LjM%qT+BE(}K^3QmK|mb@<9Fkst_2Xo^et zZ0R{5cPYYdG7HiY%tcZOKQdy|8`PYM<@V8YRt~A1<5tYs&D@%~auc^LbU9FVm zxI}%Rv3X0Tt_xqpczsWoTd;kMJrUN(=g3-3!~)VLS)X;#6V?dPvdbtjUKuYsiT>eO zR0_($7@!+!oFpJYEzukjQQ84_8GA?bSj6aXOfiwh8=$&@#=zERQ(Ak z*^aF;22RG8C|mO8xrAxfQ8(5<-kEZAqbdj)V5%a(RFmR6bMn$ochc#;(>cwz5&?W# zr+GXrIlfK{cs#H=N4!rP7Ap1+g@{L#C%iB~3ejmEnSGtQNE2j{w!t0v#I#RF??hdn zpYmJ6{>VwuzpeO%wURwrSC^@Kjm79{hGJYM=6?(%7hSW4O&3_I#aIj8tu6r|85x&} z__jK>LYJA#c&*b~kyu2Yg|Bd}A!TqU)ve#TaF?vF^i?6ixnAr(4*He~ni0CZ2 z1N|?CB_i`4t4-yfj4dYcNr$3x=t3_zrK@4&MI2HxMoX_h3=xDwP(^wXNd8(_lfxTQ z@!-IK#+`M~;7&nZFLt~c2&{3oCD-L-Tunt*N zf7LhY%dn`Uhi4~vuE&wjB9wfhHG3D~y5wp{ag=C;@nx(C${>aWrT83jiQtgHE{uUWc5C}%=&P=ujwA1^@<{EjfXYAQ>EtS_hDqz6{y`%URTxTi@GT>HZi^3Wv zZ=b^YPX_MN(#uf8ZOzf{osgk8NNB?c0WDuZjb5l?)BuIjOIX8<0gs?a77}SWW*3Xt z5)?5K(jb~Gqg2IL8p&2cr9S9SJ`V)l7&Q)si^h~=ZNjW7zZneNm1};qF=0;e4KvGX z6Xq+z1J(4s_EL(k1xGn|VM#lilg{R}voq=JOz~ZLOH5nR*_L*Cl1|UP(;qVTUYq9k z7o)c>he!Wpa!8~yyAm*G3}miADh=G|1>~k2CcEIulR&im3a$m@8_kP?>kJy21(%^z z_7ZWFUV*yq(gK;3`XQG{p<>ZwvH`PD9Hrw((5P8R7KwvR#B5hID0`#ovP2}(y`V%E zR{a`0K*lKaV&$?9<#NX9f?xIp#53(w1+Y8m>`w8Ub3_v@B292NC7qk@t-ZhFo=Xqa z>2L+(L7*Wp`3MrZD_OBQp|KTAQhJ2VfkeGO%@T)FJr6>8Y!(uWdxg@hXtIW)shchY zFpWgh&=@{rAeu#nt_ou%s%=ZG+}B_tloDe2`T(hji3@U&^XjUfKRul<@oYK!6~W|fi7f(So8qv-s!t}bjr~PX0NjWC1SSrD>n&( zRnOEyZi=tg!EHfdL4!E$Y)v{_$KBKXmSUvRMPUpcdmSUF8H*8a^-HAR%U{2Y>adHb zLyPQtCe58Aj5X{jzZL}6Id2~(F;H>Q$)91N#1okHp#nEd{Z`Km}E3(Gh*Xv*4Av+@I4M{P4nA|CtY72#^FJW z)V~EnCs}uZOsA2FPa8RZhK7X!7#r?#(v@@1<(Za!B1})Ziwu_|d4+ zlw^gz;`a=S5XZ!Q4+6UKp)hP(;+|%_y-;gDyA49F{5+5XHRA1Qz9Gps++LO9-Gosj z$hG=b%Z-+FO-r%{Xm_%v`(F3`!}nUI`4@^ghk-#vx%rnta2Lxt1N@`2NO}skvwui~ zgU0NlUX8;xBp|J!v1SmIG%__cJO+87jgWOj_emB+*2O>H0Bv_sS4{!t6cu1z1!=_i zyvX40)?3V#5)1g$28mJ@b$r;HWNIz9; zwt3oHJ#FpXdm&3g;a2StT45o>owLR$R$$PqJ+9 zjKewBJz2KucN}Z8KeoVg{xHW=<@;zEotzKs!j3goge}z!Ww6Ykb;nAPWf$Y(l8OJh=kgt&IyHb62vaYntCwjLCXj=4+n= zlfga8!C*s#Mw};Cwlp`Q1MN*(4H4qhqFJ=y(V=%Z#PVk$GvGYr;M;ucp^;AGl?a@ToC*iWBqFjjh|RJ_3NcM`fK?j8@~>kDeDJ~$Y}p3I zLS$_-K-Qh$Fvt)K>u5g~ik&`0riR!vkqAUY;ysH%*=Kk0BN&o=14zCC%{c3cTc88) zY(wMt+EjxFQMmp=W5-O+|Hq|f1m#C|s-f{#{6_q5Mn(@Nc3eIs7q_7r3 zZ3{s2*g)E~A?ezXc5P0&Hs4#GbZxnJdRo{E@Y~Q-tdyuc_%n}_o{PAe5a*x5(gYV& z=QDN)LA4BUSL>DR_$`G;f|G`iSH+_PQ+^;?02(b(IE15}w8;;3b|8;^x-Yo!o@`VK zor29j-1~^{G~)8-?Br)KDtRpsJ^HCoKx^jSuX+ zZn&+hi3q_q19SwzcxGU-#6}JvmfR}4Q3gj@vUY_t?Od01uAAo9 z!@dGEm=t#6DHU7yq_F8J+2a0*eMcqx55f+fo1l3}e!z1(93>yvESRq3cGTzZJLqax zSDAL6oEJZjq(wrq45y{Fvod(_wpevO#KC@-{@a(1gC*qrWRC|_ss9|Q*~`$lDaI1 zhke(+gJEx0_H5*S%0p^mBiFONWWr;C^al*LqcmSzcxZvLKL7VXa&3dGp=}`dH0lx_ z5K)cocPzxr)|VvcDsyRV!OUKhV`dxb1xUDok0x3mNRnCj@=RbiF@bqFDWRI5w1ip% z+{eli21DG;LwvQ+hoIajy(q$lp7OMlTncb>A ztlFl}_VJE; z+*JLx`~0-9?=fp=HR?F}2%i0IHdi(@u6~EjO3A1m%DO}+*a47`H~0ZH#po#(Q=;h| z2&-j-Zoqc~ltd>^<2%7b1KAj5jr?~AdNG-x7XSuIrf5M2N~I9`5=iI^!eS!-L;ai& zTC16`7Xx9h5^n&S=oiWno+6*G1oBt(b@&)%6uy$s;P(gy0Hp6hnR*6=mA(pTg@L19 zJjRTIgYc;U_R-LAImg)8Yy>dqe-IA_zAupXN26zlhl-)`6Xikd=ica6#}x zdJA*W!2_>_q!CFT;^@_KL>D(Jk--s$;|#)?MiS~lG*C)OSkVV&U|5#W2UbxbiuE#I z7;MDci%4z((N1VRuO4+=h7e#N1@SeD6qjFOpSXDf!ZEvK+1ncvJf7TodCIXmVg7Y_ z^JLkEU*^!q7L&`0ernAlD^*gK=AB92Iks}_Qi@l1RWLFS_>y*YedOwz5;i7XU1N^K zi;1s~_FY^4=CMgvS5nxh2-Uyin3CE*D8fb(`kQc;^>pxgp+7x3^j|PIBS6V_M9EfH z2Kq1Me4GI15&_Zi!qe!W#wLi_Lu?xs88G!PamH@x2&c7zj|!HW7WzQJ5K@dxFOkc_ zQ=uVPY>eqgv&`N;DF{QC_5g94S?u(hkFslW)FQC>9P{u%vCl zl6K>gowcuezTm~H>#8Ly;i?_hfG^PrL@9wIis~RUarHoS7*3y=RXc1K)*}Lc5>rFi z!W4*#K_Vq2Ql1Hw+ljLul>=~c1&ZS?BYF4f<^Ksi$)|vTl;<2Z8Lvz^U~6{PRx!Dx zEoEyTKQ+f#D(#6~^W~JY{+el~yz08}mN3>gUB2QG3t0~YZE-x7u5U@!x1{SgC+j!g zYfA}RKV3%IJJPoHq^CQK#blLtIfZMk@BkAsR)xJVaVS>0c-RaHR~xLrgmv@8lPZEmxz$s7xKra3!$gIPA>g&tkW?iGISS*k4< zAlxoumW*3{)+`yfTGT9k4iv5Q1~Ll+c{Q{jQ55E%2ZH9=C}VVXB0J?HsA_nU0xc1^ zvaBMX4))>avH|$gHim(8oB9b6gedVdF@kgR3vT@nj8r;+1nsDJjka##x$cg3qzfM3ChRgntNi2JhC#V6Fv9altYe&g9l(m%^NTdhVTFb zwXj+NRj(tIgh6`=^QA9)+-7(z@ro}~(>3Z5sQ5yIKaw~2-agex{}~J`??rIbdBTLQ z7=pteXXb104Uox~srwGqeXvW2KdODS8d8cmBtt~&qJ%cMVjc*EgM(2LDNrgh=giPK zfUlL#!zPu4yN7Db_YkHq@CzR+lCQ@|w2B0|G;3-cOYQkngCDfv#)oQU{S<0sO{WbM zDJ5(t$#r+(Q2%CKTAPmyMB4m;2NIs(7ewY zvY$D4Z+KI}di_3K*OIJjnW}3|32mPqp((rb>haOzs;zmYx2(63{Y4|!+iY>ueeNpB z13ZwnAlZtf7YQ0!(qSY=k(@x{LqdGk0ZfIE$Vi5f{5g_uA$b$YRV3d>^0!F-4#__w z*@gsxT@6`0k0~VhYojrQb3z$@ z)3$=+Tyu7wYnrPx11T4{7wEaVQm%Hc#?Ebc#uuiL3Gp{Tmo zf(onf!|-tv`WO;MDkU@KSu?A8ICPR|u~ND(G8C0!>gU6y*;4r18(Q-gazF?pr>f2) z2jEY8;2-cP#q~V%haA+{Alh#VN8p%fy21^|LJ$~mU~|MRGNmeK^Cuj>Tv9(6+$Q}i z)I-3Q-vIK6rD^)N)ap;Dm5&$`z4}+C#z$6`Zu-R3nKX5N%2RaLuc(c`qMT36Yv>JQ z{1Xa(JvmGVS-Sn&=_eHaesst}nS@0ABi8w`iT`FVI6|hH-#>btr}!mv6hp86t!dRG z4u1dlBY|Ym*#G(|(|UY5e6+p-2J$^~Pc3!B+S5c$)Ry+R*@?9@(x$SWZS2HW8tDH4 Df-|Os diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/initialise_test.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/initialise_test.cpython-312.pyc deleted file mode 100644 index d5c1d0308c24ae241b6639d0ce4fb553c29c8c46..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11643 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 diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/isatty_test.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/isatty_test.cpython-312.pyc deleted file mode 100644 index 78424c71ad780c830283efafaa093712b1c0ef72..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4770 zcmd^D-ER}w6~E*0*kgMfI|*VPGR`_1AecZB$O7At-IYV2yWIj+NwsS?l{M>`g25l@ zoe9(tyXph2g4h+UstTk{wOK7v8zs{A4_H+t>QnoIYa`PU5>oe};teG$A?2y(-0}De zC(v%6+AHPUd(XXd?#%g}^SdYhtKaV>(7ycBFVf#}6Y>>K>?TwP8y~{p7Ey`HWk>>h zM~2Hf5{@jN;5nl5sx#xv3JD?WO1NMysIH7VD<;IOC*jF@6W**Z;mb-13D(@#NW$+R z-zKVfji?^YvBF!YBs#Q!>bG-Jd@ACEQOZr=7a&h{Kiobw}??&y9zE^!PkOzvHE$| z%#!w}og>Q}d=Yhl+H|FL#V`s{&UESQOLVjs+5FLo@tGg!8r5gMpI7sd(@N@smP=pK zW@ZaBr_(8#hqcS;l%~%NYq`ra-%YQJp|%sQ zFK8sesU+c0IRKhh9Uz@54^mK_AYG~e(yh8ciYypB1u^O|y>GLxM`)XTjILom{v!+; zeQf*i1MX^D9~*Y?lB?WG^&Rb=BL;-J)oeRQ2uJ=COOOl=XhX1)0dk1w+!Y5$;$Ras zKjq`(3P%HQV$?|`=uEGo18&ANozs}(DSCd{VLEj!^J9|-Jai18NQ8Fbb`jx9LIBQp z*Bv!%D?4C|{u&5Ct>dK|mlpU3qI5HGJ+LS)b${Y7i+dmXcYGXwAO&xpy?%D_t)+9H zoGwfIpYo*RrJM2V@y!h%Jp4s(ted~vEyQH^|2hPI^r_E+;FOX9Bry1^7|>m~>dxhJ zm=;Yi_?X9G1(DdS_uS|_`wI-( zf%gGt)NYDKPHM+vRiJAe1}3Y`KpJ7K9qY9^zjgjBcYV}lao6^oCNNI3kisaa!$o^w z+LS`Qt-wGkoxKcTrd4fz?=~1U?0XrF9s^P1JBHe#cWLamff`J8mh0JfGrkc2G;lkx ziG$6Jet7mbQmlvntT*Ntz7%0}*Du6&dhT|)aoi)sUirVm4+*gWKXkA~i4hnz;n&xS zpH&#uAdGGUVe{7AX!>;!HKMdpJa#`c1TeNC$DRWSU?ntkFEnIfNr%8f6Hj6auvGgH zuKjkYhUrsJHYnAq?)N8%M7cEFALF=29I3mV!F z)!|mL{k$^v+2YrZw{|tb#afiYXH;8jtZf;mwQSltPQTHawJXzpGeYVYrV+HY7qN(L zY5$dPe$n%>C~Rrp{QTBDel~BL^U&8&OCd2?w{|Cv&{I}gvW!G(rh8HBLD8C);nuWl z{MTEZ-?vR(Htie*Ry`}elKg5#K3vgfI% zxwJcRWXi+;*&|GKxz9wsrkG6Tl&qFan%-nGn^)&DIF^#hpUf#4YlX__y;o4c&l-g8 zM{y9v2#P}>mI=k!U_~q4i6az9hC2HH8d-Nbg`suHDU7a5yl{Bk%fl4J=(<-Bj%{=Z z!b#3Dfe+$a;g2H8WHd+-MN^#1r47W3Vysdp>U;|PHe%$j;BO=B#|MhxU``ig+K9fq#-$ACc+%WP07j oaid?mqTEUDm!Yo-_D|m9h*Mga{~O=)cc=LCQ%`u}+{LW_3$w9zssI20 diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/utils.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/utils.cpython-312.pyc deleted file mode 100644 index c11fd06dcde2e4a8eb07e8fb0148dd086fe898fc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2459 zcmds2-A^1<6u)<7KW3MOQkE5iNSDxxi@F4jVz8QQsZyzIO~Dv8n~cNGNT;3M!Fz|& zZ6HCiXc|pQU&Mcbpowq3^g*6|p^1%Tz{JFw=-W0+QXxKg&dlrwl$iSJoy@uC{_eRS z=XdY9=f~#eFoAOShtKpOA0fYCqgwo)Qu!K`HKG!g8YE9Ml%g&eLSD>>jxHL$j1Mr; z@Mru286~Q(L{z^fZiwz2nSg@^01Y;vL5-?XiEMZaWh7{a8rmUfha1{qXg4*qn~Gv0 z@+Xc?P`f2(7A)2GX5PJ&&Uq_174;gGHc-}xMlw_-89}8bk`aqS zLbO9LP*d`glT*2>adiT8r~*vEz#>fu>B)7nPHi98jE_0m#t%Lm8P0ydHO8}-P1TH# zE4eFLLBFPDXN%c!J;zMwyRPRno=s|n>)H48=`7bR?WI{IcMUeqvpLf+nUYrqER9=y zV9wGFo}4Y(0i7$BRlM0++g;K{La;8!(^?|iC!!TlNP698OR`*0@|rB$VOh?b>YRak zM3z69Qw+CHR}Q7_ zws5OAqm46p?&HpD?z@{zvlal4Iu=C?1y0rASj=?{dk&qq06;JNm5)GNBoxlv0?jQ~ zvG+(>qzerea_#P@e*J(N-*n-JzAD&G@U{AyBw$Ehx49B(N zafb&xU~)8Ioq)B%&*LEeAb%~92Tjelhi(lmo!@S0E1fHi?)XBEwlccU@yag0n54%$ z_zm2*eu8ncYVo$-s5U3(Aay!5+)ts}OT4QWo zL=EHDZbI^8ue;U&8;)TduQD2D+N`1EGARAr~zY5`R;gfU>4)<8Z@P-Ts--dTrA8*(1^YQz6 zlN7;;LBi?0K*1cKB9! zE82TcI{nBG;8yqW0d#DIy8h-b!@SFh)N6DjlIj<~_our3cO}p^eVqVqcKcI(q0QbP zBK`iSD3?C!semYLOIm~(Q@tLf9?YMO$FC0i`-R$@l z8w6uKa9#va7Ad8_lQX}NjvW%-A)&p1K-*WPJp$F9B+%IE>!`=d5)oP-3O*XGh@h2G gD@PnGNT2mE(8_Hd)>FfJJj_qeIM_o0G$$SX1LhFd6aWAK diff --git a/.env/Lib/site-packages/colorama/tests/__pycache__/winterm_test.cpython-312.pyc b/.env/Lib/site-packages/colorama/tests/__pycache__/winterm_test.cpython-312.pyc deleted file mode 100644 index 9aaf5728589d31ec77e340ba3ff3f59f3c1f17a0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6601 zcmeHMO>7&-6`uVexg?h~YwO>xOeHpK29adTsqNN54M|oM6_(UWG+Yxx7Zi78k>-yw zyRt$;26}J<2{m9DIk-jx)IBAF1H=Uiv_Nxc&%KD10;&raNfBSvlTPPh|IPB6}3} z5m)9O3n~X>;l@cqsK641dsNw!A$caxyi63~7@>aPhy1-&{@2Kzt{L(+ca6+25$_tV z8j5hb5LHyIH?Ar=pNS{4x=+24oVt=tE2cCCrdjD`JUy*Y{PDU!uBr;v`rew3r*)S!k)sM!<8d%ux1^$h zF7i6BD(MN`n*oCcU7qRE1#IG`I*&S3@$AD;p8u%#J7ZUXp*l8_lXKz0cw$n?CT}QX zQ~9yMWP;{k>}E2dsAC;U_U72->jE|g zvPrE_2a=%Mh~kLZdjRB6$ASEqJU!I-?%6qE^=SK?up%5;ef`p$wb{Hh?9gr=g<^s~BOv^;2e_=6SU#9n=e9~^!d|3o;s3A+%T0|5&ms=PIwq>35_ zndIDzny`=wb5KPFn%@B=IHH*WG&5B+8()?^7+)5S0ecpbQGjv*^gAHgL^?Q33xk@4 zZ6`-*m*Pcs+MTF8l?S7+Of&w z;6H4myGpC=D&yAQ1O9{?qIkq>$TQ+MJOOQlHf=)EjHCgG?!B&PQXK4Dr3gFSeXZ~E z<>9NkXRy1!Zz$SLF`?1jJtG7CZ$=kcdKg!>0#T8DI82Y?xQONnZV4TRT4<^@Tyq4s zRsRA6jJIr`pVc?a@ppWym8zi;zWjCrZGF(X+#LDf^or2_wEmm-rF)Vw-)7e386Tei z=;GqV72yKRD%Bq^)*s(8Z&eK49hx6n7LPm;>+TNE55G5AYV9btc04{xZ55^XF18t zvmlp(FLYH#F}^U^RKGCb7zG04A+uerO=n|-x%z6q+1eqjZVivl>m1C&9v;C(IInXu z7+2M-cU)&A>89M}I)&CyQ)G9Q7nqc1Xe|OW*8T-$4gEjG6QgLC(THdSYKrGZ(T7l! zxDtR>FtJ2AVAt57?uD3w2AEEl&C#8JrXtf8SZ&2zn~ja&^zL+eyN#J6ow)LSAYU#( zK0SQ&=V#~m6`^Uh9*qx7v0ODYp%Hv0hYb7gC>hf_>7jBxrlXgFF6qgf8wM5R5X)C5$sw{!Pm5@Wnr6zXd;7^C}#z(?K5 zc*>~@8jo+@x6pMbXg40bQjD(%k-Z(El^4Ta*1e(?T$Q6E{4b;p5Y_`AOHDD|uHiiK zPF7RkzTyy^Q&tXJoendda&9h#*=kKOcb#C9WWsh|hv3YXo#$?G*bO&183L{0O^xKY zan!m=N0qmY?A5qq>s}1QQk-5_b@yoW((siiJr1%sh2@}u8jDY%WgB#?u>`PCDG*y| z4)1(-ebUhK?nEUBmqYK0&|;x#Rje-=Cbf`A}f^c0yo-?fU&Sf zG4j%0AO@oPMn-l-RKr$84QEGSeFuTbR?ink9MMuHiS&Z&7-7pBs24RV|eJT+@^s;M(cZIZr% zle&hdl5UtV?{+h9Ge$R&T6EcJ*$shg|&(&b`E}8|{r(h}&=S3AxT!_&{sp z*lv7;@>FC4&5gSM++0gK(dDa12d#}GHo47_4HjC4V=xX2b%u}pk`xJ1Ofqy~8lK@0 zj1<#tirGb3RRMFL178WtHy_ld?jOiZ<>0N6QuqUa$3fi#UKd{wjpqg9HKCl?8cO;e zviv@hHYE6#@LYJkjF<1ytMCcRBkDCE>m0){pA+Awq_apm*LjX<`;?FTLm*67k(_?! zrVQg-IQfiVy`JI-FU}SI#5Mn!7k=`_I()YLbA5&*!lAzt23LlQd>C1_Jy$V&EeIFL O?*iR^?hif&=)VDjqJ!}O diff --git a/.env/Lib/site-packages/colorama/tests/ansi_test.py b/.env/Lib/site-packages/colorama/tests/ansi_test.py deleted file mode 100644 index 0a20c80..0000000 --- a/.env/Lib/site-packages/colorama/tests/ansi_test.py +++ /dev/null @@ -1,76 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import sys -from unittest import TestCase, main - -from ..ansi import Back, Fore, Style -from ..ansitowin32 import AnsiToWin32 - -stdout_orig = sys.stdout -stderr_orig = sys.stderr - - -class AnsiTest(TestCase): - - def setUp(self): - # sanity check: stdout should be a file or StringIO object. - # It will only be AnsiToWin32 if init() has previously wrapped it - self.assertNotEqual(type(sys.stdout), AnsiToWin32) - self.assertNotEqual(type(sys.stderr), AnsiToWin32) - - def tearDown(self): - sys.stdout = stdout_orig - sys.stderr = stderr_orig - - - def testForeAttributes(self): - self.assertEqual(Fore.BLACK, '\033[30m') - self.assertEqual(Fore.RED, '\033[31m') - self.assertEqual(Fore.GREEN, '\033[32m') - self.assertEqual(Fore.YELLOW, '\033[33m') - self.assertEqual(Fore.BLUE, '\033[34m') - self.assertEqual(Fore.MAGENTA, '\033[35m') - self.assertEqual(Fore.CYAN, '\033[36m') - self.assertEqual(Fore.WHITE, '\033[37m') - self.assertEqual(Fore.RESET, '\033[39m') - - # Check the light, extended versions. - self.assertEqual(Fore.LIGHTBLACK_EX, '\033[90m') - self.assertEqual(Fore.LIGHTRED_EX, '\033[91m') - self.assertEqual(Fore.LIGHTGREEN_EX, '\033[92m') - self.assertEqual(Fore.LIGHTYELLOW_EX, '\033[93m') - self.assertEqual(Fore.LIGHTBLUE_EX, '\033[94m') - self.assertEqual(Fore.LIGHTMAGENTA_EX, '\033[95m') - self.assertEqual(Fore.LIGHTCYAN_EX, '\033[96m') - self.assertEqual(Fore.LIGHTWHITE_EX, '\033[97m') - - - def testBackAttributes(self): - self.assertEqual(Back.BLACK, '\033[40m') - self.assertEqual(Back.RED, '\033[41m') - self.assertEqual(Back.GREEN, '\033[42m') - self.assertEqual(Back.YELLOW, '\033[43m') - self.assertEqual(Back.BLUE, '\033[44m') - self.assertEqual(Back.MAGENTA, '\033[45m') - self.assertEqual(Back.CYAN, '\033[46m') - self.assertEqual(Back.WHITE, '\033[47m') - self.assertEqual(Back.RESET, '\033[49m') - - # Check the light, extended versions. - self.assertEqual(Back.LIGHTBLACK_EX, '\033[100m') - self.assertEqual(Back.LIGHTRED_EX, '\033[101m') - self.assertEqual(Back.LIGHTGREEN_EX, '\033[102m') - self.assertEqual(Back.LIGHTYELLOW_EX, '\033[103m') - self.assertEqual(Back.LIGHTBLUE_EX, '\033[104m') - self.assertEqual(Back.LIGHTMAGENTA_EX, '\033[105m') - self.assertEqual(Back.LIGHTCYAN_EX, '\033[106m') - self.assertEqual(Back.LIGHTWHITE_EX, '\033[107m') - - - def testStyleAttributes(self): - self.assertEqual(Style.DIM, '\033[2m') - self.assertEqual(Style.NORMAL, '\033[22m') - self.assertEqual(Style.BRIGHT, '\033[1m') - - -if __name__ == '__main__': - main() diff --git a/.env/Lib/site-packages/colorama/tests/ansitowin32_test.py b/.env/Lib/site-packages/colorama/tests/ansitowin32_test.py deleted file mode 100644 index 91ca551..0000000 --- a/.env/Lib/site-packages/colorama/tests/ansitowin32_test.py +++ /dev/null @@ -1,294 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -from io import StringIO, TextIOWrapper -from unittest import TestCase, main -try: - from contextlib import ExitStack -except ImportError: - # python 2 - from contextlib2 import ExitStack - -try: - from unittest.mock import MagicMock, Mock, patch -except ImportError: - from mock import MagicMock, Mock, patch - -from ..ansitowin32 import AnsiToWin32, StreamWrapper -from ..win32 import ENABLE_VIRTUAL_TERMINAL_PROCESSING -from .utils import osname - - -class StreamWrapperTest(TestCase): - - def testIsAProxy(self): - mockStream = Mock() - wrapper = StreamWrapper(mockStream, None) - self.assertTrue( wrapper.random_attr is mockStream.random_attr ) - - def testDelegatesWrite(self): - mockStream = Mock() - mockConverter = Mock() - wrapper = StreamWrapper(mockStream, mockConverter) - wrapper.write('hello') - self.assertTrue(mockConverter.write.call_args, (('hello',), {})) - - def testDelegatesContext(self): - mockConverter = Mock() - s = StringIO() - with StreamWrapper(s, mockConverter) as fp: - fp.write(u'hello') - self.assertTrue(s.closed) - - def testProxyNoContextManager(self): - mockStream = MagicMock() - mockStream.__enter__.side_effect = AttributeError() - mockConverter = Mock() - with self.assertRaises(AttributeError) as excinfo: - with StreamWrapper(mockStream, mockConverter) as wrapper: - wrapper.write('hello') - - def test_closed_shouldnt_raise_on_closed_stream(self): - stream = StringIO() - stream.close() - wrapper = StreamWrapper(stream, None) - self.assertEqual(wrapper.closed, True) - - def test_closed_shouldnt_raise_on_detached_stream(self): - stream = TextIOWrapper(StringIO()) - stream.detach() - wrapper = StreamWrapper(stream, None) - self.assertEqual(wrapper.closed, True) - -class AnsiToWin32Test(TestCase): - - def testInit(self): - mockStdout = Mock() - auto = Mock() - stream = AnsiToWin32(mockStdout, autoreset=auto) - self.assertEqual(stream.wrapped, mockStdout) - self.assertEqual(stream.autoreset, auto) - - @patch('colorama.ansitowin32.winterm', None) - @patch('colorama.ansitowin32.winapi_test', lambda *_: True) - def testStripIsTrueOnWindows(self): - with osname('nt'): - mockStdout = Mock() - stream = AnsiToWin32(mockStdout) - self.assertTrue(stream.strip) - - def testStripIsFalseOffWindows(self): - with osname('posix'): - mockStdout = Mock(closed=False) - stream = AnsiToWin32(mockStdout) - self.assertFalse(stream.strip) - - def testWriteStripsAnsi(self): - mockStdout = Mock() - stream = AnsiToWin32(mockStdout) - stream.wrapped = Mock() - stream.write_and_convert = Mock() - stream.strip = True - - stream.write('abc') - - self.assertFalse(stream.wrapped.write.called) - self.assertEqual(stream.write_and_convert.call_args, (('abc',), {})) - - def testWriteDoesNotStripAnsi(self): - mockStdout = Mock() - stream = AnsiToWin32(mockStdout) - stream.wrapped = Mock() - stream.write_and_convert = Mock() - stream.strip = False - stream.convert = False - - stream.write('abc') - - self.assertFalse(stream.write_and_convert.called) - self.assertEqual(stream.wrapped.write.call_args, (('abc',), {})) - - def assert_autoresets(self, convert, autoreset=True): - stream = AnsiToWin32(Mock()) - stream.convert = convert - stream.reset_all = Mock() - stream.autoreset = autoreset - stream.winterm = Mock() - - stream.write('abc') - - self.assertEqual(stream.reset_all.called, autoreset) - - def testWriteAutoresets(self): - self.assert_autoresets(convert=True) - self.assert_autoresets(convert=False) - self.assert_autoresets(convert=True, autoreset=False) - self.assert_autoresets(convert=False, autoreset=False) - - def testWriteAndConvertWritesPlainText(self): - stream = AnsiToWin32(Mock()) - stream.write_and_convert( 'abc' ) - self.assertEqual( stream.wrapped.write.call_args, (('abc',), {}) ) - - def testWriteAndConvertStripsAllValidAnsi(self): - stream = AnsiToWin32(Mock()) - stream.call_win32 = Mock() - data = [ - 'abc\033[mdef', - 'abc\033[0mdef', - 'abc\033[2mdef', - 'abc\033[02mdef', - 'abc\033[002mdef', - 'abc\033[40mdef', - 'abc\033[040mdef', - 'abc\033[0;1mdef', - 'abc\033[40;50mdef', - 'abc\033[50;30;40mdef', - 'abc\033[Adef', - 'abc\033[0Gdef', - 'abc\033[1;20;128Hdef', - ] - for datum in data: - stream.wrapped.write.reset_mock() - stream.write_and_convert( datum ) - self.assertEqual( - [args[0] for args in stream.wrapped.write.call_args_list], - [ ('abc',), ('def',) ] - ) - - def testWriteAndConvertSkipsEmptySnippets(self): - stream = AnsiToWin32(Mock()) - stream.call_win32 = Mock() - stream.write_and_convert( '\033[40m\033[41m' ) - self.assertFalse( stream.wrapped.write.called ) - - def testWriteAndConvertCallsWin32WithParamsAndCommand(self): - stream = AnsiToWin32(Mock()) - stream.convert = True - stream.call_win32 = Mock() - stream.extract_params = Mock(return_value='params') - data = { - 'abc\033[adef': ('a', 'params'), - 'abc\033[;;bdef': ('b', 'params'), - 'abc\033[0cdef': ('c', 'params'), - 'abc\033[;;0;;Gdef': ('G', 'params'), - 'abc\033[1;20;128Hdef': ('H', 'params'), - } - for datum, expected in data.items(): - stream.call_win32.reset_mock() - stream.write_and_convert( datum ) - self.assertEqual( stream.call_win32.call_args[0], expected ) - - def test_reset_all_shouldnt_raise_on_closed_orig_stdout(self): - stream = StringIO() - converter = AnsiToWin32(stream) - stream.close() - - converter.reset_all() - - def test_wrap_shouldnt_raise_on_closed_orig_stdout(self): - stream = StringIO() - stream.close() - with \ - patch("colorama.ansitowin32.os.name", "nt"), \ - patch("colorama.ansitowin32.winapi_test", lambda: True): - converter = AnsiToWin32(stream) - self.assertTrue(converter.strip) - self.assertFalse(converter.convert) - - def test_wrap_shouldnt_raise_on_missing_closed_attr(self): - with \ - patch("colorama.ansitowin32.os.name", "nt"), \ - patch("colorama.ansitowin32.winapi_test", lambda: True): - converter = AnsiToWin32(object()) - self.assertTrue(converter.strip) - self.assertFalse(converter.convert) - - def testExtractParams(self): - stream = AnsiToWin32(Mock()) - data = { - '': (0,), - ';;': (0,), - '2': (2,), - ';;002;;': (2,), - '0;1': (0, 1), - ';;003;;456;;': (3, 456), - '11;22;33;44;55': (11, 22, 33, 44, 55), - } - for datum, expected in data.items(): - self.assertEqual(stream.extract_params('m', datum), expected) - - def testCallWin32UsesLookup(self): - listener = Mock() - stream = AnsiToWin32(listener) - stream.win32_calls = { - 1: (lambda *_, **__: listener(11),), - 2: (lambda *_, **__: listener(22),), - 3: (lambda *_, **__: listener(33),), - } - stream.call_win32('m', (3, 1, 99, 2)) - self.assertEqual( - [a[0][0] for a in listener.call_args_list], - [33, 11, 22] ) - - def test_osc_codes(self): - mockStdout = Mock() - stream = AnsiToWin32(mockStdout, convert=True) - with patch('colorama.ansitowin32.winterm') as winterm: - data = [ - '\033]0\x07', # missing arguments - '\033]0;foo\x08', # wrong OSC command - '\033]0;colorama_test_title\x07', # should work - '\033]1;colorama_test_title\x07', # wrong set command - '\033]2;colorama_test_title\x07', # should work - '\033]' + ';' * 64 + '\x08', # see issue #247 - ] - for code in data: - stream.write(code) - self.assertEqual(winterm.set_title.call_count, 2) - - def test_native_windows_ansi(self): - with ExitStack() as stack: - def p(a, b): - stack.enter_context(patch(a, b, create=True)) - # Pretend to be on Windows - p("colorama.ansitowin32.os.name", "nt") - p("colorama.ansitowin32.winapi_test", lambda: True) - p("colorama.win32.winapi_test", lambda: True) - p("colorama.winterm.win32.windll", "non-None") - p("colorama.winterm.get_osfhandle", lambda _: 1234) - - # Pretend that our mock stream has native ANSI support - p( - "colorama.winterm.win32.GetConsoleMode", - lambda _: ENABLE_VIRTUAL_TERMINAL_PROCESSING, - ) - SetConsoleMode = Mock() - p("colorama.winterm.win32.SetConsoleMode", SetConsoleMode) - - stdout = Mock() - stdout.closed = False - stdout.isatty.return_value = True - stdout.fileno.return_value = 1 - - # Our fake console says it has native vt support, so AnsiToWin32 should - # enable that support and do nothing else. - stream = AnsiToWin32(stdout) - SetConsoleMode.assert_called_with(1234, ENABLE_VIRTUAL_TERMINAL_PROCESSING) - self.assertFalse(stream.strip) - self.assertFalse(stream.convert) - self.assertFalse(stream.should_wrap()) - - # Now let's pretend we're on an old Windows console, that doesn't have - # native ANSI support. - p("colorama.winterm.win32.GetConsoleMode", lambda _: 0) - SetConsoleMode = Mock() - p("colorama.winterm.win32.SetConsoleMode", SetConsoleMode) - - stream = AnsiToWin32(stdout) - SetConsoleMode.assert_called_with(1234, ENABLE_VIRTUAL_TERMINAL_PROCESSING) - self.assertTrue(stream.strip) - self.assertTrue(stream.convert) - self.assertTrue(stream.should_wrap()) - - -if __name__ == '__main__': - main() diff --git a/.env/Lib/site-packages/colorama/tests/initialise_test.py b/.env/Lib/site-packages/colorama/tests/initialise_test.py deleted file mode 100644 index 89f9b07..0000000 --- a/.env/Lib/site-packages/colorama/tests/initialise_test.py +++ /dev/null @@ -1,189 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import sys -from unittest import TestCase, main, skipUnless - -try: - from unittest.mock import patch, Mock -except ImportError: - from mock import patch, Mock - -from ..ansitowin32 import StreamWrapper -from ..initialise import init, just_fix_windows_console, _wipe_internal_state_for_tests -from .utils import osname, replace_by - -orig_stdout = sys.stdout -orig_stderr = sys.stderr - - -class InitTest(TestCase): - - @skipUnless(sys.stdout.isatty(), "sys.stdout is not a tty") - def setUp(self): - # sanity check - self.assertNotWrapped() - - def tearDown(self): - _wipe_internal_state_for_tests() - sys.stdout = orig_stdout - sys.stderr = orig_stderr - - def assertWrapped(self): - self.assertIsNot(sys.stdout, orig_stdout, 'stdout should be wrapped') - self.assertIsNot(sys.stderr, orig_stderr, 'stderr should be wrapped') - self.assertTrue(isinstance(sys.stdout, StreamWrapper), - 'bad stdout wrapper') - self.assertTrue(isinstance(sys.stderr, StreamWrapper), - 'bad stderr wrapper') - - def assertNotWrapped(self): - self.assertIs(sys.stdout, orig_stdout, 'stdout should not be wrapped') - self.assertIs(sys.stderr, orig_stderr, 'stderr should not be wrapped') - - @patch('colorama.initialise.reset_all') - @patch('colorama.ansitowin32.winapi_test', lambda *_: True) - @patch('colorama.ansitowin32.enable_vt_processing', lambda *_: False) - def testInitWrapsOnWindows(self, _): - with osname("nt"): - init() - self.assertWrapped() - - @patch('colorama.initialise.reset_all') - @patch('colorama.ansitowin32.winapi_test', lambda *_: False) - def testInitDoesntWrapOnEmulatedWindows(self, _): - with osname("nt"): - init() - self.assertNotWrapped() - - def testInitDoesntWrapOnNonWindows(self): - with osname("posix"): - init() - self.assertNotWrapped() - - def testInitDoesntWrapIfNone(self): - with replace_by(None): - init() - # We can't use assertNotWrapped here because replace_by(None) - # changes stdout/stderr already. - self.assertIsNone(sys.stdout) - self.assertIsNone(sys.stderr) - - def testInitAutoresetOnWrapsOnAllPlatforms(self): - with osname("posix"): - init(autoreset=True) - self.assertWrapped() - - def testInitWrapOffDoesntWrapOnWindows(self): - with osname("nt"): - init(wrap=False) - self.assertNotWrapped() - - def testInitWrapOffIncompatibleWithAutoresetOn(self): - self.assertRaises(ValueError, lambda: init(autoreset=True, wrap=False)) - - @patch('colorama.win32.SetConsoleTextAttribute') - @patch('colorama.initialise.AnsiToWin32') - def testAutoResetPassedOn(self, mockATW32, _): - with osname("nt"): - init(autoreset=True) - self.assertEqual(len(mockATW32.call_args_list), 2) - self.assertEqual(mockATW32.call_args_list[1][1]['autoreset'], True) - self.assertEqual(mockATW32.call_args_list[0][1]['autoreset'], True) - - @patch('colorama.initialise.AnsiToWin32') - def testAutoResetChangeable(self, mockATW32): - with osname("nt"): - init() - - init(autoreset=True) - self.assertEqual(len(mockATW32.call_args_list), 4) - self.assertEqual(mockATW32.call_args_list[2][1]['autoreset'], True) - self.assertEqual(mockATW32.call_args_list[3][1]['autoreset'], True) - - init() - self.assertEqual(len(mockATW32.call_args_list), 6) - self.assertEqual( - mockATW32.call_args_list[4][1]['autoreset'], False) - self.assertEqual( - mockATW32.call_args_list[5][1]['autoreset'], False) - - - @patch('colorama.initialise.atexit.register') - def testAtexitRegisteredOnlyOnce(self, mockRegister): - init() - self.assertTrue(mockRegister.called) - mockRegister.reset_mock() - init() - self.assertFalse(mockRegister.called) - - -class JustFixWindowsConsoleTest(TestCase): - def _reset(self): - _wipe_internal_state_for_tests() - sys.stdout = orig_stdout - sys.stderr = orig_stderr - - def tearDown(self): - self._reset() - - @patch("colorama.ansitowin32.winapi_test", lambda: True) - def testJustFixWindowsConsole(self): - if sys.platform != "win32": - # just_fix_windows_console should be a no-op - just_fix_windows_console() - self.assertIs(sys.stdout, orig_stdout) - self.assertIs(sys.stderr, orig_stderr) - else: - def fake_std(): - # Emulate stdout=not a tty, stderr=tty - # to check that we handle both cases correctly - stdout = Mock() - stdout.closed = False - stdout.isatty.return_value = False - stdout.fileno.return_value = 1 - sys.stdout = stdout - - stderr = Mock() - stderr.closed = False - stderr.isatty.return_value = True - stderr.fileno.return_value = 2 - sys.stderr = stderr - - for native_ansi in [False, True]: - with patch( - 'colorama.ansitowin32.enable_vt_processing', - lambda *_: native_ansi - ): - self._reset() - fake_std() - - # Regular single-call test - prev_stdout = sys.stdout - prev_stderr = sys.stderr - just_fix_windows_console() - self.assertIs(sys.stdout, prev_stdout) - if native_ansi: - self.assertIs(sys.stderr, prev_stderr) - else: - self.assertIsNot(sys.stderr, prev_stderr) - - # second call without resetting is always a no-op - prev_stdout = sys.stdout - prev_stderr = sys.stderr - just_fix_windows_console() - self.assertIs(sys.stdout, prev_stdout) - self.assertIs(sys.stderr, prev_stderr) - - self._reset() - fake_std() - - # If init() runs first, just_fix_windows_console should be a no-op - init() - prev_stdout = sys.stdout - prev_stderr = sys.stderr - just_fix_windows_console() - self.assertIs(prev_stdout, sys.stdout) - self.assertIs(prev_stderr, sys.stderr) - - -if __name__ == '__main__': - main() diff --git a/.env/Lib/site-packages/colorama/tests/isatty_test.py b/.env/Lib/site-packages/colorama/tests/isatty_test.py deleted file mode 100644 index 0f84e4b..0000000 --- a/.env/Lib/site-packages/colorama/tests/isatty_test.py +++ /dev/null @@ -1,57 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import sys -from unittest import TestCase, main - -from ..ansitowin32 import StreamWrapper, AnsiToWin32 -from .utils import pycharm, replace_by, replace_original_by, StreamTTY, StreamNonTTY - - -def is_a_tty(stream): - return StreamWrapper(stream, None).isatty() - -class IsattyTest(TestCase): - - def test_TTY(self): - tty = StreamTTY() - self.assertTrue(is_a_tty(tty)) - with pycharm(): - self.assertTrue(is_a_tty(tty)) - - def test_nonTTY(self): - non_tty = StreamNonTTY() - self.assertFalse(is_a_tty(non_tty)) - with pycharm(): - self.assertFalse(is_a_tty(non_tty)) - - def test_withPycharm(self): - with pycharm(): - self.assertTrue(is_a_tty(sys.stderr)) - self.assertTrue(is_a_tty(sys.stdout)) - - def test_withPycharmTTYOverride(self): - tty = StreamTTY() - with pycharm(), replace_by(tty): - self.assertTrue(is_a_tty(tty)) - - def test_withPycharmNonTTYOverride(self): - non_tty = StreamNonTTY() - with pycharm(), replace_by(non_tty): - self.assertFalse(is_a_tty(non_tty)) - - def test_withPycharmNoneOverride(self): - with pycharm(): - with replace_by(None), replace_original_by(None): - self.assertFalse(is_a_tty(None)) - self.assertFalse(is_a_tty(StreamNonTTY())) - self.assertTrue(is_a_tty(StreamTTY())) - - def test_withPycharmStreamWrapped(self): - with pycharm(): - self.assertTrue(AnsiToWin32(StreamTTY()).stream.isatty()) - self.assertFalse(AnsiToWin32(StreamNonTTY()).stream.isatty()) - self.assertTrue(AnsiToWin32(sys.stdout).stream.isatty()) - self.assertTrue(AnsiToWin32(sys.stderr).stream.isatty()) - - -if __name__ == '__main__': - main() diff --git a/.env/Lib/site-packages/colorama/tests/utils.py b/.env/Lib/site-packages/colorama/tests/utils.py deleted file mode 100644 index 472fafb..0000000 --- a/.env/Lib/site-packages/colorama/tests/utils.py +++ /dev/null @@ -1,49 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -from contextlib import contextmanager -from io import StringIO -import sys -import os - - -class StreamTTY(StringIO): - def isatty(self): - return True - -class StreamNonTTY(StringIO): - def isatty(self): - return False - -@contextmanager -def osname(name): - orig = os.name - os.name = name - yield - os.name = orig - -@contextmanager -def replace_by(stream): - orig_stdout = sys.stdout - orig_stderr = sys.stderr - sys.stdout = stream - sys.stderr = stream - yield - sys.stdout = orig_stdout - sys.stderr = orig_stderr - -@contextmanager -def replace_original_by(stream): - orig_stdout = sys.__stdout__ - orig_stderr = sys.__stderr__ - sys.__stdout__ = stream - sys.__stderr__ = stream - yield - sys.__stdout__ = orig_stdout - sys.__stderr__ = orig_stderr - -@contextmanager -def pycharm(): - os.environ["PYCHARM_HOSTED"] = "1" - non_tty = StreamNonTTY() - with replace_by(non_tty), replace_original_by(non_tty): - yield - del os.environ["PYCHARM_HOSTED"] diff --git a/.env/Lib/site-packages/colorama/tests/winterm_test.py b/.env/Lib/site-packages/colorama/tests/winterm_test.py deleted file mode 100644 index d0955f9..0000000 --- a/.env/Lib/site-packages/colorama/tests/winterm_test.py +++ /dev/null @@ -1,131 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -import sys -from unittest import TestCase, main, skipUnless - -try: - from unittest.mock import Mock, patch -except ImportError: - from mock import Mock, patch - -from ..winterm import WinColor, WinStyle, WinTerm - - -class WinTermTest(TestCase): - - @patch('colorama.winterm.win32') - def testInit(self, mockWin32): - mockAttr = Mock() - mockAttr.wAttributes = 7 + 6 * 16 + 8 - mockWin32.GetConsoleScreenBufferInfo.return_value = mockAttr - term = WinTerm() - self.assertEqual(term._fore, 7) - self.assertEqual(term._back, 6) - self.assertEqual(term._style, 8) - - @skipUnless(sys.platform.startswith("win"), "requires Windows") - def testGetAttrs(self): - term = WinTerm() - - term._fore = 0 - term._back = 0 - term._style = 0 - self.assertEqual(term.get_attrs(), 0) - - term._fore = WinColor.YELLOW - self.assertEqual(term.get_attrs(), WinColor.YELLOW) - - term._back = WinColor.MAGENTA - self.assertEqual( - term.get_attrs(), - WinColor.YELLOW + WinColor.MAGENTA * 16) - - term._style = WinStyle.BRIGHT - self.assertEqual( - term.get_attrs(), - WinColor.YELLOW + WinColor.MAGENTA * 16 + WinStyle.BRIGHT) - - @patch('colorama.winterm.win32') - def testResetAll(self, mockWin32): - mockAttr = Mock() - mockAttr.wAttributes = 1 + 2 * 16 + 8 - mockWin32.GetConsoleScreenBufferInfo.return_value = mockAttr - term = WinTerm() - - term.set_console = Mock() - term._fore = -1 - term._back = -1 - term._style = -1 - - term.reset_all() - - self.assertEqual(term._fore, 1) - self.assertEqual(term._back, 2) - self.assertEqual(term._style, 8) - self.assertEqual(term.set_console.called, True) - - @skipUnless(sys.platform.startswith("win"), "requires Windows") - def testFore(self): - term = WinTerm() - term.set_console = Mock() - term._fore = 0 - - term.fore(5) - - self.assertEqual(term._fore, 5) - self.assertEqual(term.set_console.called, True) - - @skipUnless(sys.platform.startswith("win"), "requires Windows") - def testBack(self): - term = WinTerm() - term.set_console = Mock() - term._back = 0 - - term.back(5) - - self.assertEqual(term._back, 5) - self.assertEqual(term.set_console.called, True) - - @skipUnless(sys.platform.startswith("win"), "requires Windows") - def testStyle(self): - term = WinTerm() - term.set_console = Mock() - term._style = 0 - - term.style(22) - - self.assertEqual(term._style, 22) - self.assertEqual(term.set_console.called, True) - - @patch('colorama.winterm.win32') - def testSetConsole(self, mockWin32): - mockAttr = Mock() - mockAttr.wAttributes = 0 - mockWin32.GetConsoleScreenBufferInfo.return_value = mockAttr - term = WinTerm() - term.windll = Mock() - - term.set_console() - - self.assertEqual( - mockWin32.SetConsoleTextAttribute.call_args, - ((mockWin32.STDOUT, term.get_attrs()), {}) - ) - - @patch('colorama.winterm.win32') - def testSetConsoleOnStderr(self, mockWin32): - mockAttr = Mock() - mockAttr.wAttributes = 0 - mockWin32.GetConsoleScreenBufferInfo.return_value = mockAttr - term = WinTerm() - term.windll = Mock() - - term.set_console(on_stderr=True) - - self.assertEqual( - mockWin32.SetConsoleTextAttribute.call_args, - ((mockWin32.STDERR, term.get_attrs()), {}) - ) - - -if __name__ == '__main__': - main() diff --git a/.env/Lib/site-packages/colorama/win32.py b/.env/Lib/site-packages/colorama/win32.py deleted file mode 100644 index 841b0e2..0000000 --- a/.env/Lib/site-packages/colorama/win32.py +++ /dev/null @@ -1,180 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. - -# from winbase.h -STDOUT = -11 -STDERR = -12 - -ENABLE_VIRTUAL_TERMINAL_PROCESSING = 0x0004 - -try: - import ctypes - from ctypes import LibraryLoader - windll = LibraryLoader(ctypes.WinDLL) - from ctypes import wintypes -except (AttributeError, ImportError): - windll = None - SetConsoleTextAttribute = lambda *_: None - winapi_test = lambda *_: None -else: - from ctypes import byref, Structure, c_char, POINTER - - COORD = wintypes._COORD - - class CONSOLE_SCREEN_BUFFER_INFO(Structure): - """struct in wincon.h.""" - _fields_ = [ - ("dwSize", COORD), - ("dwCursorPosition", COORD), - ("wAttributes", wintypes.WORD), - ("srWindow", wintypes.SMALL_RECT), - ("dwMaximumWindowSize", COORD), - ] - def __str__(self): - return '(%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d)' % ( - self.dwSize.Y, self.dwSize.X - , self.dwCursorPosition.Y, self.dwCursorPosition.X - , self.wAttributes - , self.srWindow.Top, self.srWindow.Left, self.srWindow.Bottom, self.srWindow.Right - , self.dwMaximumWindowSize.Y, self.dwMaximumWindowSize.X - ) - - _GetStdHandle = windll.kernel32.GetStdHandle - _GetStdHandle.argtypes = [ - wintypes.DWORD, - ] - _GetStdHandle.restype = wintypes.HANDLE - - _GetConsoleScreenBufferInfo = windll.kernel32.GetConsoleScreenBufferInfo - _GetConsoleScreenBufferInfo.argtypes = [ - wintypes.HANDLE, - POINTER(CONSOLE_SCREEN_BUFFER_INFO), - ] - _GetConsoleScreenBufferInfo.restype = wintypes.BOOL - - _SetConsoleTextAttribute = windll.kernel32.SetConsoleTextAttribute - _SetConsoleTextAttribute.argtypes = [ - wintypes.HANDLE, - wintypes.WORD, - ] - _SetConsoleTextAttribute.restype = wintypes.BOOL - - _SetConsoleCursorPosition = windll.kernel32.SetConsoleCursorPosition - _SetConsoleCursorPosition.argtypes = [ - wintypes.HANDLE, - COORD, - ] - _SetConsoleCursorPosition.restype = wintypes.BOOL - - _FillConsoleOutputCharacterA = windll.kernel32.FillConsoleOutputCharacterA - _FillConsoleOutputCharacterA.argtypes = [ - wintypes.HANDLE, - c_char, - wintypes.DWORD, - COORD, - POINTER(wintypes.DWORD), - ] - _FillConsoleOutputCharacterA.restype = wintypes.BOOL - - _FillConsoleOutputAttribute = windll.kernel32.FillConsoleOutputAttribute - _FillConsoleOutputAttribute.argtypes = [ - wintypes.HANDLE, - wintypes.WORD, - wintypes.DWORD, - COORD, - POINTER(wintypes.DWORD), - ] - _FillConsoleOutputAttribute.restype = wintypes.BOOL - - _SetConsoleTitleW = windll.kernel32.SetConsoleTitleW - _SetConsoleTitleW.argtypes = [ - wintypes.LPCWSTR - ] - _SetConsoleTitleW.restype = wintypes.BOOL - - _GetConsoleMode = windll.kernel32.GetConsoleMode - _GetConsoleMode.argtypes = [ - wintypes.HANDLE, - POINTER(wintypes.DWORD) - ] - _GetConsoleMode.restype = wintypes.BOOL - - _SetConsoleMode = windll.kernel32.SetConsoleMode - _SetConsoleMode.argtypes = [ - wintypes.HANDLE, - wintypes.DWORD - ] - _SetConsoleMode.restype = wintypes.BOOL - - def _winapi_test(handle): - csbi = CONSOLE_SCREEN_BUFFER_INFO() - success = _GetConsoleScreenBufferInfo( - handle, byref(csbi)) - return bool(success) - - def winapi_test(): - return any(_winapi_test(h) for h in - (_GetStdHandle(STDOUT), _GetStdHandle(STDERR))) - - def GetConsoleScreenBufferInfo(stream_id=STDOUT): - handle = _GetStdHandle(stream_id) - csbi = CONSOLE_SCREEN_BUFFER_INFO() - success = _GetConsoleScreenBufferInfo( - handle, byref(csbi)) - return csbi - - def SetConsoleTextAttribute(stream_id, attrs): - handle = _GetStdHandle(stream_id) - return _SetConsoleTextAttribute(handle, attrs) - - def SetConsoleCursorPosition(stream_id, position, adjust=True): - position = COORD(*position) - # If the position is out of range, do nothing. - if position.Y <= 0 or position.X <= 0: - return - # Adjust for Windows' SetConsoleCursorPosition: - # 1. being 0-based, while ANSI is 1-based. - # 2. expecting (x,y), while ANSI uses (y,x). - adjusted_position = COORD(position.Y - 1, position.X - 1) - if adjust: - # Adjust for viewport's scroll position - sr = GetConsoleScreenBufferInfo(STDOUT).srWindow - adjusted_position.Y += sr.Top - adjusted_position.X += sr.Left - # Resume normal processing - handle = _GetStdHandle(stream_id) - return _SetConsoleCursorPosition(handle, adjusted_position) - - def FillConsoleOutputCharacter(stream_id, char, length, start): - handle = _GetStdHandle(stream_id) - char = c_char(char.encode()) - length = wintypes.DWORD(length) - num_written = wintypes.DWORD(0) - # Note that this is hard-coded for ANSI (vs wide) bytes. - success = _FillConsoleOutputCharacterA( - handle, char, length, start, byref(num_written)) - return num_written.value - - def FillConsoleOutputAttribute(stream_id, attr, length, start): - ''' FillConsoleOutputAttribute( hConsole, csbi.wAttributes, dwConSize, coordScreen, &cCharsWritten )''' - handle = _GetStdHandle(stream_id) - attribute = wintypes.WORD(attr) - length = wintypes.DWORD(length) - num_written = wintypes.DWORD(0) - # Note that this is hard-coded for ANSI (vs wide) bytes. - return _FillConsoleOutputAttribute( - handle, attribute, length, start, byref(num_written)) - - def SetConsoleTitle(title): - return _SetConsoleTitleW(title) - - def GetConsoleMode(handle): - mode = wintypes.DWORD() - success = _GetConsoleMode(handle, byref(mode)) - if not success: - raise ctypes.WinError() - return mode.value - - def SetConsoleMode(handle, mode): - success = _SetConsoleMode(handle, mode) - if not success: - raise ctypes.WinError() diff --git a/.env/Lib/site-packages/colorama/winterm.py b/.env/Lib/site-packages/colorama/winterm.py deleted file mode 100644 index aad867e..0000000 --- a/.env/Lib/site-packages/colorama/winterm.py +++ /dev/null @@ -1,195 +0,0 @@ -# Copyright Jonathan Hartley 2013. BSD 3-Clause license, see LICENSE file. -try: - from msvcrt import get_osfhandle -except ImportError: - def get_osfhandle(_): - raise OSError("This isn't windows!") - - -from . import win32 - -# from wincon.h -class WinColor(object): - BLACK = 0 - BLUE = 1 - GREEN = 2 - CYAN = 3 - RED = 4 - MAGENTA = 5 - YELLOW = 6 - GREY = 7 - -# from wincon.h -class WinStyle(object): - NORMAL = 0x00 # dim text, dim background - BRIGHT = 0x08 # bright text, dim background - BRIGHT_BACKGROUND = 0x80 # dim text, bright background - -class WinTerm(object): - - def __init__(self): - self._default = win32.GetConsoleScreenBufferInfo(win32.STDOUT).wAttributes - self.set_attrs(self._default) - self._default_fore = self._fore - self._default_back = self._back - self._default_style = self._style - # In order to emulate LIGHT_EX in windows, we borrow the BRIGHT style. - # So that LIGHT_EX colors and BRIGHT style do not clobber each other, - # we track them separately, since LIGHT_EX is overwritten by Fore/Back - # and BRIGHT is overwritten by Style codes. - self._light = 0 - - def get_attrs(self): - return self._fore + self._back * 16 + (self._style | self._light) - - def set_attrs(self, value): - self._fore = value & 7 - self._back = (value >> 4) & 7 - self._style = value & (WinStyle.BRIGHT | WinStyle.BRIGHT_BACKGROUND) - - def reset_all(self, on_stderr=None): - self.set_attrs(self._default) - self.set_console(attrs=self._default) - self._light = 0 - - def fore(self, fore=None, light=False, on_stderr=False): - if fore is None: - fore = self._default_fore - self._fore = fore - # Emulate LIGHT_EX with BRIGHT Style - if light: - self._light |= WinStyle.BRIGHT - else: - self._light &= ~WinStyle.BRIGHT - self.set_console(on_stderr=on_stderr) - - def back(self, back=None, light=False, on_stderr=False): - if back is None: - back = self._default_back - self._back = back - # Emulate LIGHT_EX with BRIGHT_BACKGROUND Style - if light: - self._light |= WinStyle.BRIGHT_BACKGROUND - else: - self._light &= ~WinStyle.BRIGHT_BACKGROUND - self.set_console(on_stderr=on_stderr) - - def style(self, style=None, on_stderr=False): - if style is None: - style = self._default_style - self._style = style - self.set_console(on_stderr=on_stderr) - - def set_console(self, attrs=None, on_stderr=False): - if attrs is None: - attrs = self.get_attrs() - handle = win32.STDOUT - if on_stderr: - handle = win32.STDERR - win32.SetConsoleTextAttribute(handle, attrs) - - def get_position(self, handle): - position = win32.GetConsoleScreenBufferInfo(handle).dwCursorPosition - # Because Windows coordinates are 0-based, - # and win32.SetConsoleCursorPosition expects 1-based. - position.X += 1 - position.Y += 1 - return position - - def set_cursor_position(self, position=None, on_stderr=False): - if position is None: - # I'm not currently tracking the position, so there is no default. - # position = self.get_position() - return - handle = win32.STDOUT - if on_stderr: - handle = win32.STDERR - win32.SetConsoleCursorPosition(handle, position) - - def cursor_adjust(self, x, y, on_stderr=False): - handle = win32.STDOUT - if on_stderr: - handle = win32.STDERR - position = self.get_position(handle) - adjusted_position = (position.Y + y, position.X + x) - win32.SetConsoleCursorPosition(handle, adjusted_position, adjust=False) - - def erase_screen(self, mode=0, on_stderr=False): - # 0 should clear from the cursor to the end of the screen. - # 1 should clear from the cursor to the beginning of the screen. - # 2 should clear the entire screen, and move cursor to (1,1) - handle = win32.STDOUT - if on_stderr: - handle = win32.STDERR - csbi = win32.GetConsoleScreenBufferInfo(handle) - # get the number of character cells in the current buffer - cells_in_screen = csbi.dwSize.X * csbi.dwSize.Y - # get number of character cells before current cursor position - cells_before_cursor = csbi.dwSize.X * csbi.dwCursorPosition.Y + csbi.dwCursorPosition.X - if mode == 0: - from_coord = csbi.dwCursorPosition - cells_to_erase = cells_in_screen - cells_before_cursor - elif mode == 1: - from_coord = win32.COORD(0, 0) - cells_to_erase = cells_before_cursor - elif mode == 2: - from_coord = win32.COORD(0, 0) - cells_to_erase = cells_in_screen - else: - # invalid mode - return - # fill the entire screen with blanks - win32.FillConsoleOutputCharacter(handle, ' ', cells_to_erase, from_coord) - # now set the buffer's attributes accordingly - win32.FillConsoleOutputAttribute(handle, self.get_attrs(), cells_to_erase, from_coord) - if mode == 2: - # put the cursor where needed - win32.SetConsoleCursorPosition(handle, (1, 1)) - - def erase_line(self, mode=0, on_stderr=False): - # 0 should clear from the cursor to the end of the line. - # 1 should clear from the cursor to the beginning of the line. - # 2 should clear the entire line. - handle = win32.STDOUT - if on_stderr: - handle = win32.STDERR - csbi = win32.GetConsoleScreenBufferInfo(handle) - if mode == 0: - from_coord = csbi.dwCursorPosition - cells_to_erase = csbi.dwSize.X - csbi.dwCursorPosition.X - elif mode == 1: - from_coord = win32.COORD(0, csbi.dwCursorPosition.Y) - cells_to_erase = csbi.dwCursorPosition.X - elif mode == 2: - from_coord = win32.COORD(0, csbi.dwCursorPosition.Y) - cells_to_erase = csbi.dwSize.X - else: - # invalid mode - return - # fill the entire screen with blanks - win32.FillConsoleOutputCharacter(handle, ' ', cells_to_erase, from_coord) - # now set the buffer's attributes accordingly - win32.FillConsoleOutputAttribute(handle, self.get_attrs(), cells_to_erase, from_coord) - - def set_title(self, title): - win32.SetConsoleTitle(title) - - -def enable_vt_processing(fd): - if win32.windll is None or not win32.winapi_test(): - return False - - try: - handle = get_osfhandle(fd) - mode = win32.GetConsoleMode(handle) - win32.SetConsoleMode( - handle, - mode | win32.ENABLE_VIRTUAL_TERMINAL_PROCESSING, - ) - - mode = win32.GetConsoleMode(handle) - if mode & win32.ENABLE_VIRTUAL_TERMINAL_PROCESSING: - return True - # Can get TypeError in testsuite where 'fd' is a Mock() - except (OSError, TypeError): - return False diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/INSTALLER b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/METADATA b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/METADATA deleted file mode 100644 index a845e7e..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/METADATA +++ /dev/null @@ -1,640 +0,0 @@ -Metadata-Version: 2.4 -Name: fastapi -Version: 0.128.3 -Summary: FastAPI framework, high performance, easy to learn, fast to code, ready for production -Author-Email: =?utf-8?q?Sebasti=C3=A1n_Ram=C3=ADrez?= -License-Expression: MIT -License-File: LICENSE -Classifier: Intended Audience :: Information Technology -Classifier: Intended Audience :: System Administrators -Classifier: Operating System :: OS Independent -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python -Classifier: Topic :: Internet -Classifier: Topic :: Software Development :: Libraries :: Application Frameworks -Classifier: Topic :: Software Development :: Libraries :: Python Modules -Classifier: Topic :: Software Development :: Libraries -Classifier: Topic :: Software Development -Classifier: Typing :: Typed -Classifier: Development Status :: 4 - Beta -Classifier: Environment :: Web Environment -Classifier: Framework :: AsyncIO -Classifier: Framework :: FastAPI -Classifier: Framework :: Pydantic -Classifier: Framework :: Pydantic :: 2 -Classifier: Intended Audience :: Developers -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: 3.13 -Classifier: Programming Language :: Python :: 3.14 -Classifier: Topic :: Internet :: WWW/HTTP :: HTTP Servers -Classifier: Topic :: Internet :: WWW/HTTP -Project-URL: Homepage, https://github.com/fastapi/fastapi -Project-URL: Documentation, https://fastapi.tiangolo.com/ -Project-URL: Repository, https://github.com/fastapi/fastapi -Project-URL: Issues, https://github.com/fastapi/fastapi/issues -Project-URL: Changelog, https://fastapi.tiangolo.com/release-notes/ -Requires-Python: >=3.9 -Requires-Dist: starlette<1.0.0,>=0.40.0 -Requires-Dist: pydantic>=2.7.0 -Requires-Dist: typing-extensions>=4.8.0 -Requires-Dist: typing-inspection>=0.4.2 -Requires-Dist: annotated-doc>=0.0.2 -Provides-Extra: standard -Requires-Dist: fastapi-cli[standard]>=0.0.8; extra == "standard" -Requires-Dist: httpx<1.0.0,>=0.23.0; extra == "standard" -Requires-Dist: jinja2>=3.1.5; extra == "standard" -Requires-Dist: python-multipart>=0.0.18; extra == "standard" -Requires-Dist: email-validator>=2.0.0; extra == "standard" -Requires-Dist: uvicorn[standard]>=0.12.0; extra == "standard" -Requires-Dist: pydantic-settings>=2.0.0; extra == "standard" -Requires-Dist: pydantic-extra-types>=2.0.0; extra == "standard" -Provides-Extra: standard-no-fastapi-cloud-cli -Requires-Dist: fastapi-cli[standard-no-fastapi-cloud-cli]>=0.0.8; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: httpx<1.0.0,>=0.23.0; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: jinja2>=3.1.5; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: python-multipart>=0.0.18; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: email-validator>=2.0.0; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: uvicorn[standard]>=0.12.0; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: pydantic-settings>=2.0.0; extra == "standard-no-fastapi-cloud-cli" -Requires-Dist: pydantic-extra-types>=2.0.0; extra == "standard-no-fastapi-cloud-cli" -Provides-Extra: all -Requires-Dist: fastapi-cli[standard]>=0.0.8; extra == "all" -Requires-Dist: httpx<1.0.0,>=0.23.0; extra == "all" -Requires-Dist: jinja2>=3.1.5; extra == "all" -Requires-Dist: python-multipart>=0.0.18; extra == "all" -Requires-Dist: itsdangerous>=1.1.0; extra == "all" -Requires-Dist: pyyaml>=5.3.1; extra == "all" -Requires-Dist: ujson>=5.8.0; extra == "all" -Requires-Dist: orjson>=3.9.3; extra == "all" -Requires-Dist: email-validator>=2.0.0; extra == "all" -Requires-Dist: uvicorn[standard]>=0.12.0; extra == "all" -Requires-Dist: pydantic-settings>=2.0.0; extra == "all" -Requires-Dist: pydantic-extra-types>=2.0.0; extra == "all" -Description-Content-Type: text/markdown - -

- FastAPI -

-

- FastAPI framework, high performance, easy to learn, fast to code, ready for production -

-

- - Test - - - Coverage - - - Package version - - - Supported Python versions - -

- ---- - -**Documentation**: https://fastapi.tiangolo.com - -**Source Code**: https://github.com/fastapi/fastapi - ---- - -FastAPI is a modern, fast (high-performance), web framework for building APIs with Python based on standard Python type hints. - -The key features are: - -* **Fast**: Very high performance, on par with **NodeJS** and **Go** (thanks to Starlette and Pydantic). [One of the fastest Python frameworks available](#performance). -* **Fast to code**: Increase the speed to develop features by about 200% to 300%. * -* **Fewer bugs**: Reduce about 40% of human (developer) induced errors. * -* **Intuitive**: Great editor support. Completion everywhere. Less time debugging. -* **Easy**: Designed to be easy to use and learn. Less time reading docs. -* **Short**: Minimize code duplication. Multiple features from each parameter declaration. Fewer bugs. -* **Robust**: Get production-ready code. With automatic interactive documentation. -* **Standards-based**: Based on (and fully compatible with) the open standards for APIs: OpenAPI (previously known as Swagger) and JSON Schema. - -* estimation based on tests conducted by an internal development team, building production applications. - -## Sponsors - - -### Keystone Sponsor - - - -### Gold and Silver Sponsors - - - - - - - - - - - - - - - - - - - - - - -Other sponsors - -## Opinions - -"_[...] I'm using **FastAPI** a ton these days. [...] I'm actually planning to use it for all of my team's **ML services at Microsoft**. Some of them are getting integrated into the core **Windows** product and some **Office** products._" - -
Kabir Khan - Microsoft (ref)
- ---- - -"_We adopted the **FastAPI** library to spawn a **REST** server that can be queried to obtain **predictions**. [for Ludwig]_" - -
Piero Molino, Yaroslav Dudin, and Sai Sumanth Miryala - Uber (ref)
- ---- - -"_**Netflix** is pleased to announce the open-source release of our **crisis management** orchestration framework: **Dispatch**! [built with **FastAPI**]_" - -
Kevin Glisson, Marc Vilanova, Forest Monsen - Netflix (ref)
- ---- - -"_I’m over the moon excited about **FastAPI**. It’s so fun!_" - -
Brian Okken - Python Bytes podcast host (ref)
- ---- - -"_Honestly, what you've built looks super solid and polished. In many ways, it's what I wanted **Hug** to be - it's really inspiring to see someone build that._" - -
Timothy Crosley - Hug creator (ref)
- ---- - -"_If you're looking to learn one **modern framework** for building REST APIs, check out **FastAPI** [...] It's fast, easy to use and easy to learn [...]_" - -"_We've switched over to **FastAPI** for our **APIs** [...] I think you'll like it [...]_" - -
Ines Montani - Matthew Honnibal - Explosion AI founders - spaCy creators (ref) - (ref)
- ---- - -"_If anyone is looking to build a production Python API, I would highly recommend **FastAPI**. It is **beautifully designed**, **simple to use** and **highly scalable**, it has become a **key component** in our API first development strategy and is driving many automations and services such as our Virtual TAC Engineer._" - -
Deon Pillsbury - Cisco (ref)
- ---- - -## FastAPI mini documentary - -There's a FastAPI mini documentary released at the end of 2025, you can watch it online: - -FastAPI Mini Documentary - -## **Typer**, the FastAPI of CLIs - - - -If you are building a CLI app to be used in the terminal instead of a web API, check out **Typer**. - -**Typer** is FastAPI's little sibling. And it's intended to be the **FastAPI of CLIs**. ⌨️ 🚀 - -## Requirements - -FastAPI stands on the shoulders of giants: - -* Starlette for the web parts. -* Pydantic for the data parts. - -## Installation - -Create and activate a virtual environment and then install FastAPI: - -
- -```console -$ pip install "fastapi[standard]" - ----> 100% -``` - -
- -**Note**: Make sure you put `"fastapi[standard]"` in quotes to ensure it works in all terminals. - -## Example - -### Create it - -Create a file `main.py` with: - -```Python -from fastapi import FastAPI - -app = FastAPI() - - -@app.get("/") -def read_root(): - return {"Hello": "World"} - - -@app.get("/items/{item_id}") -def read_item(item_id: int, q: str | None = None): - return {"item_id": item_id, "q": q} -``` - -
-Or use async def... - -If your code uses `async` / `await`, use `async def`: - -```Python hl_lines="7 12" -from fastapi import FastAPI - -app = FastAPI() - - -@app.get("/") -async def read_root(): - return {"Hello": "World"} - - -@app.get("/items/{item_id}") -async def read_item(item_id: int, q: str | None = None): - return {"item_id": item_id, "q": q} -``` - -**Note**: - -If you don't know, check the _"In a hurry?"_ section about `async` and `await` in the docs. - -
- -### Run it - -Run the server with: - -
- -```console -$ fastapi dev main.py - - ╭────────── FastAPI CLI - Development mode ───────────╮ - │ │ - │ Serving at: http://127.0.0.1:8000 │ - │ │ - │ API docs: http://127.0.0.1:8000/docs │ - │ │ - │ Running in development mode, for production use: │ - │ │ - │ fastapi run │ - │ │ - ╰─────────────────────────────────────────────────────╯ - -INFO: Will watch for changes in these directories: ['/home/user/code/awesomeapp'] -INFO: Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit) -INFO: Started reloader process [2248755] using WatchFiles -INFO: Started server process [2248757] -INFO: Waiting for application startup. -INFO: Application startup complete. -``` - -
- -
-About the command fastapi dev main.py... - -The command `fastapi dev` reads your `main.py` file, detects the **FastAPI** app in it, and starts a server using Uvicorn. - -By default, `fastapi dev` will start with auto-reload enabled for local development. - -You can read more about it in the FastAPI CLI docs. - -
- -### Check it - -Open your browser at http://127.0.0.1:8000/items/5?q=somequery. - -You will see the JSON response as: - -```JSON -{"item_id": 5, "q": "somequery"} -``` - -You already created an API that: - -* Receives HTTP requests in the _paths_ `/` and `/items/{item_id}`. -* Both _paths_ take `GET` operations (also known as HTTP _methods_). -* The _path_ `/items/{item_id}` has a _path parameter_ `item_id` that should be an `int`. -* The _path_ `/items/{item_id}` has an optional `str` _query parameter_ `q`. - -### Interactive API docs - -Now go to http://127.0.0.1:8000/docs. - -You will see the automatic interactive API documentation (provided by Swagger UI): - -![Swagger UI](https://fastapi.tiangolo.com/img/index/index-01-swagger-ui-simple.png) - -### Alternative API docs - -And now, go to http://127.0.0.1:8000/redoc. - -You will see the alternative automatic documentation (provided by ReDoc): - -![ReDoc](https://fastapi.tiangolo.com/img/index/index-02-redoc-simple.png) - -## Example upgrade - -Now modify the file `main.py` to receive a body from a `PUT` request. - -Declare the body using standard Python types, thanks to Pydantic. - -```Python hl_lines="2 7-10 23-25" -from fastapi import FastAPI -from pydantic import BaseModel - -app = FastAPI() - - -class Item(BaseModel): - name: str - price: float - is_offer: bool | None = None - - -@app.get("/") -def read_root(): - return {"Hello": "World"} - - -@app.get("/items/{item_id}") -def read_item(item_id: int, q: str | None = None): - return {"item_id": item_id, "q": q} - - -@app.put("/items/{item_id}") -def update_item(item_id: int, item: Item): - return {"item_name": item.name, "item_id": item_id} -``` - -The `fastapi dev` server should reload automatically. - -### Interactive API docs upgrade - -Now go to http://127.0.0.1:8000/docs. - -* The interactive API documentation will be automatically updated, including the new body: - -![Swagger UI](https://fastapi.tiangolo.com/img/index/index-03-swagger-02.png) - -* Click on the button "Try it out", it allows you to fill the parameters and directly interact with the API: - -![Swagger UI interaction](https://fastapi.tiangolo.com/img/index/index-04-swagger-03.png) - -* Then click on the "Execute" button, the user interface will communicate with your API, send the parameters, get the results and show them on the screen: - -![Swagger UI interaction](https://fastapi.tiangolo.com/img/index/index-05-swagger-04.png) - -### Alternative API docs upgrade - -And now, go to http://127.0.0.1:8000/redoc. - -* The alternative documentation will also reflect the new query parameter and body: - -![ReDoc](https://fastapi.tiangolo.com/img/index/index-06-redoc-02.png) - -### Recap - -In summary, you declare **once** the types of parameters, body, etc. as function parameters. - -You do that with standard modern Python types. - -You don't have to learn a new syntax, the methods or classes of a specific library, etc. - -Just standard **Python**. - -For example, for an `int`: - -```Python -item_id: int -``` - -or for a more complex `Item` model: - -```Python -item: Item -``` - -...and with that single declaration you get: - -* Editor support, including: - * Completion. - * Type checks. -* Validation of data: - * Automatic and clear errors when the data is invalid. - * Validation even for deeply nested JSON objects. -* Conversion of input data: coming from the network to Python data and types. Reading from: - * JSON. - * Path parameters. - * Query parameters. - * Cookies. - * Headers. - * Forms. - * Files. -* Conversion of output data: converting from Python data and types to network data (as JSON): - * Convert Python types (`str`, `int`, `float`, `bool`, `list`, etc). - * `datetime` objects. - * `UUID` objects. - * Database models. - * ...and many more. -* Automatic interactive API documentation, including 2 alternative user interfaces: - * Swagger UI. - * ReDoc. - ---- - -Coming back to the previous code example, **FastAPI** will: - -* Validate that there is an `item_id` in the path for `GET` and `PUT` requests. -* Validate that the `item_id` is of type `int` for `GET` and `PUT` requests. - * If it is not, the client will see a useful, clear error. -* Check if there is an optional query parameter named `q` (as in `http://127.0.0.1:8000/items/foo?q=somequery`) for `GET` requests. - * As the `q` parameter is declared with `= None`, it is optional. - * Without the `None` it would be required (as is the body in the case with `PUT`). -* For `PUT` requests to `/items/{item_id}`, read the body as JSON: - * Check that it has a required attribute `name` that should be a `str`. - * Check that it has a required attribute `price` that has to be a `float`. - * Check that it has an optional attribute `is_offer`, that should be a `bool`, if present. - * All this would also work for deeply nested JSON objects. -* Convert from and to JSON automatically. -* Document everything with OpenAPI, that can be used by: - * Interactive documentation systems. - * Automatic client code generation systems, for many languages. -* Provide 2 interactive documentation web interfaces directly. - ---- - -We just scratched the surface, but you already get the idea of how it all works. - -Try changing the line with: - -```Python - return {"item_name": item.name, "item_id": item_id} -``` - -...from: - -```Python - ... "item_name": item.name ... -``` - -...to: - -```Python - ... "item_price": item.price ... -``` - -...and see how your editor will auto-complete the attributes and know their types: - -![editor support](https://fastapi.tiangolo.com/img/vscode-completion.png) - -For a more complete example including more features, see the Tutorial - User Guide. - -**Spoiler alert**: the tutorial - user guide includes: - -* Declaration of **parameters** from other different places as: **headers**, **cookies**, **form fields** and **files**. -* How to set **validation constraints** as `maximum_length` or `regex`. -* A very powerful and easy to use **Dependency Injection** system. -* Security and authentication, including support for **OAuth2** with **JWT tokens** and **HTTP Basic** auth. -* More advanced (but equally easy) techniques for declaring **deeply nested JSON models** (thanks to Pydantic). -* **GraphQL** integration with Strawberry and other libraries. -* Many extra features (thanks to Starlette) as: - * **WebSockets** - * extremely easy tests based on HTTPX and `pytest` - * **CORS** - * **Cookie Sessions** - * ...and more. - -### Deploy your app (optional) - -You can optionally deploy your FastAPI app to FastAPI Cloud, go and join the waiting list if you haven't. 🚀 - -If you already have a **FastAPI Cloud** account (we invited you from the waiting list 😉), you can deploy your application with one command. - -Before deploying, make sure you are logged in: - -
- -```console -$ fastapi login - -You are logged in to FastAPI Cloud 🚀 -``` - -
- -Then deploy your app: - -
- -```console -$ fastapi deploy - -Deploying to FastAPI Cloud... - -✅ Deployment successful! - -🐔 Ready the chicken! Your app is ready at https://myapp.fastapicloud.dev -``` - -
- -That's it! Now you can access your app at that URL. ✨ - -#### About FastAPI Cloud - -**FastAPI Cloud** is built by the same author and team behind **FastAPI**. - -It streamlines the process of **building**, **deploying**, and **accessing** an API with minimal effort. - -It brings the same **developer experience** of building apps with FastAPI to **deploying** them to the cloud. 🎉 - -FastAPI Cloud is the primary sponsor and funding provider for the *FastAPI and friends* open source projects. ✨ - -#### Deploy to other cloud providers - -FastAPI is open source and based on standards. You can deploy FastAPI apps to any cloud provider you choose. - -Follow your cloud provider's guides to deploy FastAPI apps with them. 🤓 - -## Performance - -Independent TechEmpower benchmarks show **FastAPI** applications running under Uvicorn as one of the fastest Python frameworks available, only below Starlette and Uvicorn themselves (used internally by FastAPI). (*) - -To understand more about it, see the section Benchmarks. - -## Dependencies - -FastAPI depends on Pydantic and Starlette. - -### `standard` Dependencies - -When you install FastAPI with `pip install "fastapi[standard]"` it comes with the `standard` group of optional dependencies: - -Used by Pydantic: - -* email-validator - for email validation. - -Used by Starlette: - -* httpx - Required if you want to use the `TestClient`. -* jinja2 - Required if you want to use the default template configuration. -* python-multipart - Required if you want to support form "parsing", with `request.form()`. - -Used by FastAPI: - -* uvicorn - for the server that loads and serves your application. This includes `uvicorn[standard]`, which includes some dependencies (e.g. `uvloop`) needed for high performance serving. -* `fastapi-cli[standard]` - to provide the `fastapi` command. - * This includes `fastapi-cloud-cli`, which allows you to deploy your FastAPI application to FastAPI Cloud. - -### Without `standard` Dependencies - -If you don't want to include the `standard` optional dependencies, you can install with `pip install fastapi` instead of `pip install "fastapi[standard]"`. - -### Without `fastapi-cloud-cli` - -If you want to install FastAPI with the standard dependencies but without the `fastapi-cloud-cli`, you can install with `pip install "fastapi[standard-no-fastapi-cloud-cli]"`. - -### Additional Optional Dependencies - -There are some additional dependencies you might want to install. - -Additional optional Pydantic dependencies: - -* pydantic-settings - for settings management. -* pydantic-extra-types - for extra types to be used with Pydantic. - -Additional optional FastAPI dependencies: - -* orjson - Required if you want to use `ORJSONResponse`. -* ujson - Required if you want to use `UJSONResponse`. - -## License - -This project is licensed under the terms of the MIT license. diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/RECORD b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/RECORD deleted file mode 100644 index 32bb067..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/RECORD +++ /dev/null @@ -1,103 +0,0 @@ -../../Scripts/fastapi.exe,sha256=RUA9ZtIKc-W29JBtz2_FT08kkx7vWQpR6Yea0KSLVzo,108414 -fastapi-0.128.3.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -fastapi-0.128.3.dist-info/METADATA,sha256=p_BrslUzFwlbDzEKaJFboQdc1B7Mo5COxrbDkLz6PNw,30865 -fastapi-0.128.3.dist-info/RECORD,, -fastapi-0.128.3.dist-info/REQUESTED,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -fastapi-0.128.3.dist-info/WHEEL,sha256=Wb0ASbVj8JvWHpOiIpPi7ucfIgJeCi__PzivviEAQFc,90 -fastapi-0.128.3.dist-info/entry_points.txt,sha256=GCf-WbIZxyGT4MUmrPGj1cOHYZoGsNPHAvNkT6hnGeA,61 -fastapi-0.128.3.dist-info/licenses/LICENSE,sha256=Tsif_IFIW5f-xYSy1KlhAy7v_oNEU4lP2cEnSQbMdE4,1086 -fastapi/__init__.py,sha256=NjjszBOSVnhoMU7AEUt6jVbossiWFb8CoAgthArLVpo,1081 -fastapi/__main__.py,sha256=bKePXLdO4SsVSM6r9SVoLickJDcR2c0cTOxZRKq26YQ,37 -fastapi/__pycache__/__init__.cpython-312.pyc,, -fastapi/__pycache__/__main__.cpython-312.pyc,, -fastapi/__pycache__/applications.cpython-312.pyc,, -fastapi/__pycache__/background.cpython-312.pyc,, -fastapi/__pycache__/cli.cpython-312.pyc,, -fastapi/__pycache__/concurrency.cpython-312.pyc,, -fastapi/__pycache__/datastructures.cpython-312.pyc,, -fastapi/__pycache__/encoders.cpython-312.pyc,, -fastapi/__pycache__/exception_handlers.cpython-312.pyc,, -fastapi/__pycache__/exceptions.cpython-312.pyc,, -fastapi/__pycache__/logger.cpython-312.pyc,, -fastapi/__pycache__/param_functions.cpython-312.pyc,, -fastapi/__pycache__/params.cpython-312.pyc,, -fastapi/__pycache__/requests.cpython-312.pyc,, -fastapi/__pycache__/responses.cpython-312.pyc,, -fastapi/__pycache__/routing.cpython-312.pyc,, -fastapi/__pycache__/staticfiles.cpython-312.pyc,, -fastapi/__pycache__/templating.cpython-312.pyc,, -fastapi/__pycache__/testclient.cpython-312.pyc,, -fastapi/__pycache__/types.cpython-312.pyc,, -fastapi/__pycache__/utils.cpython-312.pyc,, -fastapi/__pycache__/websockets.cpython-312.pyc,, -fastapi/_compat/__init__.py,sha256=o3dg67W5LlwA52_1Y9Re_JhelcG0oj5ke_GdQHcwBnw,2226 -fastapi/_compat/__pycache__/__init__.cpython-312.pyc,, -fastapi/_compat/__pycache__/shared.cpython-312.pyc,, -fastapi/_compat/__pycache__/v2.cpython-312.pyc,, -fastapi/_compat/shared.py,sha256=-lTz2oL6M0P1ysThFfoow5v2XyEb5I9Z9GWoVBBOFIo,6863 -fastapi/_compat/v2.py,sha256=UhPSMTjFhCBFiBvzMYKd_2TTVm1pBOyo6U3ZnqKzCfg,19529 -fastapi/applications.py,sha256=FvC__4shdf03ZobKbWFnaX0yxwJEuKcFpDMTSbkS1vE,180750 -fastapi/background.py,sha256=fDNVXWBZniIQIxW3v-Sc99FT2p4RDKOOWW2fhOe4Nko,1793 -fastapi/cli.py,sha256=OYhZb0NR_deuT5ofyPF2NoNBzZDNOP8Salef2nk-HqA,418 -fastapi/concurrency.py,sha256=xHGDEOQAA6cvFEDX46oq3r2t1Zd4sVvreaRgdIE4juM,1489 -fastapi/datastructures.py,sha256=41qs2ZhTzORMGn7JSAF9qsiPY9XP4uGyGOMKhfzg4i4,5205 -fastapi/dependencies/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -fastapi/dependencies/__pycache__/__init__.cpython-312.pyc,, -fastapi/dependencies/__pycache__/models.cpython-312.pyc,, -fastapi/dependencies/__pycache__/utils.cpython-312.pyc,, -fastapi/dependencies/models.py,sha256=TjJB2l6m-vhFkau7ysLdgcymZ6SdIJmlrJjqMJs5TZc,7317 -fastapi/dependencies/utils.py,sha256=CY5mtRONzgTMR8UehcLGZ6IJ_EVVIyYUn5AepF8vr4Y,39273 -fastapi/encoders.py,sha256=ZRFeDSeTz5nu2fyULAvctVOr3DNZ-6bzOZsPPpfC1gI,11060 -fastapi/exception_handlers.py,sha256=YVcT8Zy021VYYeecgdyh5YEUjEIHKcLspbkSf4OfbJI,1275 -fastapi/exceptions.py,sha256=zk75d2EY9WloQYEs6g_fFMFEGTy7qafFRb7uh93j7Jk,7479 -fastapi/logger.py,sha256=I9NNi3ov8AcqbsbC9wl1X-hdItKgYt2XTrx1f99Zpl4,54 -fastapi/middleware/__init__.py,sha256=oQDxiFVcc1fYJUOIFvphnK7pTT5kktmfL32QXpBFvvo,58 -fastapi/middleware/__pycache__/__init__.cpython-312.pyc,, -fastapi/middleware/__pycache__/asyncexitstack.cpython-312.pyc,, -fastapi/middleware/__pycache__/cors.cpython-312.pyc,, -fastapi/middleware/__pycache__/gzip.cpython-312.pyc,, -fastapi/middleware/__pycache__/httpsredirect.cpython-312.pyc,, -fastapi/middleware/__pycache__/trustedhost.cpython-312.pyc,, -fastapi/middleware/__pycache__/wsgi.cpython-312.pyc,, -fastapi/middleware/asyncexitstack.py,sha256=RKGlQpGzg3GLosqVhrxBy_NCZ9qJS7zQeNHt5Y3x-00,637 -fastapi/middleware/cors.py,sha256=ynwjWQZoc_vbhzZ3_ZXceoaSrslHFHPdoM52rXr0WUU,79 -fastapi/middleware/gzip.py,sha256=xM5PcsH8QlAimZw4VDvcmTnqQamslThsfe3CVN2voa0,79 -fastapi/middleware/httpsredirect.py,sha256=rL8eXMnmLijwVkH7_400zHri1AekfeBd6D6qs8ix950,115 -fastapi/middleware/trustedhost.py,sha256=eE5XGRxGa7c5zPnMJDGp3BxaL25k5iVQlhnv-Pk0Pss,109 -fastapi/middleware/wsgi.py,sha256=a_FMDoeTwcdig9wdAGumIH82oDFfuj4pxtQxLbAw2Ns,107 -fastapi/openapi/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -fastapi/openapi/__pycache__/__init__.cpython-312.pyc,, -fastapi/openapi/__pycache__/constants.cpython-312.pyc,, -fastapi/openapi/__pycache__/docs.cpython-312.pyc,, -fastapi/openapi/__pycache__/models.cpython-312.pyc,, -fastapi/openapi/__pycache__/utils.cpython-312.pyc,, -fastapi/openapi/constants.py,sha256=adGzmis1L1HJRTE3kJ5fmHS_Noq6tIY6pWv_SFzoFDU,153 -fastapi/openapi/docs.py,sha256=UIoqd1JfSQ9kRV2XpJLYjr0VGohEBJHeItTH40QO9KE,12085 -fastapi/openapi/models.py,sha256=xJfPRE7DqNvtqgdouXbtMCCLBrZ-4Bd87QaA_WPUVTA,15419 -fastapi/openapi/utils.py,sha256=CRcvCOQglyu6fS3DapPvKMKQsJpxcatZxZNAMF66nYs,23907 -fastapi/param_functions.py,sha256=ivdvGFIGJ0wigG-Uc2J4Ra7rdwUXunVHtzGWAxFVEzo,70275 -fastapi/params.py,sha256=YS7z57t0N4H8Rdogx1sU6R-KZDcLn5y46SGz8z5lX-s,26982 -fastapi/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -fastapi/requests.py,sha256=zayepKFcienBllv3snmWI20Gk0oHNVLU4DDhqXBb4LU,142 -fastapi/responses.py,sha256=QNQQlwpKhQoIPZTTWkpc9d_QGeGZ_aVQPaDV3nQ8m7c,1761 -fastapi/routing.py,sha256=z6S35MQsmmmI7fpyGR5baqtu41dpDz3H0-LcQ-jsZSw,183655 -fastapi/security/__init__.py,sha256=bO8pNmxqVRXUjfl2mOKiVZLn0FpBQ61VUYVjmppnbJw,881 -fastapi/security/__pycache__/__init__.cpython-312.pyc,, -fastapi/security/__pycache__/api_key.cpython-312.pyc,, -fastapi/security/__pycache__/base.cpython-312.pyc,, -fastapi/security/__pycache__/http.cpython-312.pyc,, -fastapi/security/__pycache__/oauth2.cpython-312.pyc,, -fastapi/security/__pycache__/open_id_connect_url.cpython-312.pyc,, -fastapi/security/__pycache__/utils.cpython-312.pyc,, -fastapi/security/api_key.py,sha256=5AriUhrA_KgdtJRJ_BtCDgcTFOUlUUvDSultdIfdApc,9799 -fastapi/security/base.py,sha256=dl4pvbC-RxjfbWgPtCWd8MVU-7CB2SZ22rJDXVCXO6c,141 -fastapi/security/http.py,sha256=gckOhSa1ubLpARU819pxKiZZnmnyg_co6AwQyNE8yxw,13518 -fastapi/security/oauth2.py,sha256=lyBBk7h-TAkCVo80yQ7ZQAdPy4ySWqomGphiS43Wb1w,24287 -fastapi/security/open_id_connect_url.py,sha256=pFvSVESThhjYXSDWPlFGtm9bN62JXHzuwnVfmtyNcZE,3158 -fastapi/security/utils.py,sha256=rCERhztRWrEwECOqHBzO8zV0mPIFg6igXFzdU67L734,294 -fastapi/staticfiles.py,sha256=iirGIt3sdY2QZXd36ijs3Cj-T0FuGFda3cd90kM9Ikw,69 -fastapi/templating.py,sha256=4zsuTWgcjcEainMJFAlW6-gnslm6AgOS1SiiDWfmQxk,76 -fastapi/testclient.py,sha256=nBvaAmX66YldReJNZXPOk1sfuo2Q6hs8bOvIaCep6LQ,66 -fastapi/types.py,sha256=4J2PivseF8hQf7xupofBzBgt7ahCSR5CLNDQMCcMQSk,430 -fastapi/utils.py,sha256=Y56jz0_05qVGcbubzamY3jz3g2kg93SjTYdLbZNgR0k,5217 -fastapi/websockets.py,sha256=419uncYObEKZG0YcrXscfQQYLSWoE10jqxVMetGdR98,222 diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/REQUESTED b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/REQUESTED deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/WHEEL b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/WHEEL deleted file mode 100644 index 18c430c..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/WHEEL +++ /dev/null @@ -1,4 +0,0 @@ -Wheel-Version: 1.0 -Generator: pdm-backend (2.4.7) -Root-Is-Purelib: true -Tag: py3-none-any diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/entry_points.txt b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/entry_points.txt deleted file mode 100644 index b81849e..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/entry_points.txt +++ /dev/null @@ -1,5 +0,0 @@ -[console_scripts] -fastapi = fastapi.cli:main - -[gui_scripts] - diff --git a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/licenses/LICENSE b/.env/Lib/site-packages/fastapi-0.128.3.dist-info/licenses/LICENSE deleted file mode 100644 index 3e92463..0000000 --- a/.env/Lib/site-packages/fastapi-0.128.3.dist-info/licenses/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2018 Sebastián Ramírez - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. diff --git a/.env/Lib/site-packages/fastapi/__init__.py b/.env/Lib/site-packages/fastapi/__init__.py deleted file mode 100644 index b23e3c4..0000000 --- a/.env/Lib/site-packages/fastapi/__init__.py +++ /dev/null @@ -1,25 +0,0 @@ -"""FastAPI framework, high performance, easy to learn, fast to code, ready for production""" - -__version__ = "0.128.3" - -from starlette import status as status - -from .applications import FastAPI as FastAPI -from .background import BackgroundTasks as BackgroundTasks -from .datastructures import UploadFile as UploadFile -from .exceptions import HTTPException as HTTPException -from .exceptions import WebSocketException as WebSocketException -from .param_functions import Body as Body -from .param_functions import Cookie as Cookie -from .param_functions import Depends as Depends -from .param_functions import File as File -from .param_functions import Form as Form -from .param_functions import Header as Header -from .param_functions import Path as Path -from .param_functions import Query as Query -from .param_functions import Security as Security -from .requests import Request as Request -from .responses import Response as Response -from .routing import APIRouter as APIRouter -from .websockets import WebSocket as WebSocket -from .websockets import WebSocketDisconnect as WebSocketDisconnect diff --git a/.env/Lib/site-packages/fastapi/__main__.py b/.env/Lib/site-packages/fastapi/__main__.py deleted file mode 100644 index fc36465..0000000 --- a/.env/Lib/site-packages/fastapi/__main__.py +++ /dev/null @@ -1,3 +0,0 @@ -from fastapi.cli import main - -main() diff --git a/.env/Lib/site-packages/fastapi/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index a745afda3b8675ebdec569bd2d3afeba24f6e370..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1095 zcmYk5%T5$Q6o#vrOV52^Zd^p@jTa&eqQ)pOMga*VYLJU2l}S29cfnMqyQ{7028Wd^ zU%_YaEnJ%T1WwY%#IRsz!j8D`RB?&D_~$$SsXEo?RL}Qvxq#sL_Vp8coJQ!k3ND|a z3po7?@EN%XBiF!&JLIOkAwPvv26EGG#!LGdoY9>1vc8E;&8C<0^Ej_L=N0@SE^5vL zmvBjQ!7KX}T+v(vuHvfZ67Vn{)?5Z2!6TY0z%^XcTm`PNGq{sf-T-1M&blXy~d%e(GR;VI3dz|(lzK>e|q@uRn^L`Kgy)~!83 ze0s=5Z`SIv{jL>KvB!l^0*B68l*qmnam%Ab1hdv2>`|e^T}TQ_+`a{itx#|`b|S`u zBlB*1;ohV6{S!6o8Do%@5s6}XVw{KO#j%j7E)l15E$FRp?1AT7#FS zLS;5c)P>*7o0tl3RakO zco0x0dOc$eOxt$3W7~tGZ68n}p%Do3Fcjg@2nwZOE`t&YLytK`^(jS0S%uxJ>4>Z{ zDlUm&`~v123o3=FSJBJyNRb0WkeQB+b=QdUw?QdKgnWJE~~ zWRMf*y-QJ7VIB_C{Y%kMu?aU8v0z^`Ra7{nyHa0?6fH2sD99Id3X0CQmKQra61whe za+h1Lh|{G3>(Nfw@4RA;;E+3D4waoY4Gual*=|R&h|Y!3I@t$XeQzXW^R~?b7TI0Q?<8m;LANykw|}A~aRaLT113v5&Hw-a diff --git a/.env/Lib/site-packages/fastapi/__pycache__/__main__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/__main__.cpython-312.pyc deleted file mode 100644 index 41b898bf5789a75b5a747c8f2cd5cd47537f2d71..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 242 zcmX@j%ge<81Sbx(WvT+{#~=<2FhLog<$#Ro3@HpLj5!Rsj8Tk?3``8EjH{v2j0}}b zn#?ajJWa-1EV+r9d48Hqx46?1i%SvejtR<7 z$ye}AOwLHn%gj!VDX5I`%}g%J2g;RYCZ`t1=%wbB#rR|<#S~|jr0NzVCTAz61BIam z>&M4~?23=qE2#X%VUwGmQks)$SHuCdgAs^}d4a?SW=2NFy9_c98CY7`ir9c60HT;b AGynhq diff --git a/.env/Lib/site-packages/fastapi/__pycache__/applications.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/applications.cpython-312.pyc deleted file mode 100644 index 32cdbd7b78d2d1f1bbb722c0ba295863595b6d51..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 86867 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 aC$0AAp04Q3WF3n~a}9V~Suo#`1@jNrRHe57 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/concurrency.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/concurrency.cpython-312.pyc deleted file mode 100644 index 63bfd8e2e6b9c5bc71826f8e3d079491b3d7d41a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1717 zcmb_c-)j^{9RJSj?)|>KANrlmZBp^IbKG}6^PSJk zcV@ny@BW(2rU2W6yBF;x4d96!L@nL{t3M)`g&Nd|19eg+L}KdDx>8m`tRPm)YKT=w ztH;W*5Nl4noIq~ONz{|&Wax`Ksd~Dc4sil;rkn|J5^=ViB|w42YN-XX7QL)9wU%ZI z)89ZxEyD=YYuYH4qfxb%{YG6_BSJaX)cWo9W}H-*R1v#C_|1vf1* z?74zXiQ}ego@R2UU`npF=B}^1<0H)mJ8AM0NXZSG4b!qk^O#+?MYyU*_NHKl?Hb|? zXJ)P8c@A>B`GotLbBlCNxWaafQJee1u)KQ1VS)wJ7zwCV4`Q5&3GP~*i-)(8$x$f4 z>M!U$3Ix+2*5_*^dZZbW1fhg0&`iUa8sUvT;$FbyeTCfX%B*rx)8Mvxoz7xPMJ(cu zB_g!ZkEJ(ogfLDf!W%uN8-LT}C%S-wl5IUAU5iMd3H#yOgZPi{1fj370Nw#^&QQ@E zjsLF`i9)pFG>u+inoiTPV&vcGUpqIxGA}_UT-1ph5LB8s?n`C# zVtaXV_`}NQKI4AnGq2|L9yhHs%(cf^rO~V$w=M4B+*#XVer1rkXDi3-YQ;yz@4+${ z$GY(=qoyy+hP@Z-#+u-qxmI(q(G1>twT@mb1ilT!%LMxm(EJWB&cX8{6nD<=dsx_Y z?pT{=LhFuap;jg^e{1mlw9$OciDTS_bust%NhO3(aT5YigRLqZ}8N{{X<81fi{N~xc*Y-Zj4=m*emb>z^ z>1*llN9K2Y_r<(9aOdVkQYT3nWUkm@|PV^_J% zESDu0#84Erdz#CN4sxNe2G@1P^O7eu81!HQIevo#8~jfaxh(%spIm-^xUBpzZ?|9- zA@HqeUM3FIz1oDs4)Q)6N|E~pnzlj+X~94X_OzhXf;XSQrWWjI!KW=a@&~1at|TCq zy)<&+%X73H1KQK3n@MUpv2BhoCEji;IQUH~Ts<_WUOlp;ZGEPpV^u-kisbQK7~@|_ CHK0lW diff --git a/.env/Lib/site-packages/fastapi/__pycache__/datastructures.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/datastructures.cpython-312.pyc deleted file mode 100644 index 8fb7a3a4d4c5a6678c5ab99623dc52920b103945..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7207 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 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/encoders.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/encoders.cpython-312.pyc deleted file mode 100644 index e76a242da53d16036768296171f6509b653a9c71..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 11055 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 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/exceptions.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/exceptions.cpython-312.pyc deleted file mode 100644 index 4d1010c2e8e19fe2a0aab0e0f01fec0c4eeb3009..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10481 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 Wew|b5BB#_0%gdaqjci36KxqH~XiEzK diff --git a/.env/Lib/site-packages/fastapi/__pycache__/param_functions.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/param_functions.cpython-312.pyc deleted file mode 100644 index 9c0aff8a9de103c92dde7ab0a3f3080448e584f5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 41548 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 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/requests.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/requests.cpython-312.pyc deleted file mode 100644 index 323f7678db81af4f832704aa44ec7bdb27482f7c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 271 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 Oc3yUVC{dh6$^QfI9>h5S diff --git a/.env/Lib/site-packages/fastapi/__pycache__/staticfiles.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/staticfiles.cpython-312.pyc deleted file mode 100644 index 04bd86485d98177096e50c1342719377cbcc7225..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 243 zcmX@j%ge<81Sbx(Wx4|C#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA!& zgG&-iGLzjhb5e`_G?{LR7MCOz<)oIBr0M}#K$$du*uC&Da}c>E8+lJ!wAI1AjSt~Mn=Ya3^qkO diff --git a/.env/Lib/site-packages/fastapi/__pycache__/templating.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/templating.cpython-312.pyc deleted file mode 100644 index fa459968530d1ab579bc26f23193a30c6a45052a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 245 zcmX@j%ge<81Sbx(W%>Z=#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA#j zy)yH%5{*Jqa|?13OHzydG?{LR6qh6x<)oIBr0SJG#542KiH8JxY!wq) zoLW>I6O^Blui%@QoRONBnVlL_P#NQ!nOu|)lq<_jPA!hnOU)~b@ySezDb6fO)h$R& z&Q44R3a2Fk?I_69N0_fyQ2C3)CO1E&G$+-rhy!Q~BM=vZ7$2A!85!>}I2W-2IRN=G BMk@dS diff --git a/.env/Lib/site-packages/fastapi/__pycache__/testclient.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/testclient.cpython-312.pyc deleted file mode 100644 index 538b18084e0d6c688c3657703557f498f488218b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 240 zcmX@j%ge<81Sbx(WjX=r#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA!& zLQ;!MoO3c$^Gf_QnQn;`mn0VDq?VMV>XiT`lED&1%s{m(89sxw{0eopiU}=FEh>%) z%1_Bx@J&q4NX^U4PK_z3jPcD(F3JbWm1QQU7RTtN=9R_xWG2NFXO^Vu79=KTC#D01 w(-MJp6lCfn%-1WZ{Ka9Do1apelWJGQ0knk?h>Jmt56p~=jQ1EUi`al10E5m%SO5S3 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/types.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/types.cpython-312.pyc deleted file mode 100644 index 2d3ef54a0515c66144dc5c9c70926b2b87047ee1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 783 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 OFWL6|(E)?p{QeimWP!*4 diff --git a/.env/Lib/site-packages/fastapi/__pycache__/websockets.cpython-312.pyc b/.env/Lib/site-packages/fastapi/__pycache__/websockets.cpython-312.pyc deleted file mode 100644 index 4772f1a139071d26b8925b333da4c50af14cfdaf..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 320 zcmX@j%ge<81Sbx(W!?kQk3k$5V1hC}YXBM38B!Qh7;_kM8KW2(L2M=_+};-+Fyr- z_>%@VSMD0#{Ok$wlL$oMg`(|A&m+!(b27)A2j^v;c>p{h2bc%JgL06$04~S^^ALDQ z4lxgdhvhKy2zW$}Fpq*qf$Or)TnEp~ zdFFZWf?Qx;05@cVxdC33i_DAQCAq}B1YVZQ%*)^vxx%~xUX`oNtKc=c#=Hhzm+PMB zuWxSrh20XH-sRZQx4T~+o;=%8FSjvVxEt;rzj*fiP#ruwe15D>PLKBY;Ma5%#i>qh z94VWqPCw95YJJuF7-qd?O&F+$DT%K`?Z|zYui9~7Lgj}#N#HjA*R@fS>c}@p*g`_L z6UI7d+0dxisVI*08Sq~8?$vl=<>3>F`MzsV>)!m3G#Q%6nkdC;lJ42fY)ZGuX}@D= zewCgMyN1?$V<9;DL?>n^j#~DdMhCQ_r_=+C8b{Z?dEsA}w*JaQ#&O^FoQtU+#bZp! zIujger`rgOoJN6Z*~rl2#oNINW`IEc)Pf3*P!P8nJog zjjg!d(do4n>9$dA-GNyqlNVy?m{2~KYNNw8I=`L}&lc7^WhdMs)pBwB{;Ob@(D1W< zYLctRsMc-dW+QH`GVKl@x7$$lF7%)D3VAn&g4c%f)x~2nqdG-5AL@NfC0H>utLd zjmjhwBo&fLk|~mDk{Oa&k~xwpNsXjVGEcHV(jZwRSt3~`Ss_^^StD5|*?{~a-ay5> z?{*$HPtk~^c^n7v@&O%!$i6boPQQ6zeHX*)S>GhhEfe*c`}VAv*wlR3(f%uaj&Muk zyY1LV%I8D!NGTiHRH>~_|K0Fs`0~4<`NDmGchPSs=|TR^d!CnxtxOzc;+stD^L_j_ zm-BLO3oOF>JQJT};&CS4=lhWJ-(#(30`PSvwlnc5=RC>8Qyv0P@S{us;O=t5?mtx7 IZQ|Ac0?9lF6#xJL diff --git a/.env/Lib/site-packages/fastapi/_compat/__pycache__/shared.cpython-312.pyc b/.env/Lib/site-packages/fastapi/_compat/__pycache__/shared.cpython-312.pyc deleted file mode 100644 index 260b3edf0cb4942f4cb2f00a4096390e84409ba9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9387 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 TAFtznT4z7L&G^%;Ji`ACAkVQr diff --git a/.env/Lib/site-packages/fastapi/_compat/shared.py b/.env/Lib/site-packages/fastapi/_compat/shared.py deleted file mode 100644 index 68b9bbd..0000000 --- a/.env/Lib/site-packages/fastapi/_compat/shared.py +++ /dev/null @@ -1,206 +0,0 @@ -import sys -import types -import typing -import warnings -from collections import deque -from collections.abc import Mapping, Sequence -from dataclasses import is_dataclass -from typing import ( - Annotated, - Any, - Union, -) - -from fastapi.types import UnionType -from pydantic import BaseModel -from pydantic.version import VERSION as PYDANTIC_VERSION -from starlette.datastructures import UploadFile -from typing_extensions import get_args, get_origin - -# Copy from Pydantic: pydantic/_internal/_typing_extra.py -if sys.version_info < (3, 10): - WithArgsTypes: tuple[Any, ...] = (typing._GenericAlias, types.GenericAlias) # type: ignore[attr-defined] -else: - WithArgsTypes: tuple[Any, ...] = ( - typing._GenericAlias, # type: ignore[attr-defined] - types.GenericAlias, - types.UnionType, - ) # pyright: ignore[reportAttributeAccessIssue] - -PYDANTIC_VERSION_MINOR_TUPLE = tuple(int(x) for x in PYDANTIC_VERSION.split(".")[:2]) -PYDANTIC_V2 = PYDANTIC_VERSION_MINOR_TUPLE[0] == 2 - - -sequence_annotation_to_type = { - Sequence: list, - list: list, - tuple: tuple, - set: set, - frozenset: frozenset, - deque: deque, -} - -sequence_types = tuple(sequence_annotation_to_type.keys()) - -Url: type[Any] - - -# Copy of Pydantic: pydantic/_internal/_utils.py -def lenient_issubclass( - cls: Any, class_or_tuple: Union[type[Any], tuple[type[Any], ...], None] -) -> bool: - try: - return isinstance(cls, type) and issubclass(cls, class_or_tuple) # type: ignore[arg-type] - except TypeError: # pragma: no cover - if isinstance(cls, WithArgsTypes): - return False - raise # pragma: no cover - - -def _annotation_is_sequence(annotation: Union[type[Any], None]) -> bool: - if lenient_issubclass(annotation, (str, bytes)): - return False - return lenient_issubclass(annotation, sequence_types) - - -def field_annotation_is_sequence(annotation: Union[type[Any], None]) -> bool: - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - for arg in get_args(annotation): - if field_annotation_is_sequence(arg): - return True - return False - return _annotation_is_sequence(annotation) or _annotation_is_sequence( - get_origin(annotation) - ) - - -def value_is_sequence(value: Any) -> bool: - return isinstance(value, sequence_types) and not isinstance(value, (str, bytes)) - - -def _annotation_is_complex(annotation: Union[type[Any], None]) -> bool: - return ( - lenient_issubclass(annotation, (BaseModel, Mapping, UploadFile)) - or _annotation_is_sequence(annotation) - or is_dataclass(annotation) - ) - - -def field_annotation_is_complex(annotation: Union[type[Any], None]) -> bool: - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - return any(field_annotation_is_complex(arg) for arg in get_args(annotation)) - - if origin is Annotated: - return field_annotation_is_complex(get_args(annotation)[0]) - - return ( - _annotation_is_complex(annotation) - or _annotation_is_complex(origin) - or hasattr(origin, "__pydantic_core_schema__") - or hasattr(origin, "__get_pydantic_core_schema__") - ) - - -def field_annotation_is_scalar(annotation: Any) -> bool: - # handle Ellipsis here to make tuple[int, ...] work nicely - return annotation is Ellipsis or not field_annotation_is_complex(annotation) - - -def field_annotation_is_scalar_sequence(annotation: Union[type[Any], None]) -> bool: - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - at_least_one_scalar_sequence = False - for arg in get_args(annotation): - if field_annotation_is_scalar_sequence(arg): - at_least_one_scalar_sequence = True - continue - elif not field_annotation_is_scalar(arg): - return False - return at_least_one_scalar_sequence - return field_annotation_is_sequence(annotation) and all( - field_annotation_is_scalar(sub_annotation) - for sub_annotation in get_args(annotation) - ) - - -def is_bytes_or_nonable_bytes_annotation(annotation: Any) -> bool: - if lenient_issubclass(annotation, bytes): - return True - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - for arg in get_args(annotation): - if lenient_issubclass(arg, bytes): - return True - return False - - -def is_uploadfile_or_nonable_uploadfile_annotation(annotation: Any) -> bool: - if lenient_issubclass(annotation, UploadFile): - return True - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - for arg in get_args(annotation): - if lenient_issubclass(arg, UploadFile): - return True - return False - - -def is_bytes_sequence_annotation(annotation: Any) -> bool: - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - at_least_one = False - for arg in get_args(annotation): - if is_bytes_sequence_annotation(arg): - at_least_one = True - continue - return at_least_one - return field_annotation_is_sequence(annotation) and all( - is_bytes_or_nonable_bytes_annotation(sub_annotation) - for sub_annotation in get_args(annotation) - ) - - -def is_uploadfile_sequence_annotation(annotation: Any) -> bool: - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - at_least_one = False - for arg in get_args(annotation): - if is_uploadfile_sequence_annotation(arg): - at_least_one = True - continue - return at_least_one - return field_annotation_is_sequence(annotation) and all( - is_uploadfile_or_nonable_uploadfile_annotation(sub_annotation) - for sub_annotation in get_args(annotation) - ) - - -def is_pydantic_v1_model_instance(obj: Any) -> bool: - with warnings.catch_warnings(): - warnings.simplefilter("ignore", UserWarning) - from pydantic import v1 - return isinstance(obj, v1.BaseModel) - - -def is_pydantic_v1_model_class(cls: Any) -> bool: - with warnings.catch_warnings(): - warnings.simplefilter("ignore", UserWarning) - from pydantic import v1 - return lenient_issubclass(cls, v1.BaseModel) - - -def annotation_is_pydantic_v1(annotation: Any) -> bool: - if is_pydantic_v1_model_class(annotation): - return True - origin = get_origin(annotation) - if origin is Union or origin is UnionType: - for arg in get_args(annotation): - if is_pydantic_v1_model_class(arg): - return True - if field_annotation_is_sequence(annotation): - for sub_annotation in get_args(annotation): - if annotation_is_pydantic_v1(sub_annotation): - return True - return False diff --git a/.env/Lib/site-packages/fastapi/_compat/v2.py b/.env/Lib/site-packages/fastapi/_compat/v2.py deleted file mode 100644 index dae78a3..0000000 --- a/.env/Lib/site-packages/fastapi/_compat/v2.py +++ /dev/null @@ -1,568 +0,0 @@ -import re -import warnings -from collections.abc import Sequence -from copy import copy, deepcopy -from dataclasses import dataclass, is_dataclass -from enum import Enum -from functools import lru_cache -from typing import ( - Annotated, - Any, - Union, - cast, -) - -from fastapi._compat import shared -from fastapi.openapi.constants import REF_TEMPLATE -from fastapi.types import IncEx, ModelNameMap, UnionType -from pydantic import BaseModel, ConfigDict, Field, TypeAdapter, create_model -from pydantic import PydanticSchemaGenerationError as PydanticSchemaGenerationError -from pydantic import PydanticUndefinedAnnotation as PydanticUndefinedAnnotation -from pydantic import ValidationError as ValidationError -from pydantic._internal._schema_generation_shared import ( # type: ignore[attr-defined] - GetJsonSchemaHandler as GetJsonSchemaHandler, -) -from pydantic._internal._typing_extra import eval_type_lenient -from pydantic._internal._utils import lenient_issubclass as lenient_issubclass -from pydantic.fields import FieldInfo as FieldInfo -from pydantic.json_schema import GenerateJsonSchema as GenerateJsonSchema -from pydantic.json_schema import JsonSchemaValue as JsonSchemaValue -from pydantic_core import CoreSchema as CoreSchema -from pydantic_core import PydanticUndefined, PydanticUndefinedType -from pydantic_core import Url as Url -from typing_extensions import Literal, get_args, get_origin - -try: - from pydantic_core.core_schema import ( - with_info_plain_validator_function as with_info_plain_validator_function, - ) -except ImportError: # pragma: no cover - from pydantic_core.core_schema import ( - general_plain_validator_function as with_info_plain_validator_function, # noqa: F401 - ) - -RequiredParam = PydanticUndefined -Undefined = PydanticUndefined -UndefinedType = PydanticUndefinedType -evaluate_forwardref = eval_type_lenient -Validator = Any - -# TODO: remove when dropping support for Pydantic < v2.12.3 -_Attrs = { - "default": ..., - "default_factory": None, - "alias": None, - "alias_priority": None, - "validation_alias": None, - "serialization_alias": None, - "title": None, - "field_title_generator": None, - "description": None, - "examples": None, - "exclude": None, - "exclude_if": None, - "discriminator": None, - "deprecated": None, - "json_schema_extra": None, - "frozen": None, - "validate_default": None, - "repr": True, - "init": None, - "init_var": None, - "kw_only": None, -} - - -# TODO: remove when dropping support for Pydantic < v2.12.3 -def asdict(field_info: FieldInfo) -> dict[str, Any]: - attributes = {} - for attr in _Attrs: - value = getattr(field_info, attr, Undefined) - if value is not Undefined: - attributes[attr] = value - return { - "annotation": field_info.annotation, - "metadata": field_info.metadata, - "attributes": attributes, - } - - -class BaseConfig: - pass - - -class ErrorWrapper(Exception): - pass - - -@dataclass -class ModelField: - field_info: FieldInfo - name: str - mode: Literal["validation", "serialization"] = "validation" - config: Union[ConfigDict, None] = None - - @property - def alias(self) -> str: - a = self.field_info.alias - return a if a is not None else self.name - - @property - def validation_alias(self) -> Union[str, None]: - va = self.field_info.validation_alias - if isinstance(va, str) and va: - return va - return None - - @property - def serialization_alias(self) -> Union[str, None]: - sa = self.field_info.serialization_alias - return sa or None - - @property - def required(self) -> bool: - return self.field_info.is_required() - - @property - def default(self) -> Any: - return self.get_default() - - @property - def type_(self) -> Any: - return self.field_info.annotation - - def __post_init__(self) -> None: - with warnings.catch_warnings(): - # Pydantic >= 2.12.0 warns about field specific metadata that is unused - # (e.g. `TypeAdapter(Annotated[int, Field(alias='b')])`). In some cases, we - # end up building the type adapter from a model field annotation so we - # need to ignore the warning: - if shared.PYDANTIC_VERSION_MINOR_TUPLE >= (2, 12): - from pydantic.warnings import UnsupportedFieldAttributeWarning - - warnings.simplefilter( - "ignore", category=UnsupportedFieldAttributeWarning - ) - # TODO: remove after dropping support for Python 3.8 and - # setting the min Pydantic to v2.12.3 that adds asdict() - field_dict = asdict(self.field_info) - annotated_args = ( - field_dict["annotation"], - *field_dict["metadata"], - # this FieldInfo needs to be created again so that it doesn't include - # the old field info metadata and only the rest of the attributes - Field(**field_dict["attributes"]), - ) - self._type_adapter: TypeAdapter[Any] = TypeAdapter( - Annotated[annotated_args], - config=self.config, - ) - - def get_default(self) -> Any: - if self.field_info.is_required(): - return Undefined - return self.field_info.get_default(call_default_factory=True) - - def validate( - self, - value: Any, - values: dict[str, Any] = {}, # noqa: B006 - *, - loc: tuple[Union[int, str], ...] = (), - ) -> tuple[Any, Union[list[dict[str, Any]], None]]: - try: - return ( - self._type_adapter.validate_python(value, from_attributes=True), - None, - ) - except ValidationError as exc: - return None, _regenerate_error_with_loc( - errors=exc.errors(include_url=False), loc_prefix=loc - ) - - def serialize( - self, - value: Any, - *, - mode: Literal["json", "python"] = "json", - include: Union[IncEx, None] = None, - exclude: Union[IncEx, None] = None, - by_alias: bool = True, - exclude_unset: bool = False, - exclude_defaults: bool = False, - exclude_none: bool = False, - ) -> Any: - # What calls this code passes a value that already called - # self._type_adapter.validate_python(value) - return self._type_adapter.dump_python( - value, - mode=mode, - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_defaults=exclude_defaults, - exclude_none=exclude_none, - ) - - def __hash__(self) -> int: - # Each ModelField is unique for our purposes, to allow making a dict from - # ModelField to its JSON Schema. - return id(self) - - -def _has_computed_fields(field: ModelField) -> bool: - computed_fields = field._type_adapter.core_schema.get("schema", {}).get( - "computed_fields", [] - ) - return len(computed_fields) > 0 - - -def get_schema_from_model_field( - *, - field: ModelField, - model_name_map: ModelNameMap, - field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], JsonSchemaValue - ], - separate_input_output_schemas: bool = True, -) -> dict[str, Any]: - override_mode: Union[Literal["validation"], None] = ( - None - if (separate_input_output_schemas or _has_computed_fields(field)) - else "validation" - ) - field_alias = ( - (field.validation_alias or field.alias) - if field.mode == "validation" - else (field.serialization_alias or field.alias) - ) - - # This expects that GenerateJsonSchema was already used to generate the definitions - json_schema = field_mapping[(field, override_mode or field.mode)] - if "$ref" not in json_schema: - # TODO remove when deprecating Pydantic v1 - # Ref: https://github.com/pydantic/pydantic/blob/d61792cc42c80b13b23e3ffa74bc37ec7c77f7d1/pydantic/schema.py#L207 - json_schema["title"] = field.field_info.title or field_alias.title().replace( - "_", " " - ) - return json_schema - - -def get_definitions( - *, - fields: Sequence[ModelField], - model_name_map: ModelNameMap, - separate_input_output_schemas: bool = True, -) -> tuple[ - dict[tuple[ModelField, Literal["validation", "serialization"]], JsonSchemaValue], - dict[str, dict[str, Any]], -]: - schema_generator = GenerateJsonSchema(ref_template=REF_TEMPLATE) - validation_fields = [field for field in fields if field.mode == "validation"] - serialization_fields = [field for field in fields if field.mode == "serialization"] - flat_validation_models = get_flat_models_from_fields( - validation_fields, known_models=set() - ) - flat_serialization_models = get_flat_models_from_fields( - serialization_fields, known_models=set() - ) - flat_validation_model_fields = [ - ModelField( - field_info=FieldInfo(annotation=model), - name=model.__name__, - mode="validation", - ) - for model in flat_validation_models - ] - flat_serialization_model_fields = [ - ModelField( - field_info=FieldInfo(annotation=model), - name=model.__name__, - mode="serialization", - ) - for model in flat_serialization_models - ] - flat_model_fields = flat_validation_model_fields + flat_serialization_model_fields - input_types = {f.type_ for f in fields} - unique_flat_model_fields = { - f for f in flat_model_fields if f.type_ not in input_types - } - inputs = [ - ( - field, - ( - field.mode - if (separate_input_output_schemas or _has_computed_fields(field)) - else "validation" - ), - field._type_adapter.core_schema, - ) - for field in list(fields) + list(unique_flat_model_fields) - ] - field_mapping, definitions = schema_generator.generate_definitions(inputs=inputs) - for item_def in cast(dict[str, dict[str, Any]], definitions).values(): - if "description" in item_def: - item_description = cast(str, item_def["description"]).split("\f")[0] - item_def["description"] = item_description - new_mapping, new_definitions = _remap_definitions_and_field_mappings( - model_name_map=model_name_map, - definitions=definitions, # type: ignore[arg-type] - field_mapping=field_mapping, - ) - return new_mapping, new_definitions - - -def _replace_refs( - *, - schema: dict[str, Any], - old_name_to_new_name_map: dict[str, str], -) -> dict[str, Any]: - new_schema = deepcopy(schema) - for key, value in new_schema.items(): - if key == "$ref": - value = schema["$ref"] - if isinstance(value, str): - ref_name = schema["$ref"].split("/")[-1] - if ref_name in old_name_to_new_name_map: - new_name = old_name_to_new_name_map[ref_name] - new_schema["$ref"] = REF_TEMPLATE.format(model=new_name) - continue - if isinstance(value, dict): - new_schema[key] = _replace_refs( - schema=value, - old_name_to_new_name_map=old_name_to_new_name_map, - ) - elif isinstance(value, list): - new_value = [] - for item in value: - if isinstance(item, dict): - new_item = _replace_refs( - schema=item, - old_name_to_new_name_map=old_name_to_new_name_map, - ) - new_value.append(new_item) - - else: - new_value.append(item) - new_schema[key] = new_value - return new_schema - - -def _remap_definitions_and_field_mappings( - *, - model_name_map: ModelNameMap, - definitions: dict[str, Any], - field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], JsonSchemaValue - ], -) -> tuple[ - dict[tuple[ModelField, Literal["validation", "serialization"]], JsonSchemaValue], - dict[str, Any], -]: - old_name_to_new_name_map = {} - for field_key, schema in field_mapping.items(): - model = field_key[0].type_ - if model not in model_name_map or "$ref" not in schema: - continue - new_name = model_name_map[model] - old_name = schema["$ref"].split("/")[-1] - if old_name in {f"{new_name}-Input", f"{new_name}-Output"}: - continue - old_name_to_new_name_map[old_name] = new_name - - new_field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], JsonSchemaValue - ] = {} - for field_key, schema in field_mapping.items(): - new_schema = _replace_refs( - schema=schema, - old_name_to_new_name_map=old_name_to_new_name_map, - ) - new_field_mapping[field_key] = new_schema - - new_definitions = {} - for key, value in definitions.items(): - if key in old_name_to_new_name_map: - new_key = old_name_to_new_name_map[key] - else: - new_key = key - new_value = _replace_refs( - schema=value, - old_name_to_new_name_map=old_name_to_new_name_map, - ) - new_definitions[new_key] = new_value - return new_field_mapping, new_definitions - - -def is_scalar_field(field: ModelField) -> bool: - from fastapi import params - - return shared.field_annotation_is_scalar( - field.field_info.annotation - ) and not isinstance(field.field_info, params.Body) - - -def is_sequence_field(field: ModelField) -> bool: - return shared.field_annotation_is_sequence(field.field_info.annotation) - - -def is_scalar_sequence_field(field: ModelField) -> bool: - return shared.field_annotation_is_scalar_sequence(field.field_info.annotation) - - -def is_bytes_field(field: ModelField) -> bool: - return shared.is_bytes_or_nonable_bytes_annotation(field.type_) - - -def is_bytes_sequence_field(field: ModelField) -> bool: - return shared.is_bytes_sequence_annotation(field.type_) - - -def copy_field_info(*, field_info: FieldInfo, annotation: Any) -> FieldInfo: - cls = type(field_info) - merged_field_info = cls.from_annotation(annotation) - new_field_info = copy(field_info) - new_field_info.metadata = merged_field_info.metadata - new_field_info.annotation = merged_field_info.annotation - return new_field_info - - -def serialize_sequence_value(*, field: ModelField, value: Any) -> Sequence[Any]: - origin_type = get_origin(field.field_info.annotation) or field.field_info.annotation - if origin_type is Union or origin_type is UnionType: # Handle optional sequences - union_args = get_args(field.field_info.annotation) - for union_arg in union_args: - if union_arg is type(None): - continue - origin_type = get_origin(union_arg) or union_arg - break - assert issubclass(origin_type, shared.sequence_types) # type: ignore[arg-type] - return shared.sequence_annotation_to_type[origin_type](value) # type: ignore[no-any-return,index] - - -def get_missing_field_error(loc: tuple[str, ...]) -> dict[str, Any]: - error = ValidationError.from_exception_data( - "Field required", [{"type": "missing", "loc": loc, "input": {}}] - ).errors(include_url=False)[0] - error["input"] = None - return error # type: ignore[return-value] - - -def create_body_model( - *, fields: Sequence[ModelField], model_name: str -) -> type[BaseModel]: - field_params = {f.name: (f.field_info.annotation, f.field_info) for f in fields} - BodyModel: type[BaseModel] = create_model(model_name, **field_params) # type: ignore[call-overload] - return BodyModel - - -def get_model_fields(model: type[BaseModel]) -> list[ModelField]: - model_fields: list[ModelField] = [] - for name, field_info in model.model_fields.items(): - type_ = field_info.annotation - if lenient_issubclass(type_, (BaseModel, dict)) or is_dataclass(type_): - model_config = None - else: - model_config = model.model_config - model_fields.append( - ModelField( - field_info=field_info, - name=name, - config=model_config, - ) - ) - return model_fields - - -@lru_cache -def get_cached_model_fields(model: type[BaseModel]) -> list[ModelField]: - return get_model_fields(model) - - -# Duplicate of several schema functions from Pydantic v1 to make them compatible with -# Pydantic v2 and allow mixing the models - -TypeModelOrEnum = Union[type["BaseModel"], type[Enum]] -TypeModelSet = set[TypeModelOrEnum] - - -def normalize_name(name: str) -> str: - return re.sub(r"[^a-zA-Z0-9.\-_]", "_", name) - - -def get_model_name_map(unique_models: TypeModelSet) -> dict[TypeModelOrEnum, str]: - name_model_map = {} - for model in unique_models: - model_name = normalize_name(model.__name__) - name_model_map[model_name] = model - return {v: k for k, v in name_model_map.items()} - - -def get_compat_model_name_map(fields: list[ModelField]) -> ModelNameMap: - all_flat_models: TypeModelSet = set() - - v2_model_fields = [field for field in fields if isinstance(field, ModelField)] - v2_flat_models = get_flat_models_from_fields(v2_model_fields, known_models=set()) - all_flat_models = all_flat_models.union(v2_flat_models) - - model_name_map = get_model_name_map(all_flat_models) - return model_name_map - - -def get_flat_models_from_model( - model: type["BaseModel"], known_models: Union[TypeModelSet, None] = None -) -> TypeModelSet: - known_models = known_models or set() - fields = get_model_fields(model) - get_flat_models_from_fields(fields, known_models=known_models) - return known_models - - -def get_flat_models_from_annotation( - annotation: Any, known_models: TypeModelSet -) -> TypeModelSet: - origin = get_origin(annotation) - if origin is not None: - for arg in get_args(annotation): - if lenient_issubclass(arg, (BaseModel, Enum)) and arg not in known_models: - known_models.add(arg) - if lenient_issubclass(arg, BaseModel): - get_flat_models_from_model(arg, known_models=known_models) - else: - get_flat_models_from_annotation(arg, known_models=known_models) - return known_models - - -def get_flat_models_from_field( - field: ModelField, known_models: TypeModelSet -) -> TypeModelSet: - field_type = field.type_ - if lenient_issubclass(field_type, BaseModel): - if field_type in known_models: - return known_models - known_models.add(field_type) - get_flat_models_from_model(field_type, known_models=known_models) - elif lenient_issubclass(field_type, Enum): - known_models.add(field_type) - else: - get_flat_models_from_annotation(field_type, known_models=known_models) - return known_models - - -def get_flat_models_from_fields( - fields: Sequence[ModelField], known_models: TypeModelSet -) -> TypeModelSet: - for field in fields: - get_flat_models_from_field(field, known_models=known_models) - return known_models - - -def _regenerate_error_with_loc( - *, errors: Sequence[Any], loc_prefix: tuple[Union[str, int], ...] -) -> list[dict[str, Any]]: - updated_loc_errors: list[Any] = [ - {**err, "loc": loc_prefix + err.get("loc", ())} for err in errors - ] - - return updated_loc_errors diff --git a/.env/Lib/site-packages/fastapi/applications.py b/.env/Lib/site-packages/fastapi/applications.py deleted file mode 100644 index 340cabf..0000000 --- a/.env/Lib/site-packages/fastapi/applications.py +++ /dev/null @@ -1,4672 +0,0 @@ -from collections.abc import Awaitable, Coroutine, Sequence -from enum import Enum -from typing import ( - Annotated, - Any, - Callable, - Optional, - TypeVar, - Union, -) - -from annotated_doc import Doc -from fastapi import routing -from fastapi.datastructures import Default, DefaultPlaceholder -from fastapi.exception_handlers import ( - http_exception_handler, - request_validation_exception_handler, - websocket_request_validation_exception_handler, -) -from fastapi.exceptions import RequestValidationError, WebSocketRequestValidationError -from fastapi.logger import logger -from fastapi.middleware.asyncexitstack import AsyncExitStackMiddleware -from fastapi.openapi.docs import ( - get_redoc_html, - get_swagger_ui_html, - get_swagger_ui_oauth2_redirect_html, -) -from fastapi.openapi.utils import get_openapi -from fastapi.params import Depends -from fastapi.types import DecoratedCallable, IncEx -from fastapi.utils import generate_unique_id -from starlette.applications import Starlette -from starlette.datastructures import State -from starlette.exceptions import HTTPException -from starlette.middleware import Middleware -from starlette.middleware.base import BaseHTTPMiddleware -from starlette.middleware.errors import ServerErrorMiddleware -from starlette.middleware.exceptions import ExceptionMiddleware -from starlette.requests import Request -from starlette.responses import HTMLResponse, JSONResponse, Response -from starlette.routing import BaseRoute -from starlette.types import ASGIApp, ExceptionHandler, Lifespan, Receive, Scope, Send -from typing_extensions import deprecated - -AppType = TypeVar("AppType", bound="FastAPI") - - -class FastAPI(Starlette): - """ - `FastAPI` app class, the main entrypoint to use FastAPI. - - Read more in the - [FastAPI docs for First Steps](https://fastapi.tiangolo.com/tutorial/first-steps/). - - ## Example - - ```python - from fastapi import FastAPI - - app = FastAPI() - ``` - """ - - def __init__( - self: AppType, - *, - debug: Annotated[ - bool, - Doc( - """ - Boolean indicating if debug tracebacks should be returned on server - errors. - - Read more in the - [Starlette docs for Applications](https://www.starlette.dev/applications/#instantiating-the-application). - """ - ), - ] = False, - routes: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - **Note**: you probably shouldn't use this parameter, it is inherited - from Starlette and supported for compatibility. - - --- - - A list of routes to serve incoming HTTP and WebSocket requests. - """ - ), - deprecated( - """ - You normally wouldn't use this parameter with FastAPI, it is inherited - from Starlette and supported for compatibility. - - In FastAPI, you normally would use the *path operation methods*, - like `app.get()`, `app.post()`, etc. - """ - ), - ] = None, - title: Annotated[ - str, - Doc( - """ - The title of the API. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(title="ChimichangApp") - ``` - """ - ), - ] = "FastAPI", - summary: Annotated[ - Optional[str], - Doc( - """ - A short summary of the API. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(summary="Deadpond's favorite app. Nuff said.") - ``` - """ - ), - ] = None, - description: Annotated[ - str, - Doc( - ''' - A description of the API. Supports Markdown (using - [CommonMark syntax](https://commonmark.org/)). - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI( - description=""" - ChimichangApp API helps you do awesome stuff. 🚀 - - ## Items - - You can **read items**. - - ## Users - - You will be able to: - - * **Create users** (_not implemented_). - * **Read users** (_not implemented_). - - """ - ) - ``` - ''' - ), - ] = "", - version: Annotated[ - str, - Doc( - """ - The version of the API. - - **Note** This is the version of your application, not the version of - the OpenAPI specification nor the version of FastAPI being used. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(version="0.0.1") - ``` - """ - ), - ] = "0.1.0", - openapi_url: Annotated[ - Optional[str], - Doc( - """ - The URL where the OpenAPI schema will be served from. - - If you set it to `None`, no OpenAPI schema will be served publicly, and - the default automatic endpoints `/docs` and `/redoc` will also be - disabled. - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#openapi-url). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(openapi_url="/api/v1/openapi.json") - ``` - """ - ), - ] = "/openapi.json", - openapi_tags: Annotated[ - Optional[list[dict[str, Any]]], - Doc( - """ - A list of tags used by OpenAPI, these are the same `tags` you can set - in the *path operations*, like: - - * `@app.get("/users/", tags=["users"])` - * `@app.get("/items/", tags=["items"])` - - The order of the tags can be used to specify the order shown in - tools like Swagger UI, used in the automatic path `/docs`. - - It's not required to specify all the tags used. - - The tags that are not declared MAY be organized randomly or based - on the tools' logic. Each tag name in the list MUST be unique. - - The value of each item is a `dict` containing: - - * `name`: The name of the tag. - * `description`: A short description of the tag. - [CommonMark syntax](https://commonmark.org/) MAY be used for rich - text representation. - * `externalDocs`: Additional external documentation for this tag. If - provided, it would contain a `dict` with: - * `description`: A short description of the target documentation. - [CommonMark syntax](https://commonmark.org/) MAY be used for - rich text representation. - * `url`: The URL for the target documentation. Value MUST be in - the form of a URL. - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-tags). - - **Example** - - ```python - from fastapi import FastAPI - - tags_metadata = [ - { - "name": "users", - "description": "Operations with users. The **login** logic is also here.", - }, - { - "name": "items", - "description": "Manage items. So _fancy_ they have their own docs.", - "externalDocs": { - "description": "Items external docs", - "url": "https://fastapi.tiangolo.com/", - }, - }, - ] - - app = FastAPI(openapi_tags=tags_metadata) - ``` - """ - ), - ] = None, - servers: Annotated[ - Optional[list[dict[str, Union[str, Any]]]], - Doc( - """ - A `list` of `dict`s with connectivity information to a target server. - - You would use it, for example, if your application is served from - different domains and you want to use the same Swagger UI in the - browser to interact with each of them (instead of having multiple - browser tabs open). Or if you want to leave fixed the possible URLs. - - If the servers `list` is not provided, or is an empty `list`, the - `servers` property in the generated OpenAPI will be: - - * a `dict` with a `url` value of the application's mounting point - (`root_path`) if it's different from `/`. - * otherwise, the `servers` property will be omitted from the OpenAPI - schema. - - Each item in the `list` is a `dict` containing: - - * `url`: A URL to the target host. This URL supports Server Variables - and MAY be relative, to indicate that the host location is relative - to the location where the OpenAPI document is being served. Variable - substitutions will be made when a variable is named in `{`brackets`}`. - * `description`: An optional string describing the host designated by - the URL. [CommonMark syntax](https://commonmark.org/) MAY be used for - rich text representation. - * `variables`: A `dict` between a variable name and its value. The value - is used for substitution in the server's URL template. - - Read more in the - [FastAPI docs for Behind a Proxy](https://fastapi.tiangolo.com/advanced/behind-a-proxy/#additional-servers). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI( - servers=[ - {"url": "https://stag.example.com", "description": "Staging environment"}, - {"url": "https://prod.example.com", "description": "Production environment"}, - ] - ) - ``` - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of global dependencies, they will be applied to each - *path operation*, including in sub-routers. - - Read more about it in the - [FastAPI docs for Global Dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/global-dependencies/). - - **Example** - - ```python - from fastapi import Depends, FastAPI - - from .dependencies import func_dep_1, func_dep_2 - - app = FastAPI(dependencies=[Depends(func_dep_1), Depends(func_dep_2)]) - ``` - """ - ), - ] = None, - default_response_class: Annotated[ - type[Response], - Doc( - """ - The default response class to be used. - - Read more in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#default-response-class). - - **Example** - - ```python - from fastapi import FastAPI - from fastapi.responses import ORJSONResponse - - app = FastAPI(default_response_class=ORJSONResponse) - ``` - """ - ), - ] = Default(JSONResponse), - redirect_slashes: Annotated[ - bool, - Doc( - """ - Whether to detect and redirect slashes in URLs when the client doesn't - use the same format. - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(redirect_slashes=True) # the default - - @app.get("/items/") - async def read_items(): - return [{"item_id": "Foo"}] - ``` - - With this app, if a client goes to `/items` (without a trailing slash), - they will be automatically redirected with an HTTP status code of 307 - to `/items/`. - """ - ), - ] = True, - docs_url: Annotated[ - Optional[str], - Doc( - """ - The path to the automatic interactive API documentation. - It is handled in the browser by Swagger UI. - - The default URL is `/docs`. You can disable it by setting it to `None`. - - If `openapi_url` is set to `None`, this will be automatically disabled. - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#docs-urls). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(docs_url="/documentation", redoc_url=None) - ``` - """ - ), - ] = "/docs", - redoc_url: Annotated[ - Optional[str], - Doc( - """ - The path to the alternative automatic interactive API documentation - provided by ReDoc. - - The default URL is `/redoc`. You can disable it by setting it to `None`. - - If `openapi_url` is set to `None`, this will be automatically disabled. - - Read more in the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#docs-urls). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(docs_url="/documentation", redoc_url="redocumentation") - ``` - """ - ), - ] = "/redoc", - swagger_ui_oauth2_redirect_url: Annotated[ - Optional[str], - Doc( - """ - The OAuth2 redirect endpoint for the Swagger UI. - - By default it is `/docs/oauth2-redirect`. - - This is only used if you use OAuth2 (with the "Authorize" button) - with Swagger UI. - """ - ), - ] = "/docs/oauth2-redirect", - swagger_ui_init_oauth: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - OAuth2 configuration for the Swagger UI, by default shown at `/docs`. - - Read more about the available configuration options in the - [Swagger UI docs](https://swagger.io/docs/open-source-tools/swagger-ui/usage/oauth2/). - """ - ), - ] = None, - middleware: Annotated[ - Optional[Sequence[Middleware]], - Doc( - """ - List of middleware to be added when creating the application. - - In FastAPI you would normally do this with `app.add_middleware()` - instead. - - Read more in the - [FastAPI docs for Middleware](https://fastapi.tiangolo.com/tutorial/middleware/). - """ - ), - ] = None, - exception_handlers: Annotated[ - Optional[ - dict[ - Union[int, type[Exception]], - Callable[[Request, Any], Coroutine[Any, Any, Response]], - ] - ], - Doc( - """ - A dictionary with handlers for exceptions. - - In FastAPI, you would normally use the decorator - `@app.exception_handler()`. - - Read more in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/). - """ - ), - ] = None, - on_startup: Annotated[ - Optional[Sequence[Callable[[], Any]]], - Doc( - """ - A list of startup event handler functions. - - You should instead use the `lifespan` handlers. - - Read more in the [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - on_shutdown: Annotated[ - Optional[Sequence[Callable[[], Any]]], - Doc( - """ - A list of shutdown event handler functions. - - You should instead use the `lifespan` handlers. - - Read more in the - [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - lifespan: Annotated[ - Optional[Lifespan[AppType]], - Doc( - """ - A `Lifespan` context manager handler. This replaces `startup` and - `shutdown` functions with a single context manager. - - Read more in the - [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - terms_of_service: Annotated[ - Optional[str], - Doc( - """ - A URL to the Terms of Service for your API. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more at the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - app = FastAPI(terms_of_service="http://example.com/terms/") - ``` - """ - ), - ] = None, - contact: Annotated[ - Optional[dict[str, Union[str, Any]]], - Doc( - """ - A dictionary with the contact information for the exposed API. - - It can contain several fields. - - * `name`: (`str`) The name of the contact person/organization. - * `url`: (`str`) A URL pointing to the contact information. MUST be in - the format of a URL. - * `email`: (`str`) The email address of the contact person/organization. - MUST be in the format of an email address. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more at the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - app = FastAPI( - contact={ - "name": "Deadpoolio the Amazing", - "url": "http://x-force.example.com/contact/", - "email": "dp@x-force.example.com", - } - ) - ``` - """ - ), - ] = None, - license_info: Annotated[ - Optional[dict[str, Union[str, Any]]], - Doc( - """ - A dictionary with the license information for the exposed API. - - It can contain several fields. - - * `name`: (`str`) **REQUIRED** (if a `license_info` is set). The - license name used for the API. - * `identifier`: (`str`) An [SPDX](https://spdx.dev/) license expression - for the API. The `identifier` field is mutually exclusive of the `url` - field. Available since OpenAPI 3.1.0, FastAPI 0.99.0. - * `url`: (`str`) A URL to the license used for the API. This MUST be - the format of a URL. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more at the - [FastAPI docs for Metadata and Docs URLs](https://fastapi.tiangolo.com/tutorial/metadata/#metadata-for-api). - - **Example** - - ```python - app = FastAPI( - license_info={ - "name": "Apache 2.0", - "url": "https://www.apache.org/licenses/LICENSE-2.0.html", - } - ) - ``` - """ - ), - ] = None, - openapi_prefix: Annotated[ - str, - Doc( - """ - A URL prefix for the OpenAPI URL. - """ - ), - deprecated( - """ - "openapi_prefix" has been deprecated in favor of "root_path", which - follows more closely the ASGI standard, is simpler, and more - automatic. - """ - ), - ] = "", - root_path: Annotated[ - str, - Doc( - """ - A path prefix handled by a proxy that is not seen by the application - but is seen by external clients, which affects things like Swagger UI. - - Read more about it at the - [FastAPI docs for Behind a Proxy](https://fastapi.tiangolo.com/advanced/behind-a-proxy/). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(root_path="/api/v1") - ``` - """ - ), - ] = "", - root_path_in_servers: Annotated[ - bool, - Doc( - """ - To disable automatically generating the URLs in the `servers` field - in the autogenerated OpenAPI using the `root_path`. - - Read more about it in the - [FastAPI docs for Behind a Proxy](https://fastapi.tiangolo.com/advanced/behind-a-proxy/#disable-automatic-server-from-root-path). - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI(root_path_in_servers=False) - ``` - """ - ), - ] = True, - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses to be shown in OpenAPI. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Additional Responses in OpenAPI](https://fastapi.tiangolo.com/advanced/additional-responses/). - - And in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - OpenAPI callbacks that should apply to all *path operations*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - webhooks: Annotated[ - Optional[routing.APIRouter], - Doc( - """ - Add OpenAPI webhooks. This is similar to `callbacks` but it doesn't - depend on specific *path operations*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - **Note**: This is available since OpenAPI 3.1.0, FastAPI 0.99.0. - - Read more about it in the - [FastAPI docs for OpenAPI Webhooks](https://fastapi.tiangolo.com/advanced/openapi-webhooks/). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark all *path operations* as deprecated. You probably don't need it, - but it's available. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#deprecate-a-path-operation). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) all the *path operations* in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - swagger_ui_parameters: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Parameters to configure Swagger UI, the autogenerated interactive API - documentation (by default at `/docs`). - - Read more about it in the - [FastAPI docs about how to Configure Swagger UI](https://fastapi.tiangolo.com/how-to/configure-swagger-ui/). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - separate_input_output_schemas: Annotated[ - bool, - Doc( - """ - Whether to generate separate OpenAPI schemas for request body and - response body when the results would be more precise. - - This is particularly useful when automatically generating clients. - - For example, if you have a model like: - - ```python - from pydantic import BaseModel - - class Item(BaseModel): - name: str - tags: list[str] = [] - ``` - - When `Item` is used for input, a request body, `tags` is not required, - the client doesn't have to provide it. - - But when using `Item` for output, for a response body, `tags` is always - available because it has a default value, even if it's just an empty - list. So, the client should be able to always expect it. - - In this case, there would be two different schemas, one for input and - another one for output. - - Read more about it in the - [FastAPI docs about how to separate schemas for input and output](https://fastapi.tiangolo.com/how-to/separate-openapi-schemas) - """ - ), - ] = True, - openapi_external_docs: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - This field allows you to provide additional external documentation links. - If provided, it must be a dictionary containing: - - * `description`: A brief description of the external documentation. - * `url`: The URL pointing to the external documentation. The value **MUST** - be a valid URL format. - - **Example**: - - ```python - from fastapi import FastAPI - - external_docs = { - "description": "Detailed API Reference", - "url": "https://example.com/api-docs", - } - - app = FastAPI(openapi_external_docs=external_docs) - ``` - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Extra keyword arguments to be stored in the app, not used by FastAPI - anywhere. - """ - ), - ], - ) -> None: - self.debug = debug - self.title = title - self.summary = summary - self.description = description - self.version = version - self.terms_of_service = terms_of_service - self.contact = contact - self.license_info = license_info - self.openapi_url = openapi_url - self.openapi_tags = openapi_tags - self.root_path_in_servers = root_path_in_servers - self.docs_url = docs_url - self.redoc_url = redoc_url - self.swagger_ui_oauth2_redirect_url = swagger_ui_oauth2_redirect_url - self.swagger_ui_init_oauth = swagger_ui_init_oauth - self.swagger_ui_parameters = swagger_ui_parameters - self.servers = servers or [] - self.separate_input_output_schemas = separate_input_output_schemas - self.openapi_external_docs = openapi_external_docs - self.extra = extra - self.openapi_version: Annotated[ - str, - Doc( - """ - The version string of OpenAPI. - - FastAPI will generate OpenAPI version 3.1.0, and will output that as - the OpenAPI version. But some tools, even though they might be - compatible with OpenAPI 3.1.0, might not recognize it as a valid. - - So you could override this value to trick those tools into using - the generated OpenAPI. Have in mind that this is a hack. But if you - avoid using features added in OpenAPI 3.1.0, it might work for your - use case. - - This is not passed as a parameter to the `FastAPI` class to avoid - giving the false idea that FastAPI would generate a different OpenAPI - schema. It is only available as an attribute. - - **Example** - - ```python - from fastapi import FastAPI - - app = FastAPI() - - app.openapi_version = "3.0.2" - ``` - """ - ), - ] = "3.1.0" - self.openapi_schema: Optional[dict[str, Any]] = None - if self.openapi_url: - assert self.title, "A title must be provided for OpenAPI, e.g.: 'My API'" - assert self.version, "A version must be provided for OpenAPI, e.g.: '2.1.0'" - # TODO: remove when discarding the openapi_prefix parameter - if openapi_prefix: - logger.warning( - '"openapi_prefix" has been deprecated in favor of "root_path", which ' - "follows more closely the ASGI standard, is simpler, and more " - "automatic. Check the docs at " - "https://fastapi.tiangolo.com/advanced/sub-applications/" - ) - self.webhooks: Annotated[ - routing.APIRouter, - Doc( - """ - The `app.webhooks` attribute is an `APIRouter` with the *path - operations* that will be used just for documentation of webhooks. - - Read more about it in the - [FastAPI docs for OpenAPI Webhooks](https://fastapi.tiangolo.com/advanced/openapi-webhooks/). - """ - ), - ] = webhooks or routing.APIRouter() - self.root_path = root_path or openapi_prefix - self.state: Annotated[ - State, - Doc( - """ - A state object for the application. This is the same object for the - entire application, it doesn't change from request to request. - - You normally wouldn't use this in FastAPI, for most of the cases you - would instead use FastAPI dependencies. - - This is simply inherited from Starlette. - - Read more about it in the - [Starlette docs for Applications](https://www.starlette.dev/applications/#storing-state-on-the-app-instance). - """ - ), - ] = State() - self.dependency_overrides: Annotated[ - dict[Callable[..., Any], Callable[..., Any]], - Doc( - """ - A dictionary with overrides for the dependencies. - - Each key is the original dependency callable, and the value is the - actual dependency that should be called. - - This is for testing, to replace expensive dependencies with testing - versions. - - Read more about it in the - [FastAPI docs for Testing Dependencies with Overrides](https://fastapi.tiangolo.com/advanced/testing-dependencies/). - """ - ), - ] = {} - self.router: routing.APIRouter = routing.APIRouter( - routes=routes, - redirect_slashes=redirect_slashes, - dependency_overrides_provider=self, - on_startup=on_startup, - on_shutdown=on_shutdown, - lifespan=lifespan, - default_response_class=default_response_class, - dependencies=dependencies, - callbacks=callbacks, - deprecated=deprecated, - include_in_schema=include_in_schema, - responses=responses, - generate_unique_id_function=generate_unique_id_function, - ) - self.exception_handlers: dict[ - Any, Callable[[Request, Any], Union[Response, Awaitable[Response]]] - ] = {} if exception_handlers is None else dict(exception_handlers) - self.exception_handlers.setdefault(HTTPException, http_exception_handler) - self.exception_handlers.setdefault( - RequestValidationError, request_validation_exception_handler - ) - self.exception_handlers.setdefault( - WebSocketRequestValidationError, - # Starlette still has incorrect type specification for the handlers - websocket_request_validation_exception_handler, # type: ignore - ) - - self.user_middleware: list[Middleware] = ( - [] if middleware is None else list(middleware) - ) - self.middleware_stack: Union[ASGIApp, None] = None - self.setup() - - def build_middleware_stack(self) -> ASGIApp: - # Duplicate/override from Starlette to add AsyncExitStackMiddleware - # inside of ExceptionMiddleware, inside of custom user middlewares - debug = self.debug - error_handler = None - exception_handlers: dict[Any, ExceptionHandler] = {} - - for key, value in self.exception_handlers.items(): - if key in (500, Exception): - error_handler = value - else: - exception_handlers[key] = value - - middleware = ( - [Middleware(ServerErrorMiddleware, handler=error_handler, debug=debug)] - + self.user_middleware - + [ - Middleware( - ExceptionMiddleware, handlers=exception_handlers, debug=debug - ), - # Add FastAPI-specific AsyncExitStackMiddleware for closing files. - # Before this was also used for closing dependencies with yield but - # those now have their own AsyncExitStack, to properly support - # streaming responses while keeping compatibility with the previous - # versions (as of writing 0.117.1) that allowed doing - # except HTTPException inside a dependency with yield. - # This needs to happen after user middlewares because those create a - # new contextvars context copy by using a new AnyIO task group. - # This AsyncExitStack preserves the context for contextvars, not - # strictly necessary for closing files but it was one of the original - # intentions. - # If the AsyncExitStack lived outside of the custom middlewares and - # contextvars were set, for example in a dependency with 'yield' - # in that internal contextvars context, the values would not be - # available in the outer context of the AsyncExitStack. - # By placing the middleware and the AsyncExitStack here, inside all - # user middlewares, the same context is used. - # This is currently not needed, only for closing files, but used to be - # important when dependencies with yield were closed here. - Middleware(AsyncExitStackMiddleware), - ] - ) - - app = self.router - for cls, args, kwargs in reversed(middleware): - app = cls(app, *args, **kwargs) - return app - - def openapi(self) -> dict[str, Any]: - """ - Generate the OpenAPI schema of the application. This is called by FastAPI - internally. - - The first time it is called it stores the result in the attribute - `app.openapi_schema`, and next times it is called, it just returns that same - result. To avoid the cost of generating the schema every time. - - If you need to modify the generated OpenAPI schema, you could modify it. - - Read more in the - [FastAPI docs for OpenAPI](https://fastapi.tiangolo.com/how-to/extending-openapi/). - """ - if not self.openapi_schema: - self.openapi_schema = get_openapi( - title=self.title, - version=self.version, - openapi_version=self.openapi_version, - summary=self.summary, - description=self.description, - terms_of_service=self.terms_of_service, - contact=self.contact, - license_info=self.license_info, - routes=self.routes, - webhooks=self.webhooks.routes, - tags=self.openapi_tags, - servers=self.servers, - separate_input_output_schemas=self.separate_input_output_schemas, - external_docs=self.openapi_external_docs, - ) - return self.openapi_schema - - def setup(self) -> None: - if self.openapi_url: - urls = (server_data.get("url") for server_data in self.servers) - server_urls = {url for url in urls if url} - - async def openapi(req: Request) -> JSONResponse: - root_path = req.scope.get("root_path", "").rstrip("/") - if root_path not in server_urls: - if root_path and self.root_path_in_servers: - self.servers.insert(0, {"url": root_path}) - server_urls.add(root_path) - return JSONResponse(self.openapi()) - - self.add_route(self.openapi_url, openapi, include_in_schema=False) - if self.openapi_url and self.docs_url: - - async def swagger_ui_html(req: Request) -> HTMLResponse: - root_path = req.scope.get("root_path", "").rstrip("/") - openapi_url = root_path + self.openapi_url - oauth2_redirect_url = self.swagger_ui_oauth2_redirect_url - if oauth2_redirect_url: - oauth2_redirect_url = root_path + oauth2_redirect_url - return get_swagger_ui_html( - openapi_url=openapi_url, - title=f"{self.title} - Swagger UI", - oauth2_redirect_url=oauth2_redirect_url, - init_oauth=self.swagger_ui_init_oauth, - swagger_ui_parameters=self.swagger_ui_parameters, - ) - - self.add_route(self.docs_url, swagger_ui_html, include_in_schema=False) - - if self.swagger_ui_oauth2_redirect_url: - - async def swagger_ui_redirect(req: Request) -> HTMLResponse: - return get_swagger_ui_oauth2_redirect_html() - - self.add_route( - self.swagger_ui_oauth2_redirect_url, - swagger_ui_redirect, - include_in_schema=False, - ) - if self.openapi_url and self.redoc_url: - - async def redoc_html(req: Request) -> HTMLResponse: - root_path = req.scope.get("root_path", "").rstrip("/") - openapi_url = root_path + self.openapi_url - return get_redoc_html( - openapi_url=openapi_url, title=f"{self.title} - ReDoc" - ) - - self.add_route(self.redoc_url, redoc_html, include_in_schema=False) - - async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None: - if self.root_path: - scope["root_path"] = self.root_path - await super().__call__(scope, receive, send) - - def add_api_route( - self, - path: str, - endpoint: Callable[..., Any], - *, - response_model: Any = Default(None), - status_code: Optional[int] = None, - tags: Optional[list[Union[str, Enum]]] = None, - dependencies: Optional[Sequence[Depends]] = None, - summary: Optional[str] = None, - description: Optional[str] = None, - response_description: str = "Successful Response", - responses: Optional[dict[Union[int, str], dict[str, Any]]] = None, - deprecated: Optional[bool] = None, - methods: Optional[list[str]] = None, - operation_id: Optional[str] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - include_in_schema: bool = True, - response_class: Union[type[Response], DefaultPlaceholder] = Default( - JSONResponse - ), - name: Optional[str] = None, - openapi_extra: Optional[dict[str, Any]] = None, - generate_unique_id_function: Callable[[routing.APIRoute], str] = Default( - generate_unique_id - ), - ) -> None: - self.router.add_api_route( - path, - endpoint=endpoint, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=methods, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def api_route( - self, - path: str, - *, - response_model: Any = Default(None), - status_code: Optional[int] = None, - tags: Optional[list[Union[str, Enum]]] = None, - dependencies: Optional[Sequence[Depends]] = None, - summary: Optional[str] = None, - description: Optional[str] = None, - response_description: str = "Successful Response", - responses: Optional[dict[Union[int, str], dict[str, Any]]] = None, - deprecated: Optional[bool] = None, - methods: Optional[list[str]] = None, - operation_id: Optional[str] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - include_in_schema: bool = True, - response_class: type[Response] = Default(JSONResponse), - name: Optional[str] = None, - openapi_extra: Optional[dict[str, Any]] = None, - generate_unique_id_function: Callable[[routing.APIRoute], str] = Default( - generate_unique_id - ), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.router.add_api_route( - path, - func, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=methods, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - return func - - return decorator - - def add_api_websocket_route( - self, - path: str, - endpoint: Callable[..., Any], - name: Optional[str] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - ) -> None: - self.router.add_api_websocket_route( - path, - endpoint, - name=name, - dependencies=dependencies, - ) - - def websocket( - self, - path: Annotated[ - str, - Doc( - """ - WebSocket path. - """ - ), - ], - name: Annotated[ - Optional[str], - Doc( - """ - A name for the WebSocket. Only used internally. - """ - ), - ] = None, - *, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be used for this - WebSocket. - - Read more about it in the - [FastAPI docs for WebSockets](https://fastapi.tiangolo.com/advanced/websockets/). - """ - ), - ] = None, - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Decorate a WebSocket function. - - Read more about it in the - [FastAPI docs for WebSockets](https://fastapi.tiangolo.com/advanced/websockets/). - - **Example** - - ```python - from fastapi import FastAPI, WebSocket - - app = FastAPI() - - @app.websocket("/ws") - async def websocket_endpoint(websocket: WebSocket): - await websocket.accept() - while True: - data = await websocket.receive_text() - await websocket.send_text(f"Message text was: {data}") - ``` - """ - - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_api_websocket_route( - path, - func, - name=name, - dependencies=dependencies, - ) - return func - - return decorator - - def include_router( - self, - router: Annotated[routing.APIRouter, Doc("The `APIRouter` to include.")], - *, - prefix: Annotated[str, Doc("An optional path prefix for the router.")] = "", - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to all the *path operations* in this - router. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to all the - *path operations* in this router. - - Read more about it in the - [FastAPI docs for Bigger Applications - Multiple Files](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - - **Example** - - ```python - from fastapi import Depends, FastAPI - - from .dependencies import get_token_header - from .internal import admin - - app = FastAPI() - - app.include_router( - admin.router, - dependencies=[Depends(get_token_header)], - ) - ``` - """ - ), - ] = None, - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses to be shown in OpenAPI. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Additional Responses in OpenAPI](https://fastapi.tiangolo.com/advanced/additional-responses/). - - And in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark all the *path operations* in this router as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - **Example** - - ```python - from fastapi import FastAPI - - from .internal import old_api - - app = FastAPI() - - app.include_router( - old_api.router, - deprecated=True, - ) - ``` - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include (or not) all the *path operations* in this router in the - generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - **Example** - - ```python - from fastapi import FastAPI - - from .internal import old_api - - app = FastAPI() - - app.include_router( - old_api.router, - include_in_schema=False, - ) - ``` - """ - ), - ] = True, - default_response_class: Annotated[ - type[Response], - Doc( - """ - Default response class to be used for the *path operations* in this - router. - - Read more in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#default-response-class). - - **Example** - - ```python - from fastapi import FastAPI - from fastapi.responses import ORJSONResponse - - from .internal import old_api - - app = FastAPI() - - app.include_router( - old_api.router, - default_response_class=ORJSONResponse, - ) - ``` - """ - ), - ] = Default(JSONResponse), - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> None: - """ - Include an `APIRouter` in the same app. - - Read more about it in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/). - - ## Example - - ```python - from fastapi import FastAPI - - from .users import users_router - - app = FastAPI() - - app.include_router(users_router) - ``` - """ - self.router.include_router( - router, - prefix=prefix, - tags=tags, - dependencies=dependencies, - responses=responses, - deprecated=deprecated, - include_in_schema=include_in_schema, - default_response_class=default_response_class, - callbacks=callbacks, - generate_unique_id_function=generate_unique_id_function, - ) - - def get( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP GET operation. - - ## Example - - ```python - from fastapi import FastAPI - - app = FastAPI() - - @app.get("/items/") - def read_items(): - return [{"name": "Empanada"}, {"name": "Arepa"}] - ``` - """ - return self.router.get( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def put( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP PUT operation. - - ## Example - - ```python - from fastapi import FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - - @app.put("/items/{item_id}") - def replace_item(item_id: str, item: Item): - return {"message": "Item replaced", "id": item_id} - ``` - """ - return self.router.put( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def post( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP POST operation. - - ## Example - - ```python - from fastapi import FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - - @app.post("/items/") - def create_item(item: Item): - return {"message": "Item created"} - ``` - """ - return self.router.post( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def delete( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP DELETE operation. - - ## Example - - ```python - from fastapi import FastAPI - - app = FastAPI() - - @app.delete("/items/{item_id}") - def delete_item(item_id: str): - return {"message": "Item deleted"} - ``` - """ - return self.router.delete( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def options( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP OPTIONS operation. - - ## Example - - ```python - from fastapi import FastAPI - - app = FastAPI() - - @app.options("/items/") - def get_item_options(): - return {"additions": ["Aji", "Guacamole"]} - ``` - """ - return self.router.options( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def head( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP HEAD operation. - - ## Example - - ```python - from fastapi import FastAPI, Response - - app = FastAPI() - - @app.head("/items/", status_code=204) - def get_items_headers(response: Response): - response.headers["X-Cat-Dog"] = "Alone in the world" - ``` - """ - return self.router.head( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def patch( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP PATCH operation. - - ## Example - - ```python - from fastapi import FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - - @app.patch("/items/") - def update_item(item: Item): - return {"message": "Item updated in place"} - ``` - """ - return self.router.patch( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def trace( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[routing.APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP TRACE operation. - - ## Example - - ```python - from fastapi import FastAPI - - app = FastAPI() - - @app.trace("/items/{item_id}") - def trace_item(item_id: str): - return None - ``` - """ - return self.router.trace( - path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def websocket_route( - self, path: str, name: Union[str, None] = None - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.router.add_websocket_route(path, func, name=name) - return func - - return decorator - - @deprecated( - """ - on_event is deprecated, use lifespan event handlers instead. - - Read more about it in the - [FastAPI docs for Lifespan Events](https://fastapi.tiangolo.com/advanced/events/). - """ - ) - def on_event( - self, - event_type: Annotated[ - str, - Doc( - """ - The type of event. `startup` or `shutdown`. - """ - ), - ], - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add an event handler for the application. - - `on_event` is deprecated, use `lifespan` event handlers instead. - - Read more about it in the - [FastAPI docs for Lifespan Events](https://fastapi.tiangolo.com/advanced/events/#alternative-events-deprecated). - """ - return self.router.on_event(event_type) - - def middleware( - self, - middleware_type: Annotated[ - str, - Doc( - """ - The type of middleware. Currently only supports `http`. - """ - ), - ], - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a middleware to the application. - - Read more about it in the - [FastAPI docs for Middleware](https://fastapi.tiangolo.com/tutorial/middleware/). - - ## Example - - ```python - import time - from typing import Awaitable, Callable - - from fastapi import FastAPI, Request, Response - - app = FastAPI() - - - @app.middleware("http") - async def add_process_time_header( - request: Request, call_next: Callable[[Request], Awaitable[Response]] - ) -> Response: - start_time = time.time() - response = await call_next(request) - process_time = time.time() - start_time - response.headers["X-Process-Time"] = str(process_time) - return response - ``` - """ - - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_middleware(BaseHTTPMiddleware, dispatch=func) - return func - - return decorator - - def exception_handler( - self, - exc_class_or_status_code: Annotated[ - Union[int, type[Exception]], - Doc( - """ - The Exception class this would handle, or a status code. - """ - ), - ], - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add an exception handler to the app. - - Read more about it in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/). - - ## Example - - ```python - from fastapi import FastAPI, Request - from fastapi.responses import JSONResponse - - - class UnicornException(Exception): - def __init__(self, name: str): - self.name = name - - - app = FastAPI() - - - @app.exception_handler(UnicornException) - async def unicorn_exception_handler(request: Request, exc: UnicornException): - return JSONResponse( - status_code=418, - content={"message": f"Oops! {exc.name} did something. There goes a rainbow..."}, - ) - ``` - """ - - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_exception_handler(exc_class_or_status_code, func) - return func - - return decorator diff --git a/.env/Lib/site-packages/fastapi/background.py b/.env/Lib/site-packages/fastapi/background.py deleted file mode 100644 index 20803ba..0000000 --- a/.env/Lib/site-packages/fastapi/background.py +++ /dev/null @@ -1,60 +0,0 @@ -from typing import Annotated, Any, Callable - -from annotated_doc import Doc -from starlette.background import BackgroundTasks as StarletteBackgroundTasks -from typing_extensions import ParamSpec - -P = ParamSpec("P") - - -class BackgroundTasks(StarletteBackgroundTasks): - """ - A collection of background tasks that will be called after a response has been - sent to the client. - - Read more about it in the - [FastAPI docs for Background Tasks](https://fastapi.tiangolo.com/tutorial/background-tasks/). - - ## Example - - ```python - from fastapi import BackgroundTasks, FastAPI - - app = FastAPI() - - - def write_notification(email: str, message=""): - with open("log.txt", mode="w") as email_file: - content = f"notification for {email}: {message}" - email_file.write(content) - - - @app.post("/send-notification/{email}") - async def send_notification(email: str, background_tasks: BackgroundTasks): - background_tasks.add_task(write_notification, email, message="some notification") - return {"message": "Notification sent in the background"} - ``` - """ - - def add_task( - self, - func: Annotated[ - Callable[P, Any], - Doc( - """ - The function to call after the response is sent. - - It can be a regular `def` function or an `async def` function. - """ - ), - ], - *args: P.args, - **kwargs: P.kwargs, - ) -> None: - """ - Add a function to be called in the background after the response is sent. - - Read more about it in the - [FastAPI docs for Background Tasks](https://fastapi.tiangolo.com/tutorial/background-tasks/). - """ - return super().add_task(func, *args, **kwargs) diff --git a/.env/Lib/site-packages/fastapi/cli.py b/.env/Lib/site-packages/fastapi/cli.py deleted file mode 100644 index 8d3301e..0000000 --- a/.env/Lib/site-packages/fastapi/cli.py +++ /dev/null @@ -1,13 +0,0 @@ -try: - from fastapi_cli.cli import main as cli_main - -except ImportError: # pragma: no cover - cli_main = None # type: ignore - - -def main() -> None: - if not cli_main: # type: ignore[truthy-function] - message = 'To use the fastapi command, please install "fastapi[standard]":\n\n\tpip install "fastapi[standard]"\n' - print(message) - raise RuntimeError(message) # noqa: B904 - cli_main() diff --git a/.env/Lib/site-packages/fastapi/concurrency.py b/.env/Lib/site-packages/fastapi/concurrency.py deleted file mode 100644 index 76a5a2e..0000000 --- a/.env/Lib/site-packages/fastapi/concurrency.py +++ /dev/null @@ -1,41 +0,0 @@ -from collections.abc import AsyncGenerator -from contextlib import AbstractContextManager -from contextlib import asynccontextmanager as asynccontextmanager -from typing import TypeVar - -import anyio.to_thread -from anyio import CapacityLimiter -from starlette.concurrency import iterate_in_threadpool as iterate_in_threadpool # noqa -from starlette.concurrency import run_in_threadpool as run_in_threadpool # noqa -from starlette.concurrency import ( # noqa - run_until_first_complete as run_until_first_complete, -) - -_T = TypeVar("_T") - - -@asynccontextmanager -async def contextmanager_in_threadpool( - cm: AbstractContextManager[_T], -) -> AsyncGenerator[_T, None]: - # blocking __exit__ from running waiting on a free thread - # can create race conditions/deadlocks if the context manager itself - # has its own internal pool (e.g. a database connection pool) - # to avoid this we let __exit__ run without a capacity limit - # since we're creating a new limiter for each call, any non-zero limit - # works (1 is arbitrary) - exit_limiter = CapacityLimiter(1) - try: - yield await run_in_threadpool(cm.__enter__) - except Exception as e: - ok = bool( - await anyio.to_thread.run_sync( - cm.__exit__, type(e), e, e.__traceback__, limiter=exit_limiter - ) - ) - if not ok: - raise e - else: - await anyio.to_thread.run_sync( - cm.__exit__, None, None, None, limiter=exit_limiter - ) diff --git a/.env/Lib/site-packages/fastapi/datastructures.py b/.env/Lib/site-packages/fastapi/datastructures.py deleted file mode 100644 index 2bf5fdb..0000000 --- a/.env/Lib/site-packages/fastapi/datastructures.py +++ /dev/null @@ -1,183 +0,0 @@ -from collections.abc import Mapping -from typing import ( - Annotated, - Any, - BinaryIO, - Callable, - Optional, - TypeVar, - cast, -) - -from annotated_doc import Doc -from pydantic import GetJsonSchemaHandler -from starlette.datastructures import URL as URL # noqa: F401 -from starlette.datastructures import Address as Address # noqa: F401 -from starlette.datastructures import FormData as FormData # noqa: F401 -from starlette.datastructures import Headers as Headers # noqa: F401 -from starlette.datastructures import QueryParams as QueryParams # noqa: F401 -from starlette.datastructures import State as State # noqa: F401 -from starlette.datastructures import UploadFile as StarletteUploadFile - - -class UploadFile(StarletteUploadFile): - """ - A file uploaded in a request. - - Define it as a *path operation function* (or dependency) parameter. - - If you are using a regular `def` function, you can use the `upload_file.file` - attribute to access the raw standard Python file (blocking, not async), useful and - needed for non-async code. - - Read more about it in the - [FastAPI docs for Request Files](https://fastapi.tiangolo.com/tutorial/request-files/). - - ## Example - - ```python - from typing import Annotated - - from fastapi import FastAPI, File, UploadFile - - app = FastAPI() - - - @app.post("/files/") - async def create_file(file: Annotated[bytes, File()]): - return {"file_size": len(file)} - - - @app.post("/uploadfile/") - async def create_upload_file(file: UploadFile): - return {"filename": file.filename} - ``` - """ - - file: Annotated[ - BinaryIO, - Doc("The standard Python file object (non-async)."), - ] - filename: Annotated[Optional[str], Doc("The original file name.")] - size: Annotated[Optional[int], Doc("The size of the file in bytes.")] - headers: Annotated[Headers, Doc("The headers of the request.")] - content_type: Annotated[ - Optional[str], Doc("The content type of the request, from the headers.") - ] - - async def write( - self, - data: Annotated[ - bytes, - Doc( - """ - The bytes to write to the file. - """ - ), - ], - ) -> None: - """ - Write some bytes to the file. - - You normally wouldn't use this from a file you read in a request. - - To be awaitable, compatible with async, this is run in threadpool. - """ - return await super().write(data) - - async def read( - self, - size: Annotated[ - int, - Doc( - """ - The number of bytes to read from the file. - """ - ), - ] = -1, - ) -> bytes: - """ - Read some bytes from the file. - - To be awaitable, compatible with async, this is run in threadpool. - """ - return await super().read(size) - - async def seek( - self, - offset: Annotated[ - int, - Doc( - """ - The position in bytes to seek to in the file. - """ - ), - ], - ) -> None: - """ - Move to a position in the file. - - Any next read or write will be done from that position. - - To be awaitable, compatible with async, this is run in threadpool. - """ - return await super().seek(offset) - - async def close(self) -> None: - """ - Close the file. - - To be awaitable, compatible with async, this is run in threadpool. - """ - return await super().close() - - @classmethod - def _validate(cls, __input_value: Any, _: Any) -> "UploadFile": - if not isinstance(__input_value, StarletteUploadFile): - raise ValueError(f"Expected UploadFile, received: {type(__input_value)}") - return cast(UploadFile, __input_value) - - @classmethod - def __get_pydantic_json_schema__( - cls, core_schema: Mapping[str, Any], handler: GetJsonSchemaHandler - ) -> dict[str, Any]: - return {"type": "string", "format": "binary"} - - @classmethod - def __get_pydantic_core_schema__( - cls, source: type[Any], handler: Callable[[Any], Mapping[str, Any]] - ) -> Mapping[str, Any]: - from ._compat.v2 import with_info_plain_validator_function - - return with_info_plain_validator_function(cls._validate) - - -class DefaultPlaceholder: - """ - You shouldn't use this class directly. - - It's used internally to recognize when a default value has been overwritten, even - if the overridden default value was truthy. - """ - - def __init__(self, value: Any): - self.value = value - - def __bool__(self) -> bool: - return bool(self.value) - - def __eq__(self, o: object) -> bool: - return isinstance(o, DefaultPlaceholder) and o.value == self.value - - -DefaultType = TypeVar("DefaultType") - - -def Default(value: DefaultType) -> DefaultType: - """ - You shouldn't use this function directly. - - It's used internally to recognize when a default value has been overwritten, even - if the overridden default value was truthy. - """ - return DefaultPlaceholder(value) # type: ignore diff --git a/.env/Lib/site-packages/fastapi/dependencies/__init__.py b/.env/Lib/site-packages/fastapi/dependencies/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/fastapi/dependencies/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/dependencies/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 0bb6d80be46dd850da9993d7d6f796317e515f01..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 193 zcmX@j%ge<81Sbx(WrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxdmEdd@6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdilIY~;;_lhPbtkwwJTx;TF3~*#URE< MW=2NFB4!{90N;r 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 diff --git a/.env/Lib/site-packages/fastapi/dependencies/models.py b/.env/Lib/site-packages/fastapi/dependencies/models.py deleted file mode 100644 index 5839232..0000000 --- a/.env/Lib/site-packages/fastapi/dependencies/models.py +++ /dev/null @@ -1,193 +0,0 @@ -import inspect -import sys -from dataclasses import dataclass, field -from functools import cached_property, partial -from typing import Any, Callable, Optional, Union - -from fastapi._compat import ModelField -from fastapi.security.base import SecurityBase -from fastapi.types import DependencyCacheKey -from typing_extensions import Literal - -if sys.version_info >= (3, 13): # pragma: no cover - from inspect import iscoroutinefunction -else: # pragma: no cover - from asyncio import iscoroutinefunction - - -def _unwrapped_call(call: Optional[Callable[..., Any]]) -> Any: - if call is None: - return call # pragma: no cover - unwrapped = inspect.unwrap(_impartial(call)) - return unwrapped - - -def _impartial(func: Callable[..., Any]) -> Callable[..., Any]: - while isinstance(func, partial): - func = func.func - return func - - -@dataclass -class Dependant: - path_params: list[ModelField] = field(default_factory=list) - query_params: list[ModelField] = field(default_factory=list) - header_params: list[ModelField] = field(default_factory=list) - cookie_params: list[ModelField] = field(default_factory=list) - body_params: list[ModelField] = field(default_factory=list) - dependencies: list["Dependant"] = field(default_factory=list) - name: Optional[str] = None - call: Optional[Callable[..., Any]] = None - request_param_name: Optional[str] = None - websocket_param_name: Optional[str] = None - http_connection_param_name: Optional[str] = None - response_param_name: Optional[str] = None - background_tasks_param_name: Optional[str] = None - security_scopes_param_name: Optional[str] = None - own_oauth_scopes: Optional[list[str]] = None - parent_oauth_scopes: Optional[list[str]] = None - use_cache: bool = True - path: Optional[str] = None - scope: Union[Literal["function", "request"], None] = None - - @cached_property - def oauth_scopes(self) -> list[str]: - scopes = self.parent_oauth_scopes.copy() if self.parent_oauth_scopes else [] - # This doesn't use a set to preserve order, just in case - for scope in self.own_oauth_scopes or []: - if scope not in scopes: - scopes.append(scope) - return scopes - - @cached_property - def cache_key(self) -> DependencyCacheKey: - scopes_for_cache = ( - tuple(sorted(set(self.oauth_scopes or []))) if self._uses_scopes else () - ) - return ( - self.call, - scopes_for_cache, - self.computed_scope or "", - ) - - @cached_property - def _uses_scopes(self) -> bool: - if self.own_oauth_scopes: - return True - if self.security_scopes_param_name is not None: - return True - if self._is_security_scheme: - return True - for sub_dep in self.dependencies: - if sub_dep._uses_scopes: - return True - return False - - @cached_property - def _is_security_scheme(self) -> bool: - if self.call is None: - return False # pragma: no cover - unwrapped = _unwrapped_call(self.call) - return isinstance(unwrapped, SecurityBase) - - # Mainly to get the type of SecurityBase, but it's the same self.call - @cached_property - def _security_scheme(self) -> SecurityBase: - unwrapped = _unwrapped_call(self.call) - assert isinstance(unwrapped, SecurityBase) - return unwrapped - - @cached_property - def _security_dependencies(self) -> list["Dependant"]: - security_deps = [dep for dep in self.dependencies if dep._is_security_scheme] - return security_deps - - @cached_property - def is_gen_callable(self) -> bool: - if self.call is None: - return False # pragma: no cover - if inspect.isgeneratorfunction( - _impartial(self.call) - ) or inspect.isgeneratorfunction(_unwrapped_call(self.call)): - return True - if inspect.isclass(_unwrapped_call(self.call)): - return False - dunder_call = getattr(_impartial(self.call), "__call__", None) # noqa: B004 - if dunder_call is None: - return False # pragma: no cover - if inspect.isgeneratorfunction( - _impartial(dunder_call) - ) or inspect.isgeneratorfunction(_unwrapped_call(dunder_call)): - return True - dunder_unwrapped_call = getattr(_unwrapped_call(self.call), "__call__", None) # noqa: B004 - if dunder_unwrapped_call is None: - return False # pragma: no cover - if inspect.isgeneratorfunction( - _impartial(dunder_unwrapped_call) - ) or inspect.isgeneratorfunction(_unwrapped_call(dunder_unwrapped_call)): - return True - return False - - @cached_property - def is_async_gen_callable(self) -> bool: - if self.call is None: - return False # pragma: no cover - if inspect.isasyncgenfunction( - _impartial(self.call) - ) or inspect.isasyncgenfunction(_unwrapped_call(self.call)): - return True - if inspect.isclass(_unwrapped_call(self.call)): - return False - dunder_call = getattr(_impartial(self.call), "__call__", None) # noqa: B004 - if dunder_call is None: - return False # pragma: no cover - if inspect.isasyncgenfunction( - _impartial(dunder_call) - ) or inspect.isasyncgenfunction(_unwrapped_call(dunder_call)): - return True - dunder_unwrapped_call = getattr(_unwrapped_call(self.call), "__call__", None) # noqa: B004 - if dunder_unwrapped_call is None: - return False # pragma: no cover - if inspect.isasyncgenfunction( - _impartial(dunder_unwrapped_call) - ) or inspect.isasyncgenfunction(_unwrapped_call(dunder_unwrapped_call)): - return True - return False - - @cached_property - def is_coroutine_callable(self) -> bool: - if self.call is None: - return False # pragma: no cover - if inspect.isroutine(_impartial(self.call)) and iscoroutinefunction( - _impartial(self.call) - ): - return True - if inspect.isroutine(_unwrapped_call(self.call)) and iscoroutinefunction( - _unwrapped_call(self.call) - ): - return True - if inspect.isclass(_unwrapped_call(self.call)): - return False - dunder_call = getattr(_impartial(self.call), "__call__", None) # noqa: B004 - if dunder_call is None: - return False # pragma: no cover - if iscoroutinefunction(_impartial(dunder_call)) or iscoroutinefunction( - _unwrapped_call(dunder_call) - ): - return True - dunder_unwrapped_call = getattr(_unwrapped_call(self.call), "__call__", None) # noqa: B004 - if dunder_unwrapped_call is None: - return False # pragma: no cover - if iscoroutinefunction( - _impartial(dunder_unwrapped_call) - ) or iscoroutinefunction(_unwrapped_call(dunder_unwrapped_call)): - return True - return False - - @cached_property - def computed_scope(self) -> Union[str, None]: - if self.scope: - return self.scope - if self.is_gen_callable or self.is_async_gen_callable: - return "request" - return None diff --git a/.env/Lib/site-packages/fastapi/dependencies/utils.py b/.env/Lib/site-packages/fastapi/dependencies/utils.py deleted file mode 100644 index b8f7f94..0000000 --- a/.env/Lib/site-packages/fastapi/dependencies/utils.py +++ /dev/null @@ -1,1039 +0,0 @@ -import dataclasses -import inspect -import sys -from collections.abc import Coroutine, Mapping, Sequence -from contextlib import AsyncExitStack, contextmanager -from copy import copy, deepcopy -from dataclasses import dataclass -from typing import ( - Annotated, - Any, - Callable, - ForwardRef, - Optional, - Union, - cast, -) - -import anyio -from fastapi import params -from fastapi._compat import ( - ModelField, - RequiredParam, - Undefined, - _regenerate_error_with_loc, - copy_field_info, - create_body_model, - evaluate_forwardref, - field_annotation_is_scalar, - get_cached_model_fields, - get_missing_field_error, - is_bytes_field, - is_bytes_sequence_field, - is_scalar_field, - is_scalar_sequence_field, - is_sequence_field, - is_uploadfile_or_nonable_uploadfile_annotation, - is_uploadfile_sequence_annotation, - lenient_issubclass, - sequence_types, - serialize_sequence_value, - value_is_sequence, -) -from fastapi.background import BackgroundTasks -from fastapi.concurrency import ( - asynccontextmanager, - contextmanager_in_threadpool, -) -from fastapi.dependencies.models import Dependant -from fastapi.exceptions import DependencyScopeError -from fastapi.logger import logger -from fastapi.security.oauth2 import SecurityScopes -from fastapi.types import DependencyCacheKey -from fastapi.utils import create_model_field, get_path_param_names -from pydantic import BaseModel, Json -from pydantic.fields import FieldInfo -from starlette.background import BackgroundTasks as StarletteBackgroundTasks -from starlette.concurrency import run_in_threadpool -from starlette.datastructures import ( - FormData, - Headers, - ImmutableMultiDict, - QueryParams, - UploadFile, -) -from starlette.requests import HTTPConnection, Request -from starlette.responses import Response -from starlette.websockets import WebSocket -from typing_extensions import Literal, get_args, get_origin -from typing_inspection.typing_objects import is_typealiastype - -multipart_not_installed_error = ( - 'Form data requires "python-multipart" to be installed. \n' - 'You can install "python-multipart" with: \n\n' - "pip install python-multipart\n" -) -multipart_incorrect_install_error = ( - 'Form data requires "python-multipart" to be installed. ' - 'It seems you installed "multipart" instead. \n' - 'You can remove "multipart" with: \n\n' - "pip uninstall multipart\n\n" - 'And then install "python-multipart" with: \n\n' - "pip install python-multipart\n" -) - - -def ensure_multipart_is_installed() -> None: - try: - from python_multipart import __version__ - - # Import an attribute that can be mocked/deleted in testing - assert __version__ > "0.0.12" - except (ImportError, AssertionError): - try: - # __version__ is available in both multiparts, and can be mocked - from multipart import __version__ # type: ignore[no-redef,import-untyped] - - assert __version__ - try: - # parse_options_header is only available in the right multipart - from multipart.multipart import ( # type: ignore[import-untyped] - parse_options_header, - ) - - assert parse_options_header - except ImportError: - logger.error(multipart_incorrect_install_error) - raise RuntimeError(multipart_incorrect_install_error) from None - except ImportError: - logger.error(multipart_not_installed_error) - raise RuntimeError(multipart_not_installed_error) from None - - -def get_parameterless_sub_dependant(*, depends: params.Depends, path: str) -> Dependant: - assert callable(depends.dependency), ( - "A parameter-less dependency must have a callable dependency" - ) - own_oauth_scopes: list[str] = [] - if isinstance(depends, params.Security) and depends.scopes: - own_oauth_scopes.extend(depends.scopes) - return get_dependant( - path=path, - call=depends.dependency, - scope=depends.scope, - own_oauth_scopes=own_oauth_scopes, - ) - - -def get_flat_dependant( - dependant: Dependant, - *, - skip_repeats: bool = False, - visited: Optional[list[DependencyCacheKey]] = None, - parent_oauth_scopes: Optional[list[str]] = None, -) -> Dependant: - if visited is None: - visited = [] - visited.append(dependant.cache_key) - use_parent_oauth_scopes = (parent_oauth_scopes or []) + ( - dependant.oauth_scopes or [] - ) - - flat_dependant = Dependant( - path_params=dependant.path_params.copy(), - query_params=dependant.query_params.copy(), - header_params=dependant.header_params.copy(), - cookie_params=dependant.cookie_params.copy(), - body_params=dependant.body_params.copy(), - name=dependant.name, - call=dependant.call, - request_param_name=dependant.request_param_name, - websocket_param_name=dependant.websocket_param_name, - http_connection_param_name=dependant.http_connection_param_name, - response_param_name=dependant.response_param_name, - background_tasks_param_name=dependant.background_tasks_param_name, - security_scopes_param_name=dependant.security_scopes_param_name, - own_oauth_scopes=dependant.own_oauth_scopes, - parent_oauth_scopes=use_parent_oauth_scopes, - use_cache=dependant.use_cache, - path=dependant.path, - scope=dependant.scope, - ) - for sub_dependant in dependant.dependencies: - if skip_repeats and sub_dependant.cache_key in visited: - continue - flat_sub = get_flat_dependant( - sub_dependant, - skip_repeats=skip_repeats, - visited=visited, - parent_oauth_scopes=flat_dependant.oauth_scopes, - ) - flat_dependant.dependencies.append(flat_sub) - flat_dependant.path_params.extend(flat_sub.path_params) - flat_dependant.query_params.extend(flat_sub.query_params) - flat_dependant.header_params.extend(flat_sub.header_params) - flat_dependant.cookie_params.extend(flat_sub.cookie_params) - flat_dependant.body_params.extend(flat_sub.body_params) - flat_dependant.dependencies.extend(flat_sub.dependencies) - - return flat_dependant - - -def _get_flat_fields_from_params(fields: list[ModelField]) -> list[ModelField]: - if not fields: - return fields - first_field = fields[0] - if len(fields) == 1 and lenient_issubclass(first_field.type_, BaseModel): - fields_to_extract = get_cached_model_fields(first_field.type_) - return fields_to_extract - return fields - - -def get_flat_params(dependant: Dependant) -> list[ModelField]: - flat_dependant = get_flat_dependant(dependant, skip_repeats=True) - path_params = _get_flat_fields_from_params(flat_dependant.path_params) - query_params = _get_flat_fields_from_params(flat_dependant.query_params) - header_params = _get_flat_fields_from_params(flat_dependant.header_params) - cookie_params = _get_flat_fields_from_params(flat_dependant.cookie_params) - return path_params + query_params + header_params + cookie_params - - -def _get_signature(call: Callable[..., Any]) -> inspect.Signature: - if sys.version_info >= (3, 10): - try: - signature = inspect.signature(call, eval_str=True) - except NameError: - # Handle type annotations with if TYPE_CHECKING, not used by FastAPI - # e.g. dependency return types - if sys.version_info >= (3, 14): - from annotationlib import Format - - signature = inspect.signature(call, annotation_format=Format.FORWARDREF) - else: - signature = inspect.signature(call) - else: - signature = inspect.signature(call) - return signature - - -def get_typed_signature(call: Callable[..., Any]) -> inspect.Signature: - signature = _get_signature(call) - unwrapped = inspect.unwrap(call) - globalns = getattr(unwrapped, "__globals__", {}) - typed_params = [ - inspect.Parameter( - name=param.name, - kind=param.kind, - default=param.default, - annotation=get_typed_annotation(param.annotation, globalns), - ) - for param in signature.parameters.values() - ] - typed_signature = inspect.Signature(typed_params) - return typed_signature - - -def get_typed_annotation(annotation: Any, globalns: dict[str, Any]) -> Any: - if isinstance(annotation, str): - annotation = ForwardRef(annotation) - annotation = evaluate_forwardref(annotation, globalns, globalns) - if annotation is type(None): - return None - return annotation - - -def get_typed_return_annotation(call: Callable[..., Any]) -> Any: - signature = _get_signature(call) - unwrapped = inspect.unwrap(call) - annotation = signature.return_annotation - - if annotation is inspect.Signature.empty: - return None - - globalns = getattr(unwrapped, "__globals__", {}) - return get_typed_annotation(annotation, globalns) - - -def get_dependant( - *, - path: str, - call: Callable[..., Any], - name: Optional[str] = None, - own_oauth_scopes: Optional[list[str]] = None, - parent_oauth_scopes: Optional[list[str]] = None, - use_cache: bool = True, - scope: Union[Literal["function", "request"], None] = None, -) -> Dependant: - dependant = Dependant( - call=call, - name=name, - path=path, - use_cache=use_cache, - scope=scope, - own_oauth_scopes=own_oauth_scopes, - parent_oauth_scopes=parent_oauth_scopes, - ) - current_scopes = (parent_oauth_scopes or []) + (own_oauth_scopes or []) - path_param_names = get_path_param_names(path) - endpoint_signature = get_typed_signature(call) - signature_params = endpoint_signature.parameters - for param_name, param in signature_params.items(): - is_path_param = param_name in path_param_names - param_details = analyze_param( - param_name=param_name, - annotation=param.annotation, - value=param.default, - is_path_param=is_path_param, - ) - if param_details.depends is not None: - assert param_details.depends.dependency - if ( - (dependant.is_gen_callable or dependant.is_async_gen_callable) - and dependant.computed_scope == "request" - and param_details.depends.scope == "function" - ): - assert dependant.call - raise DependencyScopeError( - f'The dependency "{dependant.call.__name__}" has a scope of ' - '"request", it cannot depend on dependencies with scope "function".' - ) - sub_own_oauth_scopes: list[str] = [] - if isinstance(param_details.depends, params.Security): - if param_details.depends.scopes: - sub_own_oauth_scopes = list(param_details.depends.scopes) - sub_dependant = get_dependant( - path=path, - call=param_details.depends.dependency, - name=param_name, - own_oauth_scopes=sub_own_oauth_scopes, - parent_oauth_scopes=current_scopes, - use_cache=param_details.depends.use_cache, - scope=param_details.depends.scope, - ) - dependant.dependencies.append(sub_dependant) - continue - if add_non_field_param_to_dependency( - param_name=param_name, - type_annotation=param_details.type_annotation, - dependant=dependant, - ): - assert param_details.field is None, ( - f"Cannot specify multiple FastAPI annotations for {param_name!r}" - ) - continue - assert param_details.field is not None - if isinstance(param_details.field.field_info, params.Body): - dependant.body_params.append(param_details.field) - else: - add_param_to_fields(field=param_details.field, dependant=dependant) - return dependant - - -def add_non_field_param_to_dependency( - *, param_name: str, type_annotation: Any, dependant: Dependant -) -> Optional[bool]: - if lenient_issubclass(type_annotation, Request): - dependant.request_param_name = param_name - return True - elif lenient_issubclass(type_annotation, WebSocket): - dependant.websocket_param_name = param_name - return True - elif lenient_issubclass(type_annotation, HTTPConnection): - dependant.http_connection_param_name = param_name - return True - elif lenient_issubclass(type_annotation, Response): - dependant.response_param_name = param_name - return True - elif lenient_issubclass(type_annotation, StarletteBackgroundTasks): - dependant.background_tasks_param_name = param_name - return True - elif lenient_issubclass(type_annotation, SecurityScopes): - dependant.security_scopes_param_name = param_name - return True - return None - - -@dataclass -class ParamDetails: - type_annotation: Any - depends: Optional[params.Depends] - field: Optional[ModelField] - - -def analyze_param( - *, - param_name: str, - annotation: Any, - value: Any, - is_path_param: bool, -) -> ParamDetails: - field_info = None - depends = None - type_annotation: Any = Any - use_annotation: Any = Any - if is_typealiastype(annotation): - # unpack in case PEP 695 type syntax is used - annotation = annotation.__value__ - if annotation is not inspect.Signature.empty: - use_annotation = annotation - type_annotation = annotation - # Extract Annotated info - if get_origin(use_annotation) is Annotated: - annotated_args = get_args(annotation) - type_annotation = annotated_args[0] - fastapi_annotations = [ - arg - for arg in annotated_args[1:] - if isinstance(arg, (FieldInfo, params.Depends)) - ] - fastapi_specific_annotations = [ - arg - for arg in fastapi_annotations - if isinstance( - arg, - ( - params.Param, - params.Body, - params.Depends, - ), - ) - ] - if fastapi_specific_annotations: - fastapi_annotation: Union[FieldInfo, params.Depends, None] = ( - fastapi_specific_annotations[-1] - ) - else: - fastapi_annotation = None - # Set default for Annotated FieldInfo - if isinstance(fastapi_annotation, FieldInfo): - # Copy `field_info` because we mutate `field_info.default` below. - field_info = copy_field_info( - field_info=fastapi_annotation, - annotation=use_annotation, - ) - assert ( - field_info.default == Undefined or field_info.default == RequiredParam - ), ( - f"`{field_info.__class__.__name__}` default value cannot be set in" - f" `Annotated` for {param_name!r}. Set the default value with `=` instead." - ) - if value is not inspect.Signature.empty: - assert not is_path_param, "Path parameters cannot have default values" - field_info.default = value - else: - field_info.default = RequiredParam - # Get Annotated Depends - elif isinstance(fastapi_annotation, params.Depends): - depends = fastapi_annotation - # Get Depends from default value - if isinstance(value, params.Depends): - assert depends is None, ( - "Cannot specify `Depends` in `Annotated` and default value" - f" together for {param_name!r}" - ) - assert field_info is None, ( - "Cannot specify a FastAPI annotation in `Annotated` and `Depends` as a" - f" default value together for {param_name!r}" - ) - depends = value - # Get FieldInfo from default value - elif isinstance(value, FieldInfo): - assert field_info is None, ( - "Cannot specify FastAPI annotations in `Annotated` and default value" - f" together for {param_name!r}" - ) - field_info = value - if isinstance(field_info, FieldInfo): - field_info.annotation = type_annotation - - # Get Depends from type annotation - if depends is not None and depends.dependency is None: - # Copy `depends` before mutating it - depends = copy(depends) - depends = dataclasses.replace(depends, dependency=type_annotation) - - # Handle non-param type annotations like Request - # Only apply special handling when there's no explicit Depends - if there's a Depends, - # the dependency will be called and its return value used instead of the special injection - if depends is None and lenient_issubclass( - type_annotation, - ( - Request, - WebSocket, - HTTPConnection, - Response, - StarletteBackgroundTasks, - SecurityScopes, - ), - ): - assert field_info is None, ( - f"Cannot specify FastAPI annotation for type {type_annotation!r}" - ) - # Handle default assignations, neither field_info nor depends was not found in Annotated nor default value - elif field_info is None and depends is None: - default_value = value if value is not inspect.Signature.empty else RequiredParam - if is_path_param: - # We might check here that `default_value is RequiredParam`, but the fact is that the same - # parameter might sometimes be a path parameter and sometimes not. See - # `tests/test_infer_param_optionality.py` for an example. - field_info = params.Path(annotation=use_annotation) - elif is_uploadfile_or_nonable_uploadfile_annotation( - type_annotation - ) or is_uploadfile_sequence_annotation(type_annotation): - field_info = params.File(annotation=use_annotation, default=default_value) - elif not field_annotation_is_scalar(annotation=type_annotation): - field_info = params.Body(annotation=use_annotation, default=default_value) - else: - field_info = params.Query(annotation=use_annotation, default=default_value) - - field = None - # It's a field_info, not a dependency - if field_info is not None: - # Handle field_info.in_ - if is_path_param: - assert isinstance(field_info, params.Path), ( - f"Cannot use `{field_info.__class__.__name__}` for path param" - f" {param_name!r}" - ) - elif ( - isinstance(field_info, params.Param) - and getattr(field_info, "in_", None) is None - ): - field_info.in_ = params.ParamTypes.query - use_annotation_from_field_info = use_annotation - if isinstance(field_info, params.Form): - ensure_multipart_is_installed() - if not field_info.alias and getattr(field_info, "convert_underscores", None): - alias = param_name.replace("_", "-") - else: - alias = field_info.alias or param_name - field_info.alias = alias - field = create_model_field( - name=param_name, - type_=use_annotation_from_field_info, - default=field_info.default, - alias=alias, - required=field_info.default in (RequiredParam, Undefined), - field_info=field_info, - ) - if is_path_param: - assert is_scalar_field(field=field), ( - "Path params must be of one of the supported types" - ) - elif isinstance(field_info, params.Query): - assert ( - is_scalar_field(field) - or is_scalar_sequence_field(field) - or ( - lenient_issubclass(field.type_, BaseModel) - # For Pydantic v1 - and getattr(field, "shape", 1) == 1 - ) - ), f"Query parameter {param_name!r} must be one of the supported types" - - return ParamDetails(type_annotation=type_annotation, depends=depends, field=field) - - -def add_param_to_fields(*, field: ModelField, dependant: Dependant) -> None: - field_info = field.field_info - field_info_in = getattr(field_info, "in_", None) - if field_info_in == params.ParamTypes.path: - dependant.path_params.append(field) - elif field_info_in == params.ParamTypes.query: - dependant.query_params.append(field) - elif field_info_in == params.ParamTypes.header: - dependant.header_params.append(field) - else: - assert field_info_in == params.ParamTypes.cookie, ( - f"non-body parameters must be in path, query, header or cookie: {field.name}" - ) - dependant.cookie_params.append(field) - - -async def _solve_generator( - *, dependant: Dependant, stack: AsyncExitStack, sub_values: dict[str, Any] -) -> Any: - assert dependant.call - if dependant.is_async_gen_callable: - cm = asynccontextmanager(dependant.call)(**sub_values) - elif dependant.is_gen_callable: - cm = contextmanager_in_threadpool(contextmanager(dependant.call)(**sub_values)) - return await stack.enter_async_context(cm) - - -@dataclass -class SolvedDependency: - values: dict[str, Any] - errors: list[Any] - background_tasks: Optional[StarletteBackgroundTasks] - response: Response - dependency_cache: dict[DependencyCacheKey, Any] - - -async def solve_dependencies( - *, - request: Union[Request, WebSocket], - dependant: Dependant, - body: Optional[Union[dict[str, Any], FormData]] = None, - background_tasks: Optional[StarletteBackgroundTasks] = None, - response: Optional[Response] = None, - dependency_overrides_provider: Optional[Any] = None, - dependency_cache: Optional[dict[DependencyCacheKey, Any]] = None, - # TODO: remove this parameter later, no longer used, not removing it yet as some - # people might be monkey patching this function (although that's not supported) - async_exit_stack: AsyncExitStack, - embed_body_fields: bool, -) -> SolvedDependency: - request_astack = request.scope.get("fastapi_inner_astack") - assert isinstance(request_astack, AsyncExitStack), ( - "fastapi_inner_astack not found in request scope" - ) - function_astack = request.scope.get("fastapi_function_astack") - assert isinstance(function_astack, AsyncExitStack), ( - "fastapi_function_astack not found in request scope" - ) - values: dict[str, Any] = {} - errors: list[Any] = [] - if response is None: - response = Response() - del response.headers["content-length"] - response.status_code = None # type: ignore - if dependency_cache is None: - dependency_cache = {} - for sub_dependant in dependant.dependencies: - sub_dependant.call = cast(Callable[..., Any], sub_dependant.call) - call = sub_dependant.call - use_sub_dependant = sub_dependant - if ( - dependency_overrides_provider - and dependency_overrides_provider.dependency_overrides - ): - original_call = sub_dependant.call - call = getattr( - dependency_overrides_provider, "dependency_overrides", {} - ).get(original_call, original_call) - use_path: str = sub_dependant.path # type: ignore - use_sub_dependant = get_dependant( - path=use_path, - call=call, - name=sub_dependant.name, - parent_oauth_scopes=sub_dependant.oauth_scopes, - scope=sub_dependant.scope, - ) - - solved_result = await solve_dependencies( - request=request, - dependant=use_sub_dependant, - body=body, - background_tasks=background_tasks, - response=response, - dependency_overrides_provider=dependency_overrides_provider, - dependency_cache=dependency_cache, - async_exit_stack=async_exit_stack, - embed_body_fields=embed_body_fields, - ) - background_tasks = solved_result.background_tasks - if solved_result.errors: - errors.extend(solved_result.errors) - continue - if sub_dependant.use_cache and sub_dependant.cache_key in dependency_cache: - solved = dependency_cache[sub_dependant.cache_key] - elif ( - use_sub_dependant.is_gen_callable or use_sub_dependant.is_async_gen_callable - ): - use_astack = request_astack - if sub_dependant.scope == "function": - use_astack = function_astack - solved = await _solve_generator( - dependant=use_sub_dependant, - stack=use_astack, - sub_values=solved_result.values, - ) - elif use_sub_dependant.is_coroutine_callable: - solved = await call(**solved_result.values) - else: - solved = await run_in_threadpool(call, **solved_result.values) - if sub_dependant.name is not None: - values[sub_dependant.name] = solved - if sub_dependant.cache_key not in dependency_cache: - dependency_cache[sub_dependant.cache_key] = solved - path_values, path_errors = request_params_to_args( - dependant.path_params, request.path_params - ) - query_values, query_errors = request_params_to_args( - dependant.query_params, request.query_params - ) - header_values, header_errors = request_params_to_args( - dependant.header_params, request.headers - ) - cookie_values, cookie_errors = request_params_to_args( - dependant.cookie_params, request.cookies - ) - values.update(path_values) - values.update(query_values) - values.update(header_values) - values.update(cookie_values) - errors += path_errors + query_errors + header_errors + cookie_errors - if dependant.body_params: - ( - body_values, - body_errors, - ) = await request_body_to_args( # body_params checked above - body_fields=dependant.body_params, - received_body=body, - embed_body_fields=embed_body_fields, - ) - values.update(body_values) - errors.extend(body_errors) - if dependant.http_connection_param_name: - values[dependant.http_connection_param_name] = request - if dependant.request_param_name and isinstance(request, Request): - values[dependant.request_param_name] = request - elif dependant.websocket_param_name and isinstance(request, WebSocket): - values[dependant.websocket_param_name] = request - if dependant.background_tasks_param_name: - if background_tasks is None: - background_tasks = BackgroundTasks() - values[dependant.background_tasks_param_name] = background_tasks - if dependant.response_param_name: - values[dependant.response_param_name] = response - if dependant.security_scopes_param_name: - values[dependant.security_scopes_param_name] = SecurityScopes( - scopes=dependant.oauth_scopes - ) - return SolvedDependency( - values=values, - errors=errors, - background_tasks=background_tasks, - response=response, - dependency_cache=dependency_cache, - ) - - -def _validate_value_with_model_field( - *, field: ModelField, value: Any, values: dict[str, Any], loc: tuple[str, ...] -) -> tuple[Any, list[Any]]: - if value is None: - if field.required: - return None, [get_missing_field_error(loc=loc)] - else: - return deepcopy(field.default), [] - v_, errors_ = field.validate(value, values, loc=loc) - if isinstance(errors_, list): - new_errors = _regenerate_error_with_loc(errors=errors_, loc_prefix=()) - return None, new_errors - else: - return v_, [] - - -def _is_json_field(field: ModelField) -> bool: - return any(type(item) is Json for item in field.field_info.metadata) - - -def _get_multidict_value( - field: ModelField, values: Mapping[str, Any], alias: Union[str, None] = None -) -> Any: - alias = alias or get_validation_alias(field) - if ( - (not _is_json_field(field)) - and is_sequence_field(field) - and isinstance(values, (ImmutableMultiDict, Headers)) - ): - value = values.getlist(alias) - else: - value = values.get(alias, None) - if ( - value is None - or ( - isinstance(field.field_info, params.Form) - and isinstance(value, str) # For type checks - and value == "" - ) - or (is_sequence_field(field) and len(value) == 0) - ): - if field.required: - return - else: - return deepcopy(field.default) - return value - - -def request_params_to_args( - fields: Sequence[ModelField], - received_params: Union[Mapping[str, Any], QueryParams, Headers], -) -> tuple[dict[str, Any], list[Any]]: - values: dict[str, Any] = {} - errors: list[dict[str, Any]] = [] - - if not fields: - return values, errors - - first_field = fields[0] - fields_to_extract = fields - single_not_embedded_field = False - default_convert_underscores = True - if len(fields) == 1 and lenient_issubclass(first_field.type_, BaseModel): - fields_to_extract = get_cached_model_fields(first_field.type_) - single_not_embedded_field = True - # If headers are in a Pydantic model, the way to disable convert_underscores - # would be with Header(convert_underscores=False) at the Pydantic model level - default_convert_underscores = getattr( - first_field.field_info, "convert_underscores", True - ) - - params_to_process: dict[str, Any] = {} - - processed_keys = set() - - for field in fields_to_extract: - alias = None - if isinstance(received_params, Headers): - # Handle fields extracted from a Pydantic Model for a header, each field - # doesn't have a FieldInfo of type Header with the default convert_underscores=True - convert_underscores = getattr( - field.field_info, "convert_underscores", default_convert_underscores - ) - if convert_underscores: - alias = get_validation_alias(field) - if alias == field.name: - alias = alias.replace("_", "-") - value = _get_multidict_value(field, received_params, alias=alias) - if value is not None: - params_to_process[get_validation_alias(field)] = value - processed_keys.add(alias or get_validation_alias(field)) - - for key in received_params.keys(): - if key not in processed_keys: - if hasattr(received_params, "getlist"): - value = received_params.getlist(key) - if isinstance(value, list) and (len(value) == 1): - params_to_process[key] = value[0] - else: - params_to_process[key] = value - else: - params_to_process[key] = received_params.get(key) - - if single_not_embedded_field: - field_info = first_field.field_info - assert isinstance(field_info, params.Param), ( - "Params must be subclasses of Param" - ) - loc: tuple[str, ...] = (field_info.in_.value,) - v_, errors_ = _validate_value_with_model_field( - field=first_field, value=params_to_process, values=values, loc=loc - ) - return {first_field.name: v_}, errors_ - - for field in fields: - value = _get_multidict_value(field, received_params) - field_info = field.field_info - assert isinstance(field_info, params.Param), ( - "Params must be subclasses of Param" - ) - loc = (field_info.in_.value, get_validation_alias(field)) - v_, errors_ = _validate_value_with_model_field( - field=field, value=value, values=values, loc=loc - ) - if errors_: - errors.extend(errors_) - else: - values[field.name] = v_ - return values, errors - - -def is_union_of_base_models(field_type: Any) -> bool: - """Check if field type is a Union where all members are BaseModel subclasses.""" - from fastapi.types import UnionType - - origin = get_origin(field_type) - - # Check if it's a Union type (covers both typing.Union and types.UnionType in Python 3.10+) - if origin is not Union and origin is not UnionType: - return False - - union_args = get_args(field_type) - - for arg in union_args: - if not lenient_issubclass(arg, BaseModel): - return False - - return True - - -def _should_embed_body_fields(fields: list[ModelField]) -> bool: - if not fields: - return False - # More than one dependency could have the same field, it would show up as multiple - # fields but it's the same one, so count them by name - body_param_names_set = {field.name for field in fields} - # A top level field has to be a single field, not multiple - if len(body_param_names_set) > 1: - return True - first_field = fields[0] - # If it explicitly specifies it is embedded, it has to be embedded - if getattr(first_field.field_info, "embed", None): - return True - # If it's a Form (or File) field, it has to be a BaseModel (or a union of BaseModels) to be top level - # otherwise it has to be embedded, so that the key value pair can be extracted - if ( - isinstance(first_field.field_info, params.Form) - and not lenient_issubclass(first_field.type_, BaseModel) - and not is_union_of_base_models(first_field.type_) - ): - return True - return False - - -async def _extract_form_body( - body_fields: list[ModelField], - received_body: FormData, -) -> dict[str, Any]: - values = {} - - for field in body_fields: - value = _get_multidict_value(field, received_body) - field_info = field.field_info - if ( - isinstance(field_info, params.File) - and is_bytes_field(field) - and isinstance(value, UploadFile) - ): - value = await value.read() - elif ( - is_bytes_sequence_field(field) - and isinstance(field_info, params.File) - and value_is_sequence(value) - ): - # For types - assert isinstance(value, sequence_types) - results: list[Union[bytes, str]] = [] - - async def process_fn( - fn: Callable[[], Coroutine[Any, Any, Any]], - ) -> None: - result = await fn() - results.append(result) # noqa: B023 - - async with anyio.create_task_group() as tg: - for sub_value in value: - tg.start_soon(process_fn, sub_value.read) - value = serialize_sequence_value(field=field, value=results) - if value is not None: - values[get_validation_alias(field)] = value - field_aliases = {get_validation_alias(field) for field in body_fields} - for key in received_body.keys(): - if key not in field_aliases: - param_values = received_body.getlist(key) - if len(param_values) == 1: - values[key] = param_values[0] - else: - values[key] = param_values - return values - - -async def request_body_to_args( - body_fields: list[ModelField], - received_body: Optional[Union[dict[str, Any], FormData]], - embed_body_fields: bool, -) -> tuple[dict[str, Any], list[dict[str, Any]]]: - values: dict[str, Any] = {} - errors: list[dict[str, Any]] = [] - assert body_fields, "request_body_to_args() should be called with fields" - single_not_embedded_field = len(body_fields) == 1 and not embed_body_fields - first_field = body_fields[0] - body_to_process = received_body - - fields_to_extract: list[ModelField] = body_fields - - if ( - single_not_embedded_field - and lenient_issubclass(first_field.type_, BaseModel) - and isinstance(received_body, FormData) - ): - fields_to_extract = get_cached_model_fields(first_field.type_) - - if isinstance(received_body, FormData): - body_to_process = await _extract_form_body(fields_to_extract, received_body) - - if single_not_embedded_field: - loc: tuple[str, ...] = ("body",) - v_, errors_ = _validate_value_with_model_field( - field=first_field, value=body_to_process, values=values, loc=loc - ) - return {first_field.name: v_}, errors_ - for field in body_fields: - loc = ("body", get_validation_alias(field)) - value: Optional[Any] = None - if body_to_process is not None: - try: - value = body_to_process.get(get_validation_alias(field)) - # If the received body is a list, not a dict - except AttributeError: - errors.append(get_missing_field_error(loc)) - continue - v_, errors_ = _validate_value_with_model_field( - field=field, value=value, values=values, loc=loc - ) - if errors_: - errors.extend(errors_) - else: - values[field.name] = v_ - return values, errors - - -def get_body_field( - *, flat_dependant: Dependant, name: str, embed_body_fields: bool -) -> Optional[ModelField]: - """ - Get a ModelField representing the request body for a path operation, combining - all body parameters into a single field if necessary. - - Used to check if it's form data (with `isinstance(body_field, params.Form)`) - or JSON and to generate the JSON Schema for a request body. - - This is **not** used to validate/parse the request body, that's done with each - individual body parameter. - """ - if not flat_dependant.body_params: - return None - first_param = flat_dependant.body_params[0] - if not embed_body_fields: - return first_param - model_name = "Body_" + name - BodyModel = create_body_model( - fields=flat_dependant.body_params, model_name=model_name - ) - required = any(True for f in flat_dependant.body_params if f.required) - BodyFieldInfo_kwargs: dict[str, Any] = { - "annotation": BodyModel, - "alias": "body", - } - if not required: - BodyFieldInfo_kwargs["default"] = None - if any(isinstance(f.field_info, params.File) for f in flat_dependant.body_params): - BodyFieldInfo: type[params.Body] = params.File - elif any(isinstance(f.field_info, params.Form) for f in flat_dependant.body_params): - BodyFieldInfo = params.Form - else: - BodyFieldInfo = params.Body - - body_param_media_types = [ - f.field_info.media_type - for f in flat_dependant.body_params - if isinstance(f.field_info, params.Body) - ] - if len(set(body_param_media_types)) == 1: - BodyFieldInfo_kwargs["media_type"] = body_param_media_types[0] - final_field = create_model_field( - name="body", - type_=BodyModel, - required=required, - alias="body", - field_info=BodyFieldInfo(**BodyFieldInfo_kwargs), - ) - return final_field - - -def get_validation_alias(field: ModelField) -> str: - va = getattr(field, "validation_alias", None) - return va or field.alias diff --git a/.env/Lib/site-packages/fastapi/encoders.py b/.env/Lib/site-packages/fastapi/encoders.py deleted file mode 100644 index b4661be..0000000 --- a/.env/Lib/site-packages/fastapi/encoders.py +++ /dev/null @@ -1,346 +0,0 @@ -import dataclasses -import datetime -from collections import defaultdict, deque -from decimal import Decimal -from enum import Enum -from ipaddress import ( - IPv4Address, - IPv4Interface, - IPv4Network, - IPv6Address, - IPv6Interface, - IPv6Network, -) -from pathlib import Path, PurePath -from re import Pattern -from types import GeneratorType -from typing import Annotated, Any, Callable, Optional, Union -from uuid import UUID - -from annotated_doc import Doc -from fastapi.exceptions import PydanticV1NotSupportedError -from fastapi.types import IncEx -from pydantic import BaseModel -from pydantic.color import Color -from pydantic.networks import AnyUrl, NameEmail -from pydantic.types import SecretBytes, SecretStr -from pydantic_core import PydanticUndefinedType - -from ._compat import ( - Url, - is_pydantic_v1_model_instance, -) - - -# Taken from Pydantic v1 as is -def isoformat(o: Union[datetime.date, datetime.time]) -> str: - return o.isoformat() - - -# Adapted from Pydantic v1 -# TODO: pv2 should this return strings instead? -def decimal_encoder(dec_value: Decimal) -> Union[int, float]: - """ - Encodes a Decimal as int if there's no exponent, otherwise float - - This is useful when we use ConstrainedDecimal to represent Numeric(x,0) - where an integer (but not int typed) is used. Encoding this as a float - results in failed round-tripping between encode and parse. - Our Id type is a prime example of this. - - >>> decimal_encoder(Decimal("1.0")) - 1.0 - - >>> decimal_encoder(Decimal("1")) - 1 - - >>> decimal_encoder(Decimal("NaN")) - nan - """ - exponent = dec_value.as_tuple().exponent - if isinstance(exponent, int) and exponent >= 0: - return int(dec_value) - else: - return float(dec_value) - - -ENCODERS_BY_TYPE: dict[type[Any], Callable[[Any], Any]] = { - bytes: lambda o: o.decode(), - Color: str, - datetime.date: isoformat, - datetime.datetime: isoformat, - datetime.time: isoformat, - datetime.timedelta: lambda td: td.total_seconds(), - Decimal: decimal_encoder, - Enum: lambda o: o.value, - frozenset: list, - deque: list, - GeneratorType: list, - IPv4Address: str, - IPv4Interface: str, - IPv4Network: str, - IPv6Address: str, - IPv6Interface: str, - IPv6Network: str, - NameEmail: str, - Path: str, - Pattern: lambda o: o.pattern, - SecretBytes: str, - SecretStr: str, - set: list, - UUID: str, - Url: str, - AnyUrl: str, -} - - -def generate_encoders_by_class_tuples( - type_encoder_map: dict[Any, Callable[[Any], Any]], -) -> dict[Callable[[Any], Any], tuple[Any, ...]]: - encoders_by_class_tuples: dict[Callable[[Any], Any], tuple[Any, ...]] = defaultdict( - tuple - ) - for type_, encoder in type_encoder_map.items(): - encoders_by_class_tuples[encoder] += (type_,) - return encoders_by_class_tuples - - -encoders_by_class_tuples = generate_encoders_by_class_tuples(ENCODERS_BY_TYPE) - - -def jsonable_encoder( - obj: Annotated[ - Any, - Doc( - """ - The input object to convert to JSON. - """ - ), - ], - include: Annotated[ - Optional[IncEx], - Doc( - """ - Pydantic's `include` parameter, passed to Pydantic models to set the - fields to include. - """ - ), - ] = None, - exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Pydantic's `exclude` parameter, passed to Pydantic models to set the - fields to exclude. - """ - ), - ] = None, - by_alias: Annotated[ - bool, - Doc( - """ - Pydantic's `by_alias` parameter, passed to Pydantic models to define if - the output should use the alias names (when provided) or the Python - attribute names. In an API, if you set an alias, it's probably because you - want to use it in the result, so you probably want to leave this set to - `True`. - """ - ), - ] = True, - exclude_unset: Annotated[ - bool, - Doc( - """ - Pydantic's `exclude_unset` parameter, passed to Pydantic models to define - if it should exclude from the output the fields that were not explicitly - set (and that only had their default values). - """ - ), - ] = False, - exclude_defaults: Annotated[ - bool, - Doc( - """ - Pydantic's `exclude_defaults` parameter, passed to Pydantic models to define - if it should exclude from the output the fields that had the same default - value, even when they were explicitly set. - """ - ), - ] = False, - exclude_none: Annotated[ - bool, - Doc( - """ - Pydantic's `exclude_none` parameter, passed to Pydantic models to define - if it should exclude from the output any fields that have a `None` value. - """ - ), - ] = False, - custom_encoder: Annotated[ - Optional[dict[Any, Callable[[Any], Any]]], - Doc( - """ - Pydantic's `custom_encoder` parameter, passed to Pydantic models to define - a custom encoder. - """ - ), - ] = None, - sqlalchemy_safe: Annotated[ - bool, - Doc( - """ - Exclude from the output any fields that start with the name `_sa`. - - This is mainly a hack for compatibility with SQLAlchemy objects, they - store internal SQLAlchemy-specific state in attributes named with `_sa`, - and those objects can't (and shouldn't be) serialized to JSON. - """ - ), - ] = True, -) -> Any: - """ - Convert any object to something that can be encoded in JSON. - - This is used internally by FastAPI to make sure anything you return can be - encoded as JSON before it is sent to the client. - - You can also use it yourself, for example to convert objects before saving them - in a database that supports only JSON. - - Read more about it in the - [FastAPI docs for JSON Compatible Encoder](https://fastapi.tiangolo.com/tutorial/encoder/). - """ - custom_encoder = custom_encoder or {} - if custom_encoder: - if type(obj) in custom_encoder: - return custom_encoder[type(obj)](obj) - else: - for encoder_type, encoder_instance in custom_encoder.items(): - if isinstance(obj, encoder_type): - return encoder_instance(obj) - if include is not None and not isinstance(include, (set, dict)): - include = set(include) # type: ignore[assignment] - if exclude is not None and not isinstance(exclude, (set, dict)): - exclude = set(exclude) # type: ignore[assignment] - if isinstance(obj, BaseModel): - obj_dict = obj.model_dump( - mode="json", - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_none=exclude_none, - exclude_defaults=exclude_defaults, - ) - return jsonable_encoder( - obj_dict, - exclude_none=exclude_none, - exclude_defaults=exclude_defaults, - sqlalchemy_safe=sqlalchemy_safe, - ) - if dataclasses.is_dataclass(obj): - assert not isinstance(obj, type) - obj_dict = dataclasses.asdict(obj) - return jsonable_encoder( - obj_dict, - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_defaults=exclude_defaults, - exclude_none=exclude_none, - custom_encoder=custom_encoder, - sqlalchemy_safe=sqlalchemy_safe, - ) - if isinstance(obj, Enum): - return obj.value - if isinstance(obj, PurePath): - return str(obj) - if isinstance(obj, (str, int, float, type(None))): - return obj - if isinstance(obj, PydanticUndefinedType): - return None - if isinstance(obj, dict): - encoded_dict = {} - allowed_keys = set(obj.keys()) - if include is not None: - allowed_keys &= set(include) - if exclude is not None: - allowed_keys -= set(exclude) - for key, value in obj.items(): - if ( - ( - not sqlalchemy_safe - or (not isinstance(key, str)) - or (not key.startswith("_sa")) - ) - and (value is not None or not exclude_none) - and key in allowed_keys - ): - encoded_key = jsonable_encoder( - key, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_none=exclude_none, - custom_encoder=custom_encoder, - sqlalchemy_safe=sqlalchemy_safe, - ) - encoded_value = jsonable_encoder( - value, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_none=exclude_none, - custom_encoder=custom_encoder, - sqlalchemy_safe=sqlalchemy_safe, - ) - encoded_dict[encoded_key] = encoded_value - return encoded_dict - if isinstance(obj, (list, set, frozenset, GeneratorType, tuple, deque)): - encoded_list = [] - for item in obj: - encoded_list.append( - jsonable_encoder( - item, - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_defaults=exclude_defaults, - exclude_none=exclude_none, - custom_encoder=custom_encoder, - sqlalchemy_safe=sqlalchemy_safe, - ) - ) - return encoded_list - - if type(obj) in ENCODERS_BY_TYPE: - return ENCODERS_BY_TYPE[type(obj)](obj) - for encoder, classes_tuple in encoders_by_class_tuples.items(): - if isinstance(obj, classes_tuple): - return encoder(obj) - if is_pydantic_v1_model_instance(obj): - raise PydanticV1NotSupportedError( - "pydantic.v1 models are no longer supported by FastAPI." - f" Please update the model {obj!r}." - ) - try: - data = dict(obj) - except Exception as e: - errors: list[Exception] = [] - errors.append(e) - try: - data = vars(obj) - except Exception as e: - errors.append(e) - raise ValueError(errors) from e - return jsonable_encoder( - data, - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_defaults=exclude_defaults, - exclude_none=exclude_none, - custom_encoder=custom_encoder, - sqlalchemy_safe=sqlalchemy_safe, - ) diff --git a/.env/Lib/site-packages/fastapi/exception_handlers.py b/.env/Lib/site-packages/fastapi/exception_handlers.py deleted file mode 100644 index 475dd7b..0000000 --- a/.env/Lib/site-packages/fastapi/exception_handlers.py +++ /dev/null @@ -1,34 +0,0 @@ -from fastapi.encoders import jsonable_encoder -from fastapi.exceptions import RequestValidationError, WebSocketRequestValidationError -from fastapi.utils import is_body_allowed_for_status_code -from fastapi.websockets import WebSocket -from starlette.exceptions import HTTPException -from starlette.requests import Request -from starlette.responses import JSONResponse, Response -from starlette.status import WS_1008_POLICY_VIOLATION - - -async def http_exception_handler(request: Request, exc: HTTPException) -> Response: - headers = getattr(exc, "headers", None) - if not is_body_allowed_for_status_code(exc.status_code): - return Response(status_code=exc.status_code, headers=headers) - return JSONResponse( - {"detail": exc.detail}, status_code=exc.status_code, headers=headers - ) - - -async def request_validation_exception_handler( - request: Request, exc: RequestValidationError -) -> JSONResponse: - return JSONResponse( - status_code=422, - content={"detail": jsonable_encoder(exc.errors())}, - ) - - -async def websocket_request_validation_exception_handler( - websocket: WebSocket, exc: WebSocketRequestValidationError -) -> None: - await websocket.close( - code=WS_1008_POLICY_VIOLATION, reason=jsonable_encoder(exc.errors()) - ) diff --git a/.env/Lib/site-packages/fastapi/exceptions.py b/.env/Lib/site-packages/fastapi/exceptions.py deleted file mode 100644 index 62b4674..0000000 --- a/.env/Lib/site-packages/fastapi/exceptions.py +++ /dev/null @@ -1,256 +0,0 @@ -from collections.abc import Sequence -from typing import Annotated, Any, Optional, TypedDict, Union - -from annotated_doc import Doc -from pydantic import BaseModel, create_model -from starlette.exceptions import HTTPException as StarletteHTTPException -from starlette.exceptions import WebSocketException as StarletteWebSocketException - - -class EndpointContext(TypedDict, total=False): - function: str - path: str - file: str - line: int - - -class HTTPException(StarletteHTTPException): - """ - An HTTP exception you can raise in your own code to show errors to the client. - - This is for client errors, invalid authentication, invalid data, etc. Not for server - errors in your code. - - Read more about it in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/). - - ## Example - - ```python - from fastapi import FastAPI, HTTPException - - app = FastAPI() - - items = {"foo": "The Foo Wrestlers"} - - - @app.get("/items/{item_id}") - async def read_item(item_id: str): - if item_id not in items: - raise HTTPException(status_code=404, detail="Item not found") - return {"item": items[item_id]} - ``` - """ - - def __init__( - self, - status_code: Annotated[ - int, - Doc( - """ - HTTP status code to send to the client. - - Read more about it in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/#use-httpexception) - """ - ), - ], - detail: Annotated[ - Any, - Doc( - """ - Any data to be sent to the client in the `detail` key of the JSON - response. - - Read more about it in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/#use-httpexception) - """ - ), - ] = None, - headers: Annotated[ - Optional[dict[str, str]], - Doc( - """ - Any headers to send to the client in the response. - - Read more about it in the - [FastAPI docs for Handling Errors](https://fastapi.tiangolo.com/tutorial/handling-errors/#add-custom-headers) - - """ - ), - ] = None, - ) -> None: - super().__init__(status_code=status_code, detail=detail, headers=headers) - - -class WebSocketException(StarletteWebSocketException): - """ - A WebSocket exception you can raise in your own code to show errors to the client. - - This is for client errors, invalid authentication, invalid data, etc. Not for server - errors in your code. - - Read more about it in the - [FastAPI docs for WebSockets](https://fastapi.tiangolo.com/advanced/websockets/). - - ## Example - - ```python - from typing import Annotated - - from fastapi import ( - Cookie, - FastAPI, - WebSocket, - WebSocketException, - status, - ) - - app = FastAPI() - - @app.websocket("/items/{item_id}/ws") - async def websocket_endpoint( - *, - websocket: WebSocket, - session: Annotated[str | None, Cookie()] = None, - item_id: str, - ): - if session is None: - raise WebSocketException(code=status.WS_1008_POLICY_VIOLATION) - await websocket.accept() - while True: - data = await websocket.receive_text() - await websocket.send_text(f"Session cookie is: {session}") - await websocket.send_text(f"Message text was: {data}, for item ID: {item_id}") - ``` - """ - - def __init__( - self, - code: Annotated[ - int, - Doc( - """ - A closing code from the - [valid codes defined in the specification](https://datatracker.ietf.org/doc/html/rfc6455#section-7.4.1). - """ - ), - ], - reason: Annotated[ - Union[str, None], - Doc( - """ - The reason to close the WebSocket connection. - - It is UTF-8-encoded data. The interpretation of the reason is up to the - application, it is not specified by the WebSocket specification. - - It could contain text that could be human-readable or interpretable - by the client code, etc. - """ - ), - ] = None, - ) -> None: - super().__init__(code=code, reason=reason) - - -RequestErrorModel: type[BaseModel] = create_model("Request") -WebSocketErrorModel: type[BaseModel] = create_model("WebSocket") - - -class FastAPIError(RuntimeError): - """ - A generic, FastAPI-specific error. - """ - - -class DependencyScopeError(FastAPIError): - """ - A dependency declared that it depends on another dependency with an invalid - (narrower) scope. - """ - - -class ValidationException(Exception): - def __init__( - self, - errors: Sequence[Any], - *, - endpoint_ctx: Optional[EndpointContext] = None, - ) -> None: - self._errors = errors - self.endpoint_ctx = endpoint_ctx - - ctx = endpoint_ctx or {} - self.endpoint_function = ctx.get("function") - self.endpoint_path = ctx.get("path") - self.endpoint_file = ctx.get("file") - self.endpoint_line = ctx.get("line") - - def errors(self) -> Sequence[Any]: - return self._errors - - def _format_endpoint_context(self) -> str: - if not (self.endpoint_file and self.endpoint_line and self.endpoint_function): - if self.endpoint_path: - return f"\n Endpoint: {self.endpoint_path}" - return "" - - context = f'\n File "{self.endpoint_file}", line {self.endpoint_line}, in {self.endpoint_function}' - if self.endpoint_path: - context += f"\n {self.endpoint_path}" - return context - - def __str__(self) -> str: - message = f"{len(self._errors)} validation error{'s' if len(self._errors) != 1 else ''}:\n" - for err in self._errors: - message += f" {err}\n" - message += self._format_endpoint_context() - return message.rstrip() - - -class RequestValidationError(ValidationException): - def __init__( - self, - errors: Sequence[Any], - *, - body: Any = None, - endpoint_ctx: Optional[EndpointContext] = None, - ) -> None: - super().__init__(errors, endpoint_ctx=endpoint_ctx) - self.body = body - - -class WebSocketRequestValidationError(ValidationException): - def __init__( - self, - errors: Sequence[Any], - *, - endpoint_ctx: Optional[EndpointContext] = None, - ) -> None: - super().__init__(errors, endpoint_ctx=endpoint_ctx) - - -class ResponseValidationError(ValidationException): - def __init__( - self, - errors: Sequence[Any], - *, - body: Any = None, - endpoint_ctx: Optional[EndpointContext] = None, - ) -> None: - super().__init__(errors, endpoint_ctx=endpoint_ctx) - self.body = body - - -class PydanticV1NotSupportedError(FastAPIError): - """ - A pydantic.v1 model is used, which is no longer supported. - """ - - -class FastAPIDeprecationWarning(UserWarning): - """ - A custom deprecation warning as DeprecationWarning is ignored - Ref: https://sethmlarson.dev/deprecations-via-warnings-dont-work-for-python-libraries - """ diff --git a/.env/Lib/site-packages/fastapi/logger.py b/.env/Lib/site-packages/fastapi/logger.py deleted file mode 100644 index 5b2c4ad..0000000 --- a/.env/Lib/site-packages/fastapi/logger.py +++ /dev/null @@ -1,3 +0,0 @@ -import logging - -logger = logging.getLogger("fastapi") diff --git a/.env/Lib/site-packages/fastapi/middleware/__init__.py b/.env/Lib/site-packages/fastapi/middleware/__init__.py deleted file mode 100644 index 620296d..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/__init__.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.middleware import Middleware as Middleware diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 7f15f54443e282a24764966f9235f118a0917049..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 249 zcmX@j%ge<81Sbx(Wm*C0#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA!& zd^1y0a#G6^i&FhGnQn;`mn0VDq?VMV>gB>EikN|FS2BDCY55iBY!wq)oLW>I6O^Bl zui%@QoRONBnVlL_P#NQ!nOu|)lq<_jPA!hnOU)~b@ySezDb6fO)h$R&&Q44R3a2Fk z?I_69N0_f4AD@|*SrQ+wS5Wzj!zMRBr8Fniu80F@9U~AIgBTx}85tSxF<2I{0XYB` Cf=6`# diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/asyncexitstack.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/asyncexitstack.cpython-312.pyc deleted file mode 100644 index 0ab96295bf152d91924f3673e29a2ea7e3fe6da1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1484 zcmah}U1%It6h3$U_9vUhW)stjmJLa5M%r`(ky0tjP(zTKJTzM9vM@~N-sqUwAMZ?l zR<}@0D0%3E6>NOWqa_Hw>qGag;ESb7!q7s%2lZ`rg@)pT=gjP;_CY-CJ@?%2-h1Yp z@0`6q^!H~0<^H{!-gO1w7h3d0>l&R4Xlwxo&U~n_5@Td5zEV+3YDFt)6}_ZajFQ1X zfmz_{25@Z$y^)ks4c#<;C(~rn5i4j^-I+BnTnHU^$<(5hwJ`gRRj)^>%fc1jiiq?D zw^kRCwjiq9rF%U|ca3g50!oZSNyz}uawb%+2#u>;TQs=7fe~>-7(69XJT1~Op5~bi zt(0kKW?yvd`9S78&V8}!ND-+H29tAZ)ljU3cGanf$dDpjmeqMz?S+CfAjaHHnpGA$2$l^qz7@u zQnFh?%neJ|Yacq2!m*B{^9a=?U`SYaNY4&pB-5~t1^pD6xT9>TH+3DZYfX)a{ixea z$+<*RQ{s;PZ!EF#hSF4bh*6YvY}VMPn)r;H7qMQ0t}D-flR*Q{!L7G&jNr(O!4$-2 zFrPcIk8_|yz*ZFoS1}VhzouS=RrZa#%H$FJq^U~!cwFkZ#HKC>2p|^a5UF8Q$Q#uo zpcJt5g@Sy6P^^vX`2PRYCOJZ0gDa?hgM%CJC_lQJANwgkw!OS}>h$K!=i|GBV-N6` z8{ZpzackoC#8>A3NC36%X39K0qMB{tn(l*Pjho zr`c_09W}mC(faaLYefByS$X4o4XqaRA=v6;*09#f8D!@f;lqsV5yLW5tyh>;$hA(S z$Ud1u9G`;Y-A1*aHL6Xmi?2eKIG}%?ZAV$#uGILlPj=3>KVEkHWJgkeMQRXAc@!;a zkRmE2HJdz!>MqDZvWZqnZ5%6lMuLl|HekPa=?_)Y&At3f?Rb9eOg!&r`a3$}?VPGl zZO^m;r*CKD>E3ia-GBQp0u%q_dfrS&+3u_1d)MLyhz2kp>5DKFlVPJSf;i?R$T-|M w(m1da6y0I4PtzBYq~DaIxIzDvfD&q}jIqaX@>e+j5Y9izX4%xAfGWQE59(=RqyPW_ diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/cors.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/cors.cpython-312.pyc deleted file mode 100644 index 240d6ee26272054e41f71c713e97637d61efcd9c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 254 zcmX@j%ge<81Sbx(W%>i@#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA#j zoc)7>eKS*1a#G6^i&FhGnQloImn0VDq?VMV>gB>E^pf+7ii?73o0(jc50op*OinG1(M!!Mi}A@!iYd-4N!2Y# zOwLYB2MVVp0xc@Y)JIsN53)qBpz;@oO>TZlX-=wL5eLvJMj$Q*F+MOeGBVy{a4TX1 FasZBcNk0Gp diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/gzip.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/gzip.cpython-312.pyc deleted file mode 100644 index f6db4a0b600e49d2af1dc50734b850c562c87a62..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 254 zcmX@j%ge<81Sbx(W%>i@#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA#j z+@mrJd^1y0a#G6^i&FhGnQloImn0VDq?VMV>gB>E^wO&`3yPS5npQG=2I>73o0(jc50op*OinG1(M!!Mi}A@!iYd-4N!2Y# zOwLYB2MVVp0xc@Y)JIsN53)qBpz;@oO>TZlX-=wL5eLvJMj$Q*F+MOeGBVy{a4TX1 Fasa@rNr(Ud diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/httpsredirect.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/httpsredirect.cpython-312.pyc deleted file mode 100644 index b6a8b513aefad3b5d171be16293ea78587a6d2a1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 283 zcmX@j%ge<81Sbx(WflYJ#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA!^ zJwiePf`d|1GK*4^OMEj^QgTwu6N^&)G?{KG6_+Fy<)oIBr0V6uCG;{%N(zdLpc;yp zfyS(4_zW`bSF*EJOlWaxQE^OAeoDTAZ(?#rYF=h`YD__8jBjRgQ9e+vEHgQ^I7Tlu zuPnwVGbyGxvm{lwATc>RF&!wJmI$=5AX6Vnb}|BS VF(?o|Ff%eTeq&-{WGZ3Qp*#IQvEcUZYdO(Bo^hQmXxIG<-#TON+8NJfXa)Qfkv!k z_zW`XSE933OlWaxQE^OAeoDTAZ(?#rYF=h`YD__8jBjRgQ9e+vEHgQ^I7TluuPnwV zGbyGxvm{lwATc>RF&!wJmI$=3AX6VnHZlTnG06KL Rm>C%vzcDc}G8M4_xd1BxQo{fM diff --git a/.env/Lib/site-packages/fastapi/middleware/__pycache__/wsgi.cpython-312.pyc b/.env/Lib/site-packages/fastapi/middleware/__pycache__/wsgi.cpython-312.pyc deleted file mode 100644 index 28915eb755d1ce7cee8d149a50ad95a05f34ee67..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 256 zcmX@j%ge<81Sbx(Wo85E#~=<2FhLog1%Qm{3@HpLj5!Rsj8Tk?43$ip%r8OGnvA#j z!h_vCeKS*1a#G6^i&FhGnQloImn0VDq?VMV>gB>E^va9VGmDsknpQG=2I>73o0(jc50op*OinG1(M!!Mi}A@!iYd-4N!2Y# zOwLYB2MVVp0xc@Y)JIsN53)qBpz;@oO>TZlX-=wL5eLvJMj$Q*`Sb%bBO~KCCMHIv IA~qlw06ykPKL7v# diff --git a/.env/Lib/site-packages/fastapi/middleware/asyncexitstack.py b/.env/Lib/site-packages/fastapi/middleware/asyncexitstack.py deleted file mode 100644 index 4ce3f5a..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/asyncexitstack.py +++ /dev/null @@ -1,18 +0,0 @@ -from contextlib import AsyncExitStack - -from starlette.types import ASGIApp, Receive, Scope, Send - - -# Used mainly to close files after the request is done, dependencies are closed -# in their own AsyncExitStack -class AsyncExitStackMiddleware: - def __init__( - self, app: ASGIApp, context_name: str = "fastapi_middleware_astack" - ) -> None: - self.app = app - self.context_name = context_name - - async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None: - async with AsyncExitStack() as stack: - scope[self.context_name] = stack - await self.app(scope, receive, send) diff --git a/.env/Lib/site-packages/fastapi/middleware/cors.py b/.env/Lib/site-packages/fastapi/middleware/cors.py deleted file mode 100644 index 8dfaad0..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/cors.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.middleware.cors import CORSMiddleware as CORSMiddleware # noqa diff --git a/.env/Lib/site-packages/fastapi/middleware/gzip.py b/.env/Lib/site-packages/fastapi/middleware/gzip.py deleted file mode 100644 index bbeb2cc..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/gzip.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.middleware.gzip import GZipMiddleware as GZipMiddleware # noqa diff --git a/.env/Lib/site-packages/fastapi/middleware/httpsredirect.py b/.env/Lib/site-packages/fastapi/middleware/httpsredirect.py deleted file mode 100644 index b7a3d8e..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/httpsredirect.py +++ /dev/null @@ -1,3 +0,0 @@ -from starlette.middleware.httpsredirect import ( # noqa - HTTPSRedirectMiddleware as HTTPSRedirectMiddleware, -) diff --git a/.env/Lib/site-packages/fastapi/middleware/trustedhost.py b/.env/Lib/site-packages/fastapi/middleware/trustedhost.py deleted file mode 100644 index 08d7e03..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/trustedhost.py +++ /dev/null @@ -1,3 +0,0 @@ -from starlette.middleware.trustedhost import ( # noqa - TrustedHostMiddleware as TrustedHostMiddleware, -) diff --git a/.env/Lib/site-packages/fastapi/middleware/wsgi.py b/.env/Lib/site-packages/fastapi/middleware/wsgi.py deleted file mode 100644 index 69e4dca..0000000 --- a/.env/Lib/site-packages/fastapi/middleware/wsgi.py +++ /dev/null @@ -1,3 +0,0 @@ -from starlette.middleware.wsgi import ( - WSGIMiddleware as WSGIMiddleware, -) # pragma: no cover # noqa diff --git a/.env/Lib/site-packages/fastapi/openapi/__init__.py b/.env/Lib/site-packages/fastapi/openapi/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/fastapi/openapi/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/openapi/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 979aee2d4d14b0b2b37049f59ef576f26c9d2130..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 188 zcmX@j%ge<81Sbx(WrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxd742*l6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdiZ! zrEW31yN29i4hRjo#p2=W=yHoCz&|+T7HfcGh_lBnHWybP*AUk#QDyz){M>^4ywtps zV*TRejMUu3V*M%^tisi~`6;P6wSJn+w*-A%Lp=Olg5$$ILpo0(jc z50op*OinG1(M!!Mi}A@!iYd-4N!2Y#OwLYB2MVVp7MCOzWa{S^q~?L>4BO~K25!VY0f;V_o8n|xo>o)Qf@c?B2EO%_N diff --git a/.env/Lib/site-packages/fastapi/openapi/__pycache__/docs.cpython-312.pyc b/.env/Lib/site-packages/fastapi/openapi/__pycache__/docs.cpython-312.pyc deleted file mode 100644 index 5d4ce808b84c46146ce972f9d30ec3020e285fbf..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 12411 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 diff --git a/.env/Lib/site-packages/fastapi/openapi/__pycache__/models.cpython-312.pyc b/.env/Lib/site-packages/fastapi/openapi/__pycache__/models.cpython-312.pyc deleted file mode 100644 index 28f644101555dfb964e825f12440ea6c46ebdeb0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 22795 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 L9j@Q(aRB}w9A$CV diff --git a/.env/Lib/site-packages/fastapi/openapi/__pycache__/utils.cpython-312.pyc b/.env/Lib/site-packages/fastapi/openapi/__pycache__/utils.cpython-312.pyc deleted file mode 100644 index 0302ec41a829ea2e51feb36c0a1e658ce22c8ab9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 21273 zcmc(HYj9go!%$)ap8Ee8wnf-)$8w2jFQPTYSVUb1vD@j~6ts-Cbv>Lw3q$Z`E)+!K2mDHv5)B2QQ+K@6%8&jrf6G>Mm z&8fQSIuh0-Eh+1?HAPR;DciIyWuLaE9Mg`JbJ|JrwMkdXJ?$o8UDA{CPJ2mMpY)~b zr|VPxX@4p(9Y{4yH;{BgvN6>(-9*C1WH8k{-Aux!WJ{`bx-}J=4w1My*_H}Vhe^0D z*`Df{?np(ZBP4D~cBZ`wJe_mD81>`nDe_oa4B?@0Ae_mebRav(K0JxId# zQFPw$4Xi*Yj^#>@Dadd8nnfFDRGE-N9_a9hPRGJdA% zwvq{!zRgSv-#@GL0c`vLGdK!gW+?7uhT{iGp1zcKs5*wwQu?f# z+4(E#>BGzjv+K5k*$pN4#Cw>%0DUlSjqhegE3nlVq|(g3_z|e-MchVa|F3AL$4Xca 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 diff --git a/.env/Lib/site-packages/fastapi/openapi/constants.py b/.env/Lib/site-packages/fastapi/openapi/constants.py deleted file mode 100644 index d724ee3..0000000 --- a/.env/Lib/site-packages/fastapi/openapi/constants.py +++ /dev/null @@ -1,3 +0,0 @@ -METHODS_WITH_BODY = {"GET", "HEAD", "POST", "PUT", "DELETE", "PATCH"} -REF_PREFIX = "#/components/schemas/" -REF_TEMPLATE = "#/components/schemas/{model}" diff --git a/.env/Lib/site-packages/fastapi/openapi/docs.py b/.env/Lib/site-packages/fastapi/openapi/docs.py deleted file mode 100644 index bb387c6..0000000 --- a/.env/Lib/site-packages/fastapi/openapi/docs.py +++ /dev/null @@ -1,375 +0,0 @@ -import json -from typing import Annotated, Any, Optional - -from annotated_doc import Doc -from fastapi.encoders import jsonable_encoder -from starlette.responses import HTMLResponse - -swagger_ui_default_parameters: Annotated[ - dict[str, Any], - Doc( - """ - Default configurations for Swagger UI. - - You can use it as a template to add any other configurations needed. - """ - ), -] = { - "dom_id": "#swagger-ui", - "layout": "BaseLayout", - "deepLinking": True, - "showExtensions": True, - "showCommonExtensions": True, -} - - -def get_swagger_ui_html( - *, - openapi_url: Annotated[ - str, - Doc( - """ - The OpenAPI URL that Swagger UI should load and use. - - This is normally done automatically by FastAPI using the default URL - `/openapi.json`. - - Read more about it in the - [FastAPI docs for Conditional OpenAPI](https://fastapi.tiangolo.com/how-to/conditional-openapi/#conditional-openapi-from-settings-and-env-vars) - """ - ), - ], - title: Annotated[ - str, - Doc( - """ - The HTML `` content, normally shown in the browser tab. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ], - swagger_js_url: Annotated[ - str, - Doc( - """ - The URL to use to load the Swagger UI JavaScript. - - It is normally set to a CDN URL. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ] = "https://cdn.jsdelivr.net/npm/swagger-ui-dist@5/swagger-ui-bundle.js", - swagger_css_url: Annotated[ - str, - Doc( - """ - The URL to use to load the Swagger UI CSS. - - It is normally set to a CDN URL. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ] = "https://cdn.jsdelivr.net/npm/swagger-ui-dist@5/swagger-ui.css", - swagger_favicon_url: Annotated[ - str, - Doc( - """ - The URL of the favicon to use. It is normally shown in the browser tab. - """ - ), - ] = "https://fastapi.tiangolo.com/img/favicon.png", - oauth2_redirect_url: Annotated[ - Optional[str], - Doc( - """ - The OAuth2 redirect URL, it is normally automatically handled by FastAPI. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ] = None, - init_oauth: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - A dictionary with Swagger UI OAuth2 initialization configurations. - - Read more about the available configuration options in the - [Swagger UI docs](https://swagger.io/docs/open-source-tools/swagger-ui/usage/oauth2/). - """ - ), - ] = None, - swagger_ui_parameters: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Configuration parameters for Swagger UI. - - It defaults to [swagger_ui_default_parameters][fastapi.openapi.docs.swagger_ui_default_parameters]. - - Read more about it in the - [FastAPI docs about how to Configure Swagger UI](https://fastapi.tiangolo.com/how-to/configure-swagger-ui/). - """ - ), - ] = None, -) -> HTMLResponse: - """ - Generate and return the HTML that loads Swagger UI for the interactive - API docs (normally served at `/docs`). - - You would only call this function yourself if you needed to override some parts, - for example the URLs to use to load Swagger UI's JavaScript and CSS. - - Read more about it in the - [FastAPI docs for Configure Swagger UI](https://fastapi.tiangolo.com/how-to/configure-swagger-ui/) - and the [FastAPI docs for Custom Docs UI Static Assets (Self-Hosting)](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/). - """ - current_swagger_ui_parameters = swagger_ui_default_parameters.copy() - if swagger_ui_parameters: - current_swagger_ui_parameters.update(swagger_ui_parameters) - - html = f""" - <!DOCTYPE html> - <html> - <head> - <meta name="viewport" content="width=device-width, initial-scale=1.0"> - <link type="text/css" rel="stylesheet" href="{swagger_css_url}"> - <link rel="shortcut icon" href="{swagger_favicon_url}"> - <title>{title} - - -
-
- - - - - - """ - return HTMLResponse(html) - - -def get_redoc_html( - *, - openapi_url: Annotated[ - str, - Doc( - """ - The OpenAPI URL that ReDoc should load and use. - - This is normally done automatically by FastAPI using the default URL - `/openapi.json`. - - Read more about it in the - [FastAPI docs for Conditional OpenAPI](https://fastapi.tiangolo.com/how-to/conditional-openapi/#conditional-openapi-from-settings-and-env-vars) - """ - ), - ], - title: Annotated[ - str, - Doc( - """ - The HTML `` content, normally shown in the browser tab. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ], - redoc_js_url: Annotated[ - str, - Doc( - """ - The URL to use to load the ReDoc JavaScript. - - It is normally set to a CDN URL. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/) - """ - ), - ] = "https://cdn.jsdelivr.net/npm/redoc@2/bundles/redoc.standalone.js", - redoc_favicon_url: Annotated[ - str, - Doc( - """ - The URL of the favicon to use. It is normally shown in the browser tab. - """ - ), - ] = "https://fastapi.tiangolo.com/img/favicon.png", - with_google_fonts: Annotated[ - bool, - Doc( - """ - Load and use Google Fonts. - """ - ), - ] = True, -) -> HTMLResponse: - """ - Generate and return the HTML response that loads ReDoc for the alternative - API docs (normally served at `/redoc`). - - You would only call this function yourself if you needed to override some parts, - for example the URLs to use to load ReDoc's JavaScript and CSS. - - Read more about it in the - [FastAPI docs for Custom Docs UI Static Assets (Self-Hosting)](https://fastapi.tiangolo.com/how-to/custom-docs-ui-assets/). - """ - html = f""" - <!DOCTYPE html> - <html> - <head> - <title>{title} - - - - """ - if with_google_fonts: - html += """ - - """ - html += f""" - - - - - - - - - - - """ - return HTMLResponse(html) - - -def get_swagger_ui_oauth2_redirect_html() -> HTMLResponse: - """ - Generate the HTML response with the OAuth2 redirection for Swagger UI. - - You normally don't need to use or change this. - """ - # copied from https://github.com/swagger-api/swagger-ui/blob/v4.14.0/dist/oauth2-redirect.html - html = """ - - - - Swagger UI: OAuth2 Redirect - - - - - - """ - return HTMLResponse(content=html) diff --git a/.env/Lib/site-packages/fastapi/openapi/models.py b/.env/Lib/site-packages/fastapi/openapi/models.py deleted file mode 100644 index ac6a6d5..0000000 --- a/.env/Lib/site-packages/fastapi/openapi/models.py +++ /dev/null @@ -1,438 +0,0 @@ -from collections.abc import Iterable, Mapping -from enum import Enum -from typing import Annotated, Any, Callable, Optional, Union - -from fastapi._compat import with_info_plain_validator_function -from fastapi.logger import logger -from pydantic import ( - AnyUrl, - BaseModel, - Field, - GetJsonSchemaHandler, -) -from typing_extensions import Literal, TypedDict -from typing_extensions import deprecated as typing_deprecated - -try: - import email_validator - - assert email_validator # make autoflake ignore the unused import - from pydantic import EmailStr -except ImportError: # pragma: no cover - - class EmailStr(str): # type: ignore - @classmethod - def __get_validators__(cls) -> Iterable[Callable[..., Any]]: - yield cls.validate - - @classmethod - def validate(cls, v: Any) -> str: - logger.warning( - "email-validator not installed, email fields will be treated as str.\n" - "To install, run: pip install email-validator" - ) - return str(v) - - @classmethod - def _validate(cls, __input_value: Any, _: Any) -> str: - logger.warning( - "email-validator not installed, email fields will be treated as str.\n" - "To install, run: pip install email-validator" - ) - return str(__input_value) - - @classmethod - def __get_pydantic_json_schema__( - cls, core_schema: Mapping[str, Any], handler: GetJsonSchemaHandler - ) -> dict[str, Any]: - return {"type": "string", "format": "email"} - - @classmethod - def __get_pydantic_core_schema__( - cls, source: type[Any], handler: Callable[[Any], Mapping[str, Any]] - ) -> Mapping[str, Any]: - return with_info_plain_validator_function(cls._validate) - - -class BaseModelWithConfig(BaseModel): - model_config = {"extra": "allow"} - - -class Contact(BaseModelWithConfig): - name: Optional[str] = None - url: Optional[AnyUrl] = None - email: Optional[EmailStr] = None - - -class License(BaseModelWithConfig): - name: str - identifier: Optional[str] = None - url: Optional[AnyUrl] = None - - -class Info(BaseModelWithConfig): - title: str - summary: Optional[str] = None - description: Optional[str] = None - termsOfService: Optional[str] = None - contact: Optional[Contact] = None - license: Optional[License] = None - version: str - - -class ServerVariable(BaseModelWithConfig): - enum: Annotated[Optional[list[str]], Field(min_length=1)] = None - default: str - description: Optional[str] = None - - -class Server(BaseModelWithConfig): - url: Union[AnyUrl, str] - description: Optional[str] = None - variables: Optional[dict[str, ServerVariable]] = None - - -class Reference(BaseModel): - ref: str = Field(alias="$ref") - - -class Discriminator(BaseModel): - propertyName: str - mapping: Optional[dict[str, str]] = None - - -class XML(BaseModelWithConfig): - name: Optional[str] = None - namespace: Optional[str] = None - prefix: Optional[str] = None - attribute: Optional[bool] = None - wrapped: Optional[bool] = None - - -class ExternalDocumentation(BaseModelWithConfig): - description: Optional[str] = None - url: AnyUrl - - -# Ref JSON Schema 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation#name-type -SchemaType = Literal[ - "array", "boolean", "integer", "null", "number", "object", "string" -] - - -class Schema(BaseModelWithConfig): - # Ref: JSON Schema 2020-12: https://json-schema.org/draft/2020-12/json-schema-core.html#name-the-json-schema-core-vocabu - # Core Vocabulary - schema_: Optional[str] = Field(default=None, alias="$schema") - vocabulary: Optional[str] = Field(default=None, alias="$vocabulary") - id: Optional[str] = Field(default=None, alias="$id") - anchor: Optional[str] = Field(default=None, alias="$anchor") - dynamicAnchor: Optional[str] = Field(default=None, alias="$dynamicAnchor") - ref: Optional[str] = Field(default=None, alias="$ref") - dynamicRef: Optional[str] = Field(default=None, alias="$dynamicRef") - defs: Optional[dict[str, "SchemaOrBool"]] = Field(default=None, alias="$defs") - comment: Optional[str] = Field(default=None, alias="$comment") - # Ref: JSON Schema 2020-12: https://json-schema.org/draft/2020-12/json-schema-core.html#name-a-vocabulary-for-applying-s - # A Vocabulary for Applying Subschemas - allOf: Optional[list["SchemaOrBool"]] = None - anyOf: Optional[list["SchemaOrBool"]] = None - oneOf: Optional[list["SchemaOrBool"]] = None - not_: Optional["SchemaOrBool"] = Field(default=None, alias="not") - if_: Optional["SchemaOrBool"] = Field(default=None, alias="if") - then: Optional["SchemaOrBool"] = None - else_: Optional["SchemaOrBool"] = Field(default=None, alias="else") - dependentSchemas: Optional[dict[str, "SchemaOrBool"]] = None - prefixItems: Optional[list["SchemaOrBool"]] = None - # TODO: uncomment and remove below when deprecating Pydantic v1 - # It generates a list of schemas for tuples, before prefixItems was available - # items: Optional["SchemaOrBool"] = None - items: Optional[Union["SchemaOrBool", list["SchemaOrBool"]]] = None - contains: Optional["SchemaOrBool"] = None - properties: Optional[dict[str, "SchemaOrBool"]] = None - patternProperties: Optional[dict[str, "SchemaOrBool"]] = None - additionalProperties: Optional["SchemaOrBool"] = None - propertyNames: Optional["SchemaOrBool"] = None - unevaluatedItems: Optional["SchemaOrBool"] = None - unevaluatedProperties: Optional["SchemaOrBool"] = None - # Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-structural - # A Vocabulary for Structural Validation - type: Optional[Union[SchemaType, list[SchemaType]]] = None - enum: Optional[list[Any]] = None - const: Optional[Any] = None - multipleOf: Optional[float] = Field(default=None, gt=0) - maximum: Optional[float] = None - exclusiveMaximum: Optional[float] = None - minimum: Optional[float] = None - exclusiveMinimum: Optional[float] = None - maxLength: Optional[int] = Field(default=None, ge=0) - minLength: Optional[int] = Field(default=None, ge=0) - pattern: Optional[str] = None - maxItems: Optional[int] = Field(default=None, ge=0) - minItems: Optional[int] = Field(default=None, ge=0) - uniqueItems: Optional[bool] = None - maxContains: Optional[int] = Field(default=None, ge=0) - minContains: Optional[int] = Field(default=None, ge=0) - maxProperties: Optional[int] = Field(default=None, ge=0) - minProperties: Optional[int] = Field(default=None, ge=0) - required: Optional[list[str]] = None - dependentRequired: Optional[dict[str, set[str]]] = None - # Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-vocabularies-for-semantic-c - # Vocabularies for Semantic Content With "format" - format: Optional[str] = None - # Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-the-conten - # A Vocabulary for the Contents of String-Encoded Data - contentEncoding: Optional[str] = None - contentMediaType: Optional[str] = None - contentSchema: Optional["SchemaOrBool"] = None - # Ref: JSON Schema Validation 2020-12: https://json-schema.org/draft/2020-12/json-schema-validation.html#name-a-vocabulary-for-basic-meta - # A Vocabulary for Basic Meta-Data Annotations - title: Optional[str] = None - description: Optional[str] = None - default: Optional[Any] = None - deprecated: Optional[bool] = None - readOnly: Optional[bool] = None - writeOnly: Optional[bool] = None - examples: Optional[list[Any]] = None - # Ref: OpenAPI 3.1.0: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#schema-object - # Schema Object - discriminator: Optional[Discriminator] = None - xml: Optional[XML] = None - externalDocs: Optional[ExternalDocumentation] = None - example: Annotated[ - Optional[Any], - typing_deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = None - - -# Ref: https://json-schema.org/draft/2020-12/json-schema-core.html#name-json-schema-documents -# A JSON Schema MUST be an object or a boolean. -SchemaOrBool = Union[Schema, bool] - - -class Example(TypedDict, total=False): - summary: Optional[str] - description: Optional[str] - value: Optional[Any] - externalValue: Optional[AnyUrl] - - __pydantic_config__ = {"extra": "allow"} # type: ignore[misc] - - -class ParameterInType(Enum): - query = "query" - header = "header" - path = "path" - cookie = "cookie" - - -class Encoding(BaseModelWithConfig): - contentType: Optional[str] = None - headers: Optional[dict[str, Union["Header", Reference]]] = None - style: Optional[str] = None - explode: Optional[bool] = None - allowReserved: Optional[bool] = None - - -class MediaType(BaseModelWithConfig): - schema_: Optional[Union[Schema, Reference]] = Field(default=None, alias="schema") - example: Optional[Any] = None - examples: Optional[dict[str, Union[Example, Reference]]] = None - encoding: Optional[dict[str, Encoding]] = None - - -class ParameterBase(BaseModelWithConfig): - description: Optional[str] = None - required: Optional[bool] = None - deprecated: Optional[bool] = None - # Serialization rules for simple scenarios - style: Optional[str] = None - explode: Optional[bool] = None - allowReserved: Optional[bool] = None - schema_: Optional[Union[Schema, Reference]] = Field(default=None, alias="schema") - example: Optional[Any] = None - examples: Optional[dict[str, Union[Example, Reference]]] = None - # Serialization rules for more complex scenarios - content: Optional[dict[str, MediaType]] = None - - -class Parameter(ParameterBase): - name: str - in_: ParameterInType = Field(alias="in") - - -class Header(ParameterBase): - pass - - -class RequestBody(BaseModelWithConfig): - description: Optional[str] = None - content: dict[str, MediaType] - required: Optional[bool] = None - - -class Link(BaseModelWithConfig): - operationRef: Optional[str] = None - operationId: Optional[str] = None - parameters: Optional[dict[str, Union[Any, str]]] = None - requestBody: Optional[Union[Any, str]] = None - description: Optional[str] = None - server: Optional[Server] = None - - -class Response(BaseModelWithConfig): - description: str - headers: Optional[dict[str, Union[Header, Reference]]] = None - content: Optional[dict[str, MediaType]] = None - links: Optional[dict[str, Union[Link, Reference]]] = None - - -class Operation(BaseModelWithConfig): - tags: Optional[list[str]] = None - summary: Optional[str] = None - description: Optional[str] = None - externalDocs: Optional[ExternalDocumentation] = None - operationId: Optional[str] = None - parameters: Optional[list[Union[Parameter, Reference]]] = None - requestBody: Optional[Union[RequestBody, Reference]] = None - # Using Any for Specification Extensions - responses: Optional[dict[str, Union[Response, Any]]] = None - callbacks: Optional[dict[str, Union[dict[str, "PathItem"], Reference]]] = None - deprecated: Optional[bool] = None - security: Optional[list[dict[str, list[str]]]] = None - servers: Optional[list[Server]] = None - - -class PathItem(BaseModelWithConfig): - ref: Optional[str] = Field(default=None, alias="$ref") - summary: Optional[str] = None - description: Optional[str] = None - get: Optional[Operation] = None - put: Optional[Operation] = None - post: Optional[Operation] = None - delete: Optional[Operation] = None - options: Optional[Operation] = None - head: Optional[Operation] = None - patch: Optional[Operation] = None - trace: Optional[Operation] = None - servers: Optional[list[Server]] = None - parameters: Optional[list[Union[Parameter, Reference]]] = None - - -class SecuritySchemeType(Enum): - apiKey = "apiKey" - http = "http" - oauth2 = "oauth2" - openIdConnect = "openIdConnect" - - -class SecurityBase(BaseModelWithConfig): - type_: SecuritySchemeType = Field(alias="type") - description: Optional[str] = None - - -class APIKeyIn(Enum): - query = "query" - header = "header" - cookie = "cookie" - - -class APIKey(SecurityBase): - type_: SecuritySchemeType = Field(default=SecuritySchemeType.apiKey, alias="type") - in_: APIKeyIn = Field(alias="in") - name: str - - -class HTTPBase(SecurityBase): - type_: SecuritySchemeType = Field(default=SecuritySchemeType.http, alias="type") - scheme: str - - -class HTTPBearer(HTTPBase): - scheme: Literal["bearer"] = "bearer" - bearerFormat: Optional[str] = None - - -class OAuthFlow(BaseModelWithConfig): - refreshUrl: Optional[str] = None - scopes: dict[str, str] = {} - - -class OAuthFlowImplicit(OAuthFlow): - authorizationUrl: str - - -class OAuthFlowPassword(OAuthFlow): - tokenUrl: str - - -class OAuthFlowClientCredentials(OAuthFlow): - tokenUrl: str - - -class OAuthFlowAuthorizationCode(OAuthFlow): - authorizationUrl: str - tokenUrl: str - - -class OAuthFlows(BaseModelWithConfig): - implicit: Optional[OAuthFlowImplicit] = None - password: Optional[OAuthFlowPassword] = None - clientCredentials: Optional[OAuthFlowClientCredentials] = None - authorizationCode: Optional[OAuthFlowAuthorizationCode] = None - - -class OAuth2(SecurityBase): - type_: SecuritySchemeType = Field(default=SecuritySchemeType.oauth2, alias="type") - flows: OAuthFlows - - -class OpenIdConnect(SecurityBase): - type_: SecuritySchemeType = Field( - default=SecuritySchemeType.openIdConnect, alias="type" - ) - openIdConnectUrl: str - - -SecurityScheme = Union[APIKey, HTTPBase, OAuth2, OpenIdConnect, HTTPBearer] - - -class Components(BaseModelWithConfig): - schemas: Optional[dict[str, Union[Schema, Reference]]] = None - responses: Optional[dict[str, Union[Response, Reference]]] = None - parameters: Optional[dict[str, Union[Parameter, Reference]]] = None - examples: Optional[dict[str, Union[Example, Reference]]] = None - requestBodies: Optional[dict[str, Union[RequestBody, Reference]]] = None - headers: Optional[dict[str, Union[Header, Reference]]] = None - securitySchemes: Optional[dict[str, Union[SecurityScheme, Reference]]] = None - links: Optional[dict[str, Union[Link, Reference]]] = None - # Using Any for Specification Extensions - callbacks: Optional[dict[str, Union[dict[str, PathItem], Reference, Any]]] = None - pathItems: Optional[dict[str, Union[PathItem, Reference]]] = None - - -class Tag(BaseModelWithConfig): - name: str - description: Optional[str] = None - externalDocs: Optional[ExternalDocumentation] = None - - -class OpenAPI(BaseModelWithConfig): - openapi: str - info: Info - jsonSchemaDialect: Optional[str] = None - servers: Optional[list[Server]] = None - # Using Any for Specification Extensions - paths: Optional[dict[str, Union[PathItem, Any]]] = None - webhooks: Optional[dict[str, Union[PathItem, Reference]]] = None - components: Optional[Components] = None - security: Optional[list[dict[str, list[str]]]] = None - tags: Optional[list[Tag]] = None - externalDocs: Optional[ExternalDocumentation] = None - - -Schema.model_rebuild() -Operation.model_rebuild() -Encoding.model_rebuild() diff --git a/.env/Lib/site-packages/fastapi/openapi/utils.py b/.env/Lib/site-packages/fastapi/openapi/utils.py deleted file mode 100644 index 5736af3..0000000 --- a/.env/Lib/site-packages/fastapi/openapi/utils.py +++ /dev/null @@ -1,570 +0,0 @@ -import copy -import http.client -import inspect -import warnings -from collections.abc import Sequence -from typing import Any, Optional, Union, cast - -from fastapi import routing -from fastapi._compat import ( - ModelField, - Undefined, - get_compat_model_name_map, - get_definitions, - get_schema_from_model_field, - lenient_issubclass, -) -from fastapi.datastructures import DefaultPlaceholder -from fastapi.dependencies.models import Dependant -from fastapi.dependencies.utils import ( - _get_flat_fields_from_params, - get_flat_dependant, - get_flat_params, - get_validation_alias, -) -from fastapi.encoders import jsonable_encoder -from fastapi.exceptions import FastAPIDeprecationWarning -from fastapi.openapi.constants import METHODS_WITH_BODY, REF_PREFIX -from fastapi.openapi.models import OpenAPI -from fastapi.params import Body, ParamTypes -from fastapi.responses import Response -from fastapi.types import ModelNameMap -from fastapi.utils import ( - deep_dict_update, - generate_operation_id_for_path, - is_body_allowed_for_status_code, -) -from pydantic import BaseModel -from starlette.responses import JSONResponse -from starlette.routing import BaseRoute -from typing_extensions import Literal - -validation_error_definition = { - "title": "ValidationError", - "type": "object", - "properties": { - "loc": { - "title": "Location", - "type": "array", - "items": {"anyOf": [{"type": "string"}, {"type": "integer"}]}, - }, - "msg": {"title": "Message", "type": "string"}, - "type": {"title": "Error Type", "type": "string"}, - "input": {"title": "Input"}, - "ctx": {"title": "Context", "type": "object"}, - }, - "required": ["loc", "msg", "type"], -} - -validation_error_response_definition = { - "title": "HTTPValidationError", - "type": "object", - "properties": { - "detail": { - "title": "Detail", - "type": "array", - "items": {"$ref": REF_PREFIX + "ValidationError"}, - } - }, -} - -status_code_ranges: dict[str, str] = { - "1XX": "Information", - "2XX": "Success", - "3XX": "Redirection", - "4XX": "Client Error", - "5XX": "Server Error", - "DEFAULT": "Default Response", -} - - -def get_openapi_security_definitions( - flat_dependant: Dependant, -) -> tuple[dict[str, Any], list[dict[str, Any]]]: - security_definitions = {} - # Use a dict to merge scopes for same security scheme - operation_security_dict: dict[str, list[str]] = {} - for security_dependency in flat_dependant._security_dependencies: - security_definition = jsonable_encoder( - security_dependency._security_scheme.model, - by_alias=True, - exclude_none=True, - ) - security_name = security_dependency._security_scheme.scheme_name - security_definitions[security_name] = security_definition - # Merge scopes for the same security scheme - if security_name not in operation_security_dict: - operation_security_dict[security_name] = [] - for scope in security_dependency.oauth_scopes or []: - if scope not in operation_security_dict[security_name]: - operation_security_dict[security_name].append(scope) - operation_security = [ - {name: scopes} for name, scopes in operation_security_dict.items() - ] - return security_definitions, operation_security - - -def _get_openapi_operation_parameters( - *, - dependant: Dependant, - model_name_map: ModelNameMap, - field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], dict[str, Any] - ], - separate_input_output_schemas: bool = True, -) -> list[dict[str, Any]]: - parameters = [] - flat_dependant = get_flat_dependant(dependant, skip_repeats=True) - path_params = _get_flat_fields_from_params(flat_dependant.path_params) - query_params = _get_flat_fields_from_params(flat_dependant.query_params) - header_params = _get_flat_fields_from_params(flat_dependant.header_params) - cookie_params = _get_flat_fields_from_params(flat_dependant.cookie_params) - parameter_groups = [ - (ParamTypes.path, path_params), - (ParamTypes.query, query_params), - (ParamTypes.header, header_params), - (ParamTypes.cookie, cookie_params), - ] - default_convert_underscores = True - if len(flat_dependant.header_params) == 1: - first_field = flat_dependant.header_params[0] - if lenient_issubclass(first_field.type_, BaseModel): - default_convert_underscores = getattr( - first_field.field_info, "convert_underscores", True - ) - for param_type, param_group in parameter_groups: - for param in param_group: - field_info = param.field_info - # field_info = cast(Param, field_info) - if not getattr(field_info, "include_in_schema", True): - continue - param_schema = get_schema_from_model_field( - field=param, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - name = get_validation_alias(param) - convert_underscores = getattr( - param.field_info, - "convert_underscores", - default_convert_underscores, - ) - if ( - param_type == ParamTypes.header - and name == param.name - and convert_underscores - ): - name = param.name.replace("_", "-") - - parameter = { - "name": name, - "in": param_type.value, - "required": param.required, - "schema": param_schema, - } - if field_info.description: - parameter["description"] = field_info.description - openapi_examples = getattr(field_info, "openapi_examples", None) - example = getattr(field_info, "example", None) - if openapi_examples: - parameter["examples"] = jsonable_encoder(openapi_examples) - elif example != Undefined: - parameter["example"] = jsonable_encoder(example) - if getattr(field_info, "deprecated", None): - parameter["deprecated"] = True - parameters.append(parameter) - return parameters - - -def get_openapi_operation_request_body( - *, - body_field: Optional[ModelField], - model_name_map: ModelNameMap, - field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], dict[str, Any] - ], - separate_input_output_schemas: bool = True, -) -> Optional[dict[str, Any]]: - if not body_field: - return None - assert isinstance(body_field, ModelField) - body_schema = get_schema_from_model_field( - field=body_field, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - field_info = cast(Body, body_field.field_info) - request_media_type = field_info.media_type - required = body_field.required - request_body_oai: dict[str, Any] = {} - if required: - request_body_oai["required"] = required - request_media_content: dict[str, Any] = {"schema": body_schema} - if field_info.openapi_examples: - request_media_content["examples"] = jsonable_encoder( - field_info.openapi_examples - ) - elif field_info.example != Undefined: - request_media_content["example"] = jsonable_encoder(field_info.example) - request_body_oai["content"] = {request_media_type: request_media_content} - return request_body_oai - - -def generate_operation_id( - *, route: routing.APIRoute, method: str -) -> str: # pragma: nocover - warnings.warn( - message="fastapi.openapi.utils.generate_operation_id() was deprecated, " - "it is not used internally, and will be removed soon", - category=FastAPIDeprecationWarning, - stacklevel=2, - ) - if route.operation_id: - return route.operation_id - path: str = route.path_format - return generate_operation_id_for_path(name=route.name, path=path, method=method) - - -def generate_operation_summary(*, route: routing.APIRoute, method: str) -> str: - if route.summary: - return route.summary - return route.name.replace("_", " ").title() - - -def get_openapi_operation_metadata( - *, route: routing.APIRoute, method: str, operation_ids: set[str] -) -> dict[str, Any]: - operation: dict[str, Any] = {} - if route.tags: - operation["tags"] = route.tags - operation["summary"] = generate_operation_summary(route=route, method=method) - if route.description: - operation["description"] = route.description - operation_id = route.operation_id or route.unique_id - if operation_id in operation_ids: - message = ( - f"Duplicate Operation ID {operation_id} for function " - + f"{route.endpoint.__name__}" - ) - file_name = getattr(route.endpoint, "__globals__", {}).get("__file__") - if file_name: - message += f" at {file_name}" - warnings.warn(message, stacklevel=1) - operation_ids.add(operation_id) - operation["operationId"] = operation_id - if route.deprecated: - operation["deprecated"] = route.deprecated - return operation - - -def get_openapi_path( - *, - route: routing.APIRoute, - operation_ids: set[str], - model_name_map: ModelNameMap, - field_mapping: dict[ - tuple[ModelField, Literal["validation", "serialization"]], dict[str, Any] - ], - separate_input_output_schemas: bool = True, -) -> tuple[dict[str, Any], dict[str, Any], dict[str, Any]]: - path = {} - security_schemes: dict[str, Any] = {} - definitions: dict[str, Any] = {} - assert route.methods is not None, "Methods must be a list" - if isinstance(route.response_class, DefaultPlaceholder): - current_response_class: type[Response] = route.response_class.value - else: - current_response_class = route.response_class - assert current_response_class, "A response class is needed to generate OpenAPI" - route_response_media_type: Optional[str] = current_response_class.media_type - if route.include_in_schema: - for method in route.methods: - operation = get_openapi_operation_metadata( - route=route, method=method, operation_ids=operation_ids - ) - parameters: list[dict[str, Any]] = [] - flat_dependant = get_flat_dependant(route.dependant, skip_repeats=True) - security_definitions, operation_security = get_openapi_security_definitions( - flat_dependant=flat_dependant - ) - if operation_security: - operation.setdefault("security", []).extend(operation_security) - if security_definitions: - security_schemes.update(security_definitions) - operation_parameters = _get_openapi_operation_parameters( - dependant=route.dependant, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - parameters.extend(operation_parameters) - if parameters: - all_parameters = { - (param["in"], param["name"]): param for param in parameters - } - required_parameters = { - (param["in"], param["name"]): param - for param in parameters - if param.get("required") - } - # Make sure required definitions of the same parameter take precedence - # over non-required definitions - all_parameters.update(required_parameters) - operation["parameters"] = list(all_parameters.values()) - if method in METHODS_WITH_BODY: - request_body_oai = get_openapi_operation_request_body( - body_field=route.body_field, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - if request_body_oai: - operation["requestBody"] = request_body_oai - if route.callbacks: - callbacks = {} - for callback in route.callbacks: - if isinstance(callback, routing.APIRoute): - ( - cb_path, - cb_security_schemes, - cb_definitions, - ) = get_openapi_path( - route=callback, - operation_ids=operation_ids, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - callbacks[callback.name] = {callback.path: cb_path} - operation["callbacks"] = callbacks - if route.status_code is not None: - status_code = str(route.status_code) - else: - # It would probably make more sense for all response classes to have an - # explicit default status_code, and to extract it from them, instead of - # doing this inspection tricks, that would probably be in the future - # TODO: probably make status_code a default class attribute for all - # responses in Starlette - response_signature = inspect.signature(current_response_class.__init__) - status_code_param = response_signature.parameters.get("status_code") - if status_code_param is not None: - if isinstance(status_code_param.default, int): - status_code = str(status_code_param.default) - operation.setdefault("responses", {}).setdefault(status_code, {})[ - "description" - ] = route.response_description - if route_response_media_type and is_body_allowed_for_status_code( - route.status_code - ): - response_schema = {"type": "string"} - if lenient_issubclass(current_response_class, JSONResponse): - if route.response_field: - response_schema = get_schema_from_model_field( - field=route.response_field, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - else: - response_schema = {} - operation.setdefault("responses", {}).setdefault( - status_code, {} - ).setdefault("content", {}).setdefault(route_response_media_type, {})[ - "schema" - ] = response_schema - if route.responses: - operation_responses = operation.setdefault("responses", {}) - for ( - additional_status_code, - additional_response, - ) in route.responses.items(): - process_response = copy.deepcopy(additional_response) - process_response.pop("model", None) - status_code_key = str(additional_status_code).upper() - if status_code_key == "DEFAULT": - status_code_key = "default" - openapi_response = operation_responses.setdefault( - status_code_key, {} - ) - assert isinstance(process_response, dict), ( - "An additional response must be a dict" - ) - field = route.response_fields.get(additional_status_code) - additional_field_schema: Optional[dict[str, Any]] = None - if field: - additional_field_schema = get_schema_from_model_field( - field=field, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - media_type = route_response_media_type or "application/json" - additional_schema = ( - process_response.setdefault("content", {}) - .setdefault(media_type, {}) - .setdefault("schema", {}) - ) - deep_dict_update(additional_schema, additional_field_schema) - status_text: Optional[str] = status_code_ranges.get( - str(additional_status_code).upper() - ) or http.client.responses.get(int(additional_status_code)) - description = ( - process_response.get("description") - or openapi_response.get("description") - or status_text - or "Additional Response" - ) - deep_dict_update(openapi_response, process_response) - openapi_response["description"] = description - http422 = "422" - all_route_params = get_flat_params(route.dependant) - if (all_route_params or route.body_field) and not any( - status in operation["responses"] - for status in [http422, "4XX", "default"] - ): - operation["responses"][http422] = { - "description": "Validation Error", - "content": { - "application/json": { - "schema": {"$ref": REF_PREFIX + "HTTPValidationError"} - } - }, - } - if "ValidationError" not in definitions: - definitions.update( - { - "ValidationError": validation_error_definition, - "HTTPValidationError": validation_error_response_definition, - } - ) - if route.openapi_extra: - deep_dict_update(operation, route.openapi_extra) - path[method.lower()] = operation - return path, security_schemes, definitions - - -def get_fields_from_routes( - routes: Sequence[BaseRoute], -) -> list[ModelField]: - body_fields_from_routes: list[ModelField] = [] - responses_from_routes: list[ModelField] = [] - request_fields_from_routes: list[ModelField] = [] - callback_flat_models: list[ModelField] = [] - for route in routes: - if getattr(route, "include_in_schema", None) and isinstance( - route, routing.APIRoute - ): - if route.body_field: - assert isinstance(route.body_field, ModelField), ( - "A request body must be a Pydantic Field" - ) - body_fields_from_routes.append(route.body_field) - if route.response_field: - responses_from_routes.append(route.response_field) - if route.response_fields: - responses_from_routes.extend(route.response_fields.values()) - if route.callbacks: - callback_flat_models.extend(get_fields_from_routes(route.callbacks)) - params = get_flat_params(route.dependant) - request_fields_from_routes.extend(params) - - flat_models = callback_flat_models + list( - body_fields_from_routes + responses_from_routes + request_fields_from_routes - ) - return flat_models - - -def get_openapi( - *, - title: str, - version: str, - openapi_version: str = "3.1.0", - summary: Optional[str] = None, - description: Optional[str] = None, - routes: Sequence[BaseRoute], - webhooks: Optional[Sequence[BaseRoute]] = None, - tags: Optional[list[dict[str, Any]]] = None, - servers: Optional[list[dict[str, Union[str, Any]]]] = None, - terms_of_service: Optional[str] = None, - contact: Optional[dict[str, Union[str, Any]]] = None, - license_info: Optional[dict[str, Union[str, Any]]] = None, - separate_input_output_schemas: bool = True, - external_docs: Optional[dict[str, Any]] = None, -) -> dict[str, Any]: - info: dict[str, Any] = {"title": title, "version": version} - if summary: - info["summary"] = summary - if description: - info["description"] = description - if terms_of_service: - info["termsOfService"] = terms_of_service - if contact: - info["contact"] = contact - if license_info: - info["license"] = license_info - output: dict[str, Any] = {"openapi": openapi_version, "info": info} - if servers: - output["servers"] = servers - components: dict[str, dict[str, Any]] = {} - paths: dict[str, dict[str, Any]] = {} - webhook_paths: dict[str, dict[str, Any]] = {} - operation_ids: set[str] = set() - all_fields = get_fields_from_routes(list(routes or []) + list(webhooks or [])) - model_name_map = get_compat_model_name_map(all_fields) - field_mapping, definitions = get_definitions( - fields=all_fields, - model_name_map=model_name_map, - separate_input_output_schemas=separate_input_output_schemas, - ) - for route in routes or []: - if isinstance(route, routing.APIRoute): - result = get_openapi_path( - route=route, - operation_ids=operation_ids, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - if result: - path, security_schemes, path_definitions = result - if path: - paths.setdefault(route.path_format, {}).update(path) - if security_schemes: - components.setdefault("securitySchemes", {}).update( - security_schemes - ) - if path_definitions: - definitions.update(path_definitions) - for webhook in webhooks or []: - if isinstance(webhook, routing.APIRoute): - result = get_openapi_path( - route=webhook, - operation_ids=operation_ids, - model_name_map=model_name_map, - field_mapping=field_mapping, - separate_input_output_schemas=separate_input_output_schemas, - ) - if result: - path, security_schemes, path_definitions = result - if path: - webhook_paths.setdefault(webhook.path_format, {}).update(path) - if security_schemes: - components.setdefault("securitySchemes", {}).update( - security_schemes - ) - if path_definitions: - definitions.update(path_definitions) - if definitions: - components["schemas"] = {k: definitions[k] for k in sorted(definitions)} - if components: - output["components"] = components - output["paths"] = paths - if webhook_paths: - output["webhooks"] = webhook_paths - if tags: - output["tags"] = tags - if external_docs: - output["externalDocs"] = external_docs - return jsonable_encoder(OpenAPI(**output), by_alias=True, exclude_none=True) # type: ignore diff --git a/.env/Lib/site-packages/fastapi/param_functions.py b/.env/Lib/site-packages/fastapi/param_functions.py deleted file mode 100644 index 9bd92be..0000000 --- a/.env/Lib/site-packages/fastapi/param_functions.py +++ /dev/null @@ -1,2461 +0,0 @@ -from collections.abc import Sequence -from typing import Annotated, Any, Callable, Optional, Union - -from annotated_doc import Doc -from fastapi import params -from fastapi._compat import Undefined -from fastapi.openapi.models import Example -from pydantic import AliasChoices, AliasPath -from typing_extensions import Literal, deprecated - -_Unset: Any = Undefined - - -def Path( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = ..., - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - - Read more about it in the - [FastAPI docs for Path Parameters and Numeric Validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#declare-metadata) - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - """ - Declare a path parameter for a *path operation*. - - Read more about it in the - [FastAPI docs for Path Parameters and Numeric Validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/). - - ```python - from typing import Annotated - - from fastapi import FastAPI, Path - - app = FastAPI() - - - @app.get("/items/{item_id}") - async def read_items( - item_id: Annotated[int, Path(title="The ID of the item to get")], - ): - return {"item_id": item_id} - ``` - """ - return params.Path( - default=default, - default_factory=default_factory, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Query( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#alternative-old-query-as-the-default-value) - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#alias-parameters) - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#declare-more-metadata) - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#declare-more-metadata) - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - - Read more about it in the - [FastAPI docs about Path parameters numeric validations](https://fastapi.tiangolo.com/tutorial/path-params-numeric-validations/#number-validations-greater-than-and-less-than-or-equal) - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/) - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/) - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#add-regular-expressions - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#deprecating-parameters) - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs about Query parameters](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.Query( - default=default, - default_factory=default_factory, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Header( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - convert_underscores: Annotated[ - bool, - Doc( - """ - Automatically convert underscores to hyphens in the parameter field name. - - Read more about it in the - [FastAPI docs for Header Parameters](https://fastapi.tiangolo.com/tutorial/header-params/#automatic-conversion) - """ - ), - ] = True, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.Header( - default=default, - default_factory=default_factory, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - convert_underscores=convert_underscores, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Cookie( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.Cookie( - default=default, - default_factory=default_factory, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Body( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - embed: Annotated[ - Union[bool, None], - Doc( - """ - When `embed` is `True`, the parameter will be expected in a JSON body as a - key instead of being the JSON body itself. - - This happens automatically when more than one `Body` parameter is declared. - - Read more about it in the - [FastAPI docs for Body - Multiple Parameters](https://fastapi.tiangolo.com/tutorial/body-multiple-params/#embed-a-single-body-parameter). - """ - ), - ] = None, - media_type: Annotated[ - str, - Doc( - """ - The media type of this parameter field. Changing it would affect the - generated OpenAPI, but currently it doesn't affect the parsing of the data. - """ - ), - ] = "application/json", - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.Body( - default=default, - default_factory=default_factory, - embed=embed, - media_type=media_type, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Form( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - media_type: Annotated[ - str, - Doc( - """ - The media type of this parameter field. Changing it would affect the - generated OpenAPI, but currently it doesn't affect the parsing of the data. - """ - ), - ] = "application/x-www-form-urlencoded", - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.Form( - default=default, - default_factory=default_factory, - media_type=media_type, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def File( # noqa: N802 - default: Annotated[ - Any, - Doc( - """ - Default value if the parameter field is not set. - """ - ), - ] = Undefined, - *, - default_factory: Annotated[ - Union[Callable[[], Any], None], - Doc( - """ - A callable to generate the default value. - - This doesn't affect `Path` parameters as the value is always required. - The parameter is available only for compatibility. - """ - ), - ] = _Unset, - media_type: Annotated[ - str, - Doc( - """ - The media type of this parameter field. Changing it would affect the - generated OpenAPI, but currently it doesn't affect the parsing of the data. - """ - ), - ] = "multipart/form-data", - alias: Annotated[ - Optional[str], - Doc( - """ - An alternative name for the parameter field. - - This will be used to extract the data and for the generated OpenAPI. - It is particularly useful when you can't use the name you want because it - is a Python reserved keyword or similar. - """ - ), - ] = None, - alias_priority: Annotated[ - Union[int, None], - Doc( - """ - Priority of the alias. This affects whether an alias generator is used. - """ - ), - ] = _Unset, - validation_alias: Annotated[ - Union[str, AliasPath, AliasChoices, None], - Doc( - """ - 'Whitelist' validation step. The parameter field will be the single one - allowed by the alias or set of aliases defined. - """ - ), - ] = None, - serialization_alias: Annotated[ - Union[str, None], - Doc( - """ - 'Blacklist' validation step. The vanilla parameter field will be the - single one of the alias' or set of aliases' fields and all the other - fields will be ignored at serialization time. - """ - ), - ] = None, - title: Annotated[ - Optional[str], - Doc( - """ - Human-readable title. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Human-readable description. - """ - ), - ] = None, - gt: Annotated[ - Optional[float], - Doc( - """ - Greater than. If set, value must be greater than this. Only applicable to - numbers. - """ - ), - ] = None, - ge: Annotated[ - Optional[float], - Doc( - """ - Greater than or equal. If set, value must be greater than or equal to - this. Only applicable to numbers. - """ - ), - ] = None, - lt: Annotated[ - Optional[float], - Doc( - """ - Less than. If set, value must be less than this. Only applicable to numbers. - """ - ), - ] = None, - le: Annotated[ - Optional[float], - Doc( - """ - Less than or equal. If set, value must be less than or equal to this. - Only applicable to numbers. - """ - ), - ] = None, - min_length: Annotated[ - Optional[int], - Doc( - """ - Minimum length for strings. - """ - ), - ] = None, - max_length: Annotated[ - Optional[int], - Doc( - """ - Maximum length for strings. - """ - ), - ] = None, - pattern: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - ] = None, - regex: Annotated[ - Optional[str], - Doc( - """ - RegEx pattern for strings. - """ - ), - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Annotated[ - Union[str, None], - Doc( - """ - Parameter field name for discriminating the type in a tagged union. - """ - ), - ] = None, - strict: Annotated[ - Union[bool, None], - Doc( - """ - If `True`, strict validation is applied to the field. - """ - ), - ] = _Unset, - multiple_of: Annotated[ - Union[float, None], - Doc( - """ - Value must be a multiple of this. Only applicable to numbers. - """ - ), - ] = _Unset, - allow_inf_nan: Annotated[ - Union[bool, None], - Doc( - """ - Allow `inf`, `-inf`, `nan`. Only applicable to numbers. - """ - ), - ] = _Unset, - max_digits: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of allow digits for strings. - """ - ), - ] = _Unset, - decimal_places: Annotated[ - Union[int, None], - Doc( - """ - Maximum number of decimal places allowed for numbers. - """ - ), - ] = _Unset, - examples: Annotated[ - Optional[list[Any]], - Doc( - """ - Example values for this field. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/) - """ - ), - ] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Annotated[ - Optional[dict[str, Example]], - Doc( - """ - OpenAPI-specific examples. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Swagger UI (that provides the `/docs` interface) has better support for the - OpenAPI-specific examples than the JSON Schema `examples`, that's the main - use case for this. - - Read more about it in the - [FastAPI docs for Declare Request Example Data](https://fastapi.tiangolo.com/tutorial/schema-extra-example/#using-the-openapi_examples-parameter). - """ - ), - ] = None, - deprecated: Annotated[ - Union[deprecated, str, bool, None], - Doc( - """ - Mark this parameter field as deprecated. - - It will affect the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) this parameter field in the generated OpenAPI. - You probably don't need it, but it's available. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - json_schema_extra: Annotated[ - Union[dict[str, Any], None], - Doc( - """ - Any additional JSON schema data. - """ - ), - ] = None, - **extra: Annotated[ - Any, - Doc( - """ - Include extra fields used by the JSON Schema. - """ - ), - deprecated( - """ - The `extra` kwargs is deprecated. Use `json_schema_extra` instead. - """ - ), - ], -) -> Any: - return params.File( - default=default, - default_factory=default_factory, - media_type=media_type, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - example=example, - examples=examples, - openapi_examples=openapi_examples, - deprecated=deprecated, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -def Depends( # noqa: N802 - dependency: Annotated[ - Optional[Callable[..., Any]], - Doc( - """ - A "dependable" callable (like a function). - - Don't call it directly, FastAPI will call it for you, just pass the object - directly. - - Read more about it in the - [FastAPI docs for Dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/) - """ - ), - ] = None, - *, - use_cache: Annotated[ - bool, - Doc( - """ - By default, after a dependency is called the first time in a request, if - the dependency is declared again for the rest of the request (for example - if the dependency is needed by several dependencies), the value will be - re-used for the rest of the request. - - Set `use_cache` to `False` to disable this behavior and ensure the - dependency is called again (if declared more than once) in the same request. - - Read more about it in the - [FastAPI docs about sub-dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/sub-dependencies/#using-the-same-dependency-multiple-times) - """ - ), - ] = True, - scope: Annotated[ - Union[Literal["function", "request"], None], - Doc( - """ - Mainly for dependencies with `yield`, define when the dependency function - should start (the code before `yield`) and when it should end (the code - after `yield`). - - * `"function"`: start the dependency before the *path operation function* - that handles the request, end the dependency after the *path operation - function* ends, but **before** the response is sent back to the client. - So, the dependency function will be executed **around** the *path operation - **function***. - * `"request"`: start the dependency before the *path operation function* - that handles the request (similar to when using `"function"`), but end - **after** the response is sent back to the client. So, the dependency - function will be executed **around** the **request** and response cycle. - - Read more about it in the - [FastAPI docs for FastAPI Dependencies with yield](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-with-yield/#early-exit-and-scope) - """ - ), - ] = None, -) -> Any: - """ - Declare a FastAPI dependency. - - It takes a single "dependable" callable (like a function). - - Don't call it directly, FastAPI will call it for you. - - Read more about it in the - [FastAPI docs for Dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/). - - **Example** - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - - app = FastAPI() - - - async def common_parameters(q: str | None = None, skip: int = 0, limit: int = 100): - return {"q": q, "skip": skip, "limit": limit} - - - @app.get("/items/") - async def read_items(commons: Annotated[dict, Depends(common_parameters)]): - return commons - ``` - """ - return params.Depends(dependency=dependency, use_cache=use_cache, scope=scope) - - -def Security( # noqa: N802 - dependency: Annotated[ - Optional[Callable[..., Any]], - Doc( - """ - A "dependable" callable (like a function). - - Don't call it directly, FastAPI will call it for you, just pass the object - directly. - - Read more about it in the - [FastAPI docs for Dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/) - """ - ), - ] = None, - *, - scopes: Annotated[ - Optional[Sequence[str]], - Doc( - """ - OAuth2 scopes required for the *path operation* that uses this Security - dependency. - - The term "scope" comes from the OAuth2 specification, it seems to be - intentionally vague and interpretable. It normally refers to permissions, - in cases to roles. - - These scopes are integrated with OpenAPI (and the API docs at `/docs`). - So they are visible in the OpenAPI specification. - - Read more about it in the - [FastAPI docs about OAuth2 scopes](https://fastapi.tiangolo.com/advanced/security/oauth2-scopes/) - """ - ), - ] = None, - use_cache: Annotated[ - bool, - Doc( - """ - By default, after a dependency is called the first time in a request, if - the dependency is declared again for the rest of the request (for example - if the dependency is needed by several dependencies), the value will be - re-used for the rest of the request. - - Set `use_cache` to `False` to disable this behavior and ensure the - dependency is called again (if declared more than once) in the same request. - - Read more about it in the - [FastAPI docs about sub-dependencies](https://fastapi.tiangolo.com/tutorial/dependencies/sub-dependencies/#using-the-same-dependency-multiple-times) - """ - ), - ] = True, -) -> Any: - """ - Declare a FastAPI Security dependency. - - The only difference with a regular dependency is that it can declare OAuth2 - scopes that will be integrated with OpenAPI and the automatic UI docs (by default - at `/docs`). - - It takes a single "dependable" callable (like a function). - - Don't call it directly, FastAPI will call it for you. - - Read more about it in the - [FastAPI docs for Security](https://fastapi.tiangolo.com/tutorial/security/) and - in the - [FastAPI docs for OAuth2 scopes](https://fastapi.tiangolo.com/advanced/security/oauth2-scopes/). - - **Example** - - ```python - from typing import Annotated - - from fastapi import Security, FastAPI - - from .db import User - from .security import get_current_active_user - - app = FastAPI() - - @app.get("/users/me/items/") - async def read_own_items( - current_user: Annotated[User, Security(get_current_active_user, scopes=["items"])] - ): - return [{"item_id": "Foo", "owner": current_user.username}] - ``` - """ - return params.Security(dependency=dependency, scopes=scopes, use_cache=use_cache) diff --git a/.env/Lib/site-packages/fastapi/params.py b/.env/Lib/site-packages/fastapi/params.py deleted file mode 100644 index 72e797f..0000000 --- a/.env/Lib/site-packages/fastapi/params.py +++ /dev/null @@ -1,755 +0,0 @@ -import warnings -from collections.abc import Sequence -from dataclasses import dataclass -from enum import Enum -from typing import Annotated, Any, Callable, Optional, Union - -from fastapi.exceptions import FastAPIDeprecationWarning -from fastapi.openapi.models import Example -from pydantic import AliasChoices, AliasPath -from pydantic.fields import FieldInfo -from typing_extensions import Literal, deprecated - -from ._compat import ( - Undefined, -) - -_Unset: Any = Undefined - - -class ParamTypes(Enum): - query = "query" - header = "header" - path = "path" - cookie = "cookie" - - -class Param(FieldInfo): # type: ignore[misc] - in_: ParamTypes - - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - if example is not _Unset: - warnings.warn( - "`example` has been deprecated, please use `examples` instead", - category=FastAPIDeprecationWarning, - stacklevel=4, - ) - self.example = example - self.include_in_schema = include_in_schema - self.openapi_examples = openapi_examples - kwargs = dict( - default=default, - default_factory=default_factory, - alias=alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - discriminator=discriminator, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - **extra, - ) - if examples is not None: - kwargs["examples"] = examples - if regex is not None: - warnings.warn( - "`regex` has been deprecated, please use `pattern` instead", - category=FastAPIDeprecationWarning, - stacklevel=4, - ) - current_json_schema_extra = json_schema_extra or extra - kwargs["deprecated"] = deprecated - - if serialization_alias in (_Unset, None) and isinstance(alias, str): - serialization_alias = alias - if validation_alias in (_Unset, None): - validation_alias = alias - kwargs.update( - { - "annotation": annotation, - "alias_priority": alias_priority, - "validation_alias": validation_alias, - "serialization_alias": serialization_alias, - "strict": strict, - "json_schema_extra": current_json_schema_extra, - } - ) - kwargs["pattern"] = pattern or regex - - use_kwargs = {k: v for k, v in kwargs.items() if v is not _Unset} - - super().__init__(**use_kwargs) - - def __repr__(self) -> str: - return f"{self.__class__.__name__}({self.default})" - - -class Path(Param): # type: ignore[misc] - in_ = ParamTypes.path - - def __init__( - self, - default: Any = ..., - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - assert default is ..., "Path parameters cannot have a default value" - self.in_ = self.in_ - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -class Query(Param): # type: ignore[misc] - in_ = ParamTypes.query - - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -class Header(Param): # type: ignore[misc] - in_ = ParamTypes.header - - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - convert_underscores: bool = True, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - self.convert_underscores = convert_underscores - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -class Cookie(Param): # type: ignore[misc] - in_ = ParamTypes.cookie - - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -class Body(FieldInfo): # type: ignore[misc] - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - embed: Union[bool, None] = None, - media_type: str = "application/json", - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - self.embed = embed - self.media_type = media_type - if example is not _Unset: - warnings.warn( - "`example` has been deprecated, please use `examples` instead", - category=FastAPIDeprecationWarning, - stacklevel=4, - ) - self.example = example - self.include_in_schema = include_in_schema - self.openapi_examples = openapi_examples - kwargs = dict( - default=default, - default_factory=default_factory, - alias=alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - discriminator=discriminator, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - **extra, - ) - if examples is not None: - kwargs["examples"] = examples - if regex is not None: - warnings.warn( - "`regex` has been deprecated, please use `pattern` instead", - category=FastAPIDeprecationWarning, - stacklevel=4, - ) - current_json_schema_extra = json_schema_extra or extra - kwargs["deprecated"] = deprecated - if serialization_alias in (_Unset, None) and isinstance(alias, str): - serialization_alias = alias - if validation_alias in (_Unset, None): - validation_alias = alias - kwargs.update( - { - "annotation": annotation, - "alias_priority": alias_priority, - "validation_alias": validation_alias, - "serialization_alias": serialization_alias, - "strict": strict, - "json_schema_extra": current_json_schema_extra, - } - ) - kwargs["pattern"] = pattern or regex - - use_kwargs = {k: v for k, v in kwargs.items() if v is not _Unset} - - super().__init__(**use_kwargs) - - def __repr__(self) -> str: - return f"{self.__class__.__name__}({self.default})" - - -class Form(Body): # type: ignore[misc] - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - media_type: str = "application/x-www-form-urlencoded", - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - media_type=media_type, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -class File(Form): # type: ignore[misc] - def __init__( - self, - default: Any = Undefined, - *, - default_factory: Union[Callable[[], Any], None] = _Unset, - annotation: Optional[Any] = None, - media_type: str = "multipart/form-data", - alias: Optional[str] = None, - alias_priority: Union[int, None] = _Unset, - validation_alias: Union[str, AliasPath, AliasChoices, None] = None, - serialization_alias: Union[str, None] = None, - title: Optional[str] = None, - description: Optional[str] = None, - gt: Optional[float] = None, - ge: Optional[float] = None, - lt: Optional[float] = None, - le: Optional[float] = None, - min_length: Optional[int] = None, - max_length: Optional[int] = None, - pattern: Optional[str] = None, - regex: Annotated[ - Optional[str], - deprecated( - "Deprecated in FastAPI 0.100.0 and Pydantic v2, use `pattern` instead." - ), - ] = None, - discriminator: Union[str, None] = None, - strict: Union[bool, None] = _Unset, - multiple_of: Union[float, None] = _Unset, - allow_inf_nan: Union[bool, None] = _Unset, - max_digits: Union[int, None] = _Unset, - decimal_places: Union[int, None] = _Unset, - examples: Optional[list[Any]] = None, - example: Annotated[ - Optional[Any], - deprecated( - "Deprecated in OpenAPI 3.1.0 that now uses JSON Schema 2020-12, " - "although still supported. Use examples instead." - ), - ] = _Unset, - openapi_examples: Optional[dict[str, Example]] = None, - deprecated: Union[deprecated, str, bool, None] = None, - include_in_schema: bool = True, - json_schema_extra: Union[dict[str, Any], None] = None, - **extra: Any, - ): - super().__init__( - default=default, - default_factory=default_factory, - annotation=annotation, - media_type=media_type, - alias=alias, - alias_priority=alias_priority, - validation_alias=validation_alias, - serialization_alias=serialization_alias, - title=title, - description=description, - gt=gt, - ge=ge, - lt=lt, - le=le, - min_length=min_length, - max_length=max_length, - pattern=pattern, - regex=regex, - discriminator=discriminator, - strict=strict, - multiple_of=multiple_of, - allow_inf_nan=allow_inf_nan, - max_digits=max_digits, - decimal_places=decimal_places, - deprecated=deprecated, - example=example, - examples=examples, - openapi_examples=openapi_examples, - include_in_schema=include_in_schema, - json_schema_extra=json_schema_extra, - **extra, - ) - - -@dataclass(frozen=True) -class Depends: - dependency: Optional[Callable[..., Any]] = None - use_cache: bool = True - scope: Union[Literal["function", "request"], None] = None - - -@dataclass(frozen=True) -class Security(Depends): - scopes: Optional[Sequence[str]] = None diff --git a/.env/Lib/site-packages/fastapi/py.typed b/.env/Lib/site-packages/fastapi/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/fastapi/requests.py b/.env/Lib/site-packages/fastapi/requests.py deleted file mode 100644 index d16552c..0000000 --- a/.env/Lib/site-packages/fastapi/requests.py +++ /dev/null @@ -1,2 +0,0 @@ -from starlette.requests import HTTPConnection as HTTPConnection # noqa: F401 -from starlette.requests import Request as Request # noqa: F401 diff --git a/.env/Lib/site-packages/fastapi/responses.py b/.env/Lib/site-packages/fastapi/responses.py deleted file mode 100644 index 6c8db6f..0000000 --- a/.env/Lib/site-packages/fastapi/responses.py +++ /dev/null @@ -1,48 +0,0 @@ -from typing import Any - -from starlette.responses import FileResponse as FileResponse # noqa -from starlette.responses import HTMLResponse as HTMLResponse # noqa -from starlette.responses import JSONResponse as JSONResponse # noqa -from starlette.responses import PlainTextResponse as PlainTextResponse # noqa -from starlette.responses import RedirectResponse as RedirectResponse # noqa -from starlette.responses import Response as Response # noqa -from starlette.responses import StreamingResponse as StreamingResponse # noqa - -try: - import ujson -except ImportError: # pragma: nocover - ujson = None # type: ignore - - -try: - import orjson -except ImportError: # pragma: nocover - orjson = None # type: ignore - - -class UJSONResponse(JSONResponse): - """ - JSON response using the high-performance ujson library to serialize data to JSON. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/). - """ - - def render(self, content: Any) -> bytes: - assert ujson is not None, "ujson must be installed to use UJSONResponse" - return ujson.dumps(content, ensure_ascii=False).encode("utf-8") - - -class ORJSONResponse(JSONResponse): - """ - JSON response using the high-performance orjson library to serialize data to JSON. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/). - """ - - def render(self, content: Any) -> bytes: - assert orjson is not None, "orjson must be installed to use ORJSONResponse" - return orjson.dumps( - content, option=orjson.OPT_NON_STR_KEYS | orjson.OPT_SERIALIZE_NUMPY - ) diff --git a/.env/Lib/site-packages/fastapi/routing.py b/.env/Lib/site-packages/fastapi/routing.py deleted file mode 100644 index c95f624..0000000 --- a/.env/Lib/site-packages/fastapi/routing.py +++ /dev/null @@ -1,4657 +0,0 @@ -import contextlib -import email.message -import functools -import inspect -import json -import types -from collections.abc import ( - AsyncIterator, - Awaitable, - Collection, - Coroutine, - Generator, - Mapping, - Sequence, -) -from contextlib import ( - AbstractAsyncContextManager, - AbstractContextManager, - AsyncExitStack, - asynccontextmanager, -) -from enum import Enum, IntEnum -from typing import ( - Annotated, - Any, - Callable, - Optional, - TypeVar, - Union, -) - -from annotated_doc import Doc -from fastapi import params -from fastapi._compat import ( - ModelField, - Undefined, - annotation_is_pydantic_v1, - lenient_issubclass, -) -from fastapi.datastructures import Default, DefaultPlaceholder -from fastapi.dependencies.models import Dependant -from fastapi.dependencies.utils import ( - _should_embed_body_fields, - get_body_field, - get_dependant, - get_flat_dependant, - get_parameterless_sub_dependant, - get_typed_return_annotation, - solve_dependencies, -) -from fastapi.encoders import jsonable_encoder -from fastapi.exceptions import ( - EndpointContext, - FastAPIError, - PydanticV1NotSupportedError, - RequestValidationError, - ResponseValidationError, - WebSocketRequestValidationError, -) -from fastapi.types import DecoratedCallable, IncEx -from fastapi.utils import ( - create_cloned_field, - create_model_field, - generate_unique_id, - get_value_or_default, - is_body_allowed_for_status_code, -) -from starlette import routing -from starlette._exception_handler import wrap_app_handling_exceptions -from starlette._utils import is_async_callable -from starlette.concurrency import run_in_threadpool -from starlette.exceptions import HTTPException -from starlette.requests import Request -from starlette.responses import JSONResponse, Response -from starlette.routing import ( - BaseRoute, - Match, - compile_path, - get_name, -) -from starlette.routing import Mount as Mount # noqa -from starlette.types import AppType, ASGIApp, Lifespan, Receive, Scope, Send -from starlette.websockets import WebSocket -from typing_extensions import deprecated - - -# Copy of starlette.routing.request_response modified to include the -# dependencies' AsyncExitStack -def request_response( - func: Callable[[Request], Union[Awaitable[Response], Response]], -) -> ASGIApp: - """ - Takes a function or coroutine `func(request) -> response`, - and returns an ASGI application. - """ - f: Callable[[Request], Awaitable[Response]] = ( - func if is_async_callable(func) else functools.partial(run_in_threadpool, func) # type:ignore - ) - - async def app(scope: Scope, receive: Receive, send: Send) -> None: - request = Request(scope, receive, send) - - async def app(scope: Scope, receive: Receive, send: Send) -> None: - # Starts customization - response_awaited = False - async with AsyncExitStack() as request_stack: - scope["fastapi_inner_astack"] = request_stack - async with AsyncExitStack() as function_stack: - scope["fastapi_function_astack"] = function_stack - response = await f(request) - await response(scope, receive, send) - # Continues customization - response_awaited = True - if not response_awaited: - raise FastAPIError( - "Response not awaited. There's a high chance that the " - "application code is raising an exception and a dependency with yield " - "has a block with a bare except, or a block with except Exception, " - "and is not raising the exception again. Read more about it in the " - "docs: https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-with-yield/#dependencies-with-yield-and-except" - ) - - # Same as in Starlette - await wrap_app_handling_exceptions(app, request)(scope, receive, send) - - return app - - -# Copy of starlette.routing.websocket_session modified to include the -# dependencies' AsyncExitStack -def websocket_session( - func: Callable[[WebSocket], Awaitable[None]], -) -> ASGIApp: - """ - Takes a coroutine `func(session)`, and returns an ASGI application. - """ - # assert asyncio.iscoroutinefunction(func), "WebSocket endpoints must be async" - - async def app(scope: Scope, receive: Receive, send: Send) -> None: - session = WebSocket(scope, receive=receive, send=send) - - async def app(scope: Scope, receive: Receive, send: Send) -> None: - async with AsyncExitStack() as request_stack: - scope["fastapi_inner_astack"] = request_stack - async with AsyncExitStack() as function_stack: - scope["fastapi_function_astack"] = function_stack - await func(session) - - # Same as in Starlette - await wrap_app_handling_exceptions(app, session)(scope, receive, send) - - return app - - -_T = TypeVar("_T") - - -# Vendored from starlette.routing to avoid importing private symbols -class _AsyncLiftContextManager(AbstractAsyncContextManager[_T]): - """ - Wraps a synchronous context manager to make it async. - - This is vendored from Starlette to avoid importing private symbols. - """ - - def __init__(self, cm: AbstractContextManager[_T]) -> None: - self._cm = cm - - async def __aenter__(self) -> _T: - return self._cm.__enter__() - - async def __aexit__( - self, - exc_type: Optional[type[BaseException]], - exc_value: Optional[BaseException], - traceback: Optional[types.TracebackType], - ) -> Optional[bool]: - return self._cm.__exit__(exc_type, exc_value, traceback) - - -# Vendored from starlette.routing to avoid importing private symbols -def _wrap_gen_lifespan_context( - lifespan_context: Callable[[Any], Generator[Any, Any, Any]], -) -> Callable[[Any], AbstractAsyncContextManager[Any]]: - """ - Wrap a generator-based lifespan context into an async context manager. - - This is vendored from Starlette to avoid importing private symbols. - """ - cmgr = contextlib.contextmanager(lifespan_context) - - @functools.wraps(cmgr) - def wrapper(app: Any) -> _AsyncLiftContextManager[Any]: - return _AsyncLiftContextManager(cmgr(app)) - - return wrapper - - -def _merge_lifespan_context( - original_context: Lifespan[Any], nested_context: Lifespan[Any] -) -> Lifespan[Any]: - @asynccontextmanager - async def merged_lifespan( - app: AppType, - ) -> AsyncIterator[Optional[Mapping[str, Any]]]: - async with original_context(app) as maybe_original_state: - async with nested_context(app) as maybe_nested_state: - if maybe_nested_state is None and maybe_original_state is None: - yield None # old ASGI compatibility - else: - yield {**(maybe_nested_state or {}), **(maybe_original_state or {})} - - return merged_lifespan # type: ignore[return-value] - - -class _DefaultLifespan: - """ - Default lifespan context manager that runs on_startup and on_shutdown handlers. - - This is a copy of the Starlette _DefaultLifespan class that was removed - in Starlette. FastAPI keeps it to maintain backward compatibility with - on_startup and on_shutdown event handlers. - - Ref: https://github.com/Kludex/starlette/pull/3117 - """ - - def __init__(self, router: "APIRouter") -> None: - self._router = router - - async def __aenter__(self) -> None: - await self._router._startup() - - async def __aexit__(self, *exc_info: object) -> None: - await self._router._shutdown() - - def __call__(self: _T, app: object) -> _T: - return self - - -# Cache for endpoint context to avoid re-extracting on every request -_endpoint_context_cache: dict[int, EndpointContext] = {} - - -def _extract_endpoint_context(func: Any) -> EndpointContext: - """Extract endpoint context with caching to avoid repeated file I/O.""" - func_id = id(func) - - if func_id in _endpoint_context_cache: - return _endpoint_context_cache[func_id] - - try: - ctx: EndpointContext = {} - - if (source_file := inspect.getsourcefile(func)) is not None: - ctx["file"] = source_file - if (line_number := inspect.getsourcelines(func)[1]) is not None: - ctx["line"] = line_number - if (func_name := getattr(func, "__name__", None)) is not None: - ctx["function"] = func_name - except Exception: - ctx = EndpointContext() - - _endpoint_context_cache[func_id] = ctx - return ctx - - -async def serialize_response( - *, - field: Optional[ModelField] = None, - response_content: Any, - include: Optional[IncEx] = None, - exclude: Optional[IncEx] = None, - by_alias: bool = True, - exclude_unset: bool = False, - exclude_defaults: bool = False, - exclude_none: bool = False, - is_coroutine: bool = True, - endpoint_ctx: Optional[EndpointContext] = None, -) -> Any: - if field: - errors = [] - if is_coroutine: - value, errors_ = field.validate(response_content, {}, loc=("response",)) - else: - value, errors_ = await run_in_threadpool( - field.validate, response_content, {}, loc=("response",) - ) - if isinstance(errors_, list): - errors.extend(errors_) - if errors: - ctx = endpoint_ctx or EndpointContext() - raise ResponseValidationError( - errors=errors, - body=response_content, - endpoint_ctx=ctx, - ) - - return field.serialize( - value, - include=include, - exclude=exclude, - by_alias=by_alias, - exclude_unset=exclude_unset, - exclude_defaults=exclude_defaults, - exclude_none=exclude_none, - ) - - else: - return jsonable_encoder(response_content) - - -async def run_endpoint_function( - *, dependant: Dependant, values: dict[str, Any], is_coroutine: bool -) -> Any: - # Only called by get_request_handler. Has been split into its own function to - # facilitate profiling endpoints, since inner functions are harder to profile. - assert dependant.call is not None, "dependant.call must be a function" - - if is_coroutine: - return await dependant.call(**values) - else: - return await run_in_threadpool(dependant.call, **values) - - -def get_request_handler( - dependant: Dependant, - body_field: Optional[ModelField] = None, - status_code: Optional[int] = None, - response_class: Union[type[Response], DefaultPlaceholder] = Default(JSONResponse), - response_field: Optional[ModelField] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - dependency_overrides_provider: Optional[Any] = None, - embed_body_fields: bool = False, -) -> Callable[[Request], Coroutine[Any, Any, Response]]: - assert dependant.call is not None, "dependant.call must be a function" - is_coroutine = dependant.is_coroutine_callable - is_body_form = body_field and isinstance(body_field.field_info, params.Form) - if isinstance(response_class, DefaultPlaceholder): - actual_response_class: type[Response] = response_class.value - else: - actual_response_class = response_class - - async def app(request: Request) -> Response: - response: Union[Response, None] = None - file_stack = request.scope.get("fastapi_middleware_astack") - assert isinstance(file_stack, AsyncExitStack), ( - "fastapi_middleware_astack not found in request scope" - ) - - # Extract endpoint context for error messages - endpoint_ctx = ( - _extract_endpoint_context(dependant.call) - if dependant.call - else EndpointContext() - ) - - if dependant.path: - # For mounted sub-apps, include the mount path prefix - mount_path = request.scope.get("root_path", "").rstrip("/") - endpoint_ctx["path"] = f"{request.method} {mount_path}{dependant.path}" - - # Read body and auto-close files - try: - body: Any = None - if body_field: - if is_body_form: - body = await request.form() - file_stack.push_async_callback(body.close) - else: - body_bytes = await request.body() - if body_bytes: - json_body: Any = Undefined - content_type_value = request.headers.get("content-type") - if not content_type_value: - json_body = await request.json() - else: - message = email.message.Message() - message["content-type"] = content_type_value - if message.get_content_maintype() == "application": - subtype = message.get_content_subtype() - if subtype == "json" or subtype.endswith("+json"): - json_body = await request.json() - if json_body != Undefined: - body = json_body - else: - body = body_bytes - except json.JSONDecodeError as e: - validation_error = RequestValidationError( - [ - { - "type": "json_invalid", - "loc": ("body", e.pos), - "msg": "JSON decode error", - "input": {}, - "ctx": {"error": e.msg}, - } - ], - body=e.doc, - endpoint_ctx=endpoint_ctx, - ) - raise validation_error from e - except HTTPException: - # If a middleware raises an HTTPException, it should be raised again - raise - except Exception as e: - http_error = HTTPException( - status_code=400, detail="There was an error parsing the body" - ) - raise http_error from e - - # Solve dependencies and run path operation function, auto-closing dependencies - errors: list[Any] = [] - async_exit_stack = request.scope.get("fastapi_inner_astack") - assert isinstance(async_exit_stack, AsyncExitStack), ( - "fastapi_inner_astack not found in request scope" - ) - solved_result = await solve_dependencies( - request=request, - dependant=dependant, - body=body, - dependency_overrides_provider=dependency_overrides_provider, - async_exit_stack=async_exit_stack, - embed_body_fields=embed_body_fields, - ) - errors = solved_result.errors - if not errors: - raw_response = await run_endpoint_function( - dependant=dependant, - values=solved_result.values, - is_coroutine=is_coroutine, - ) - if isinstance(raw_response, Response): - if raw_response.background is None: - raw_response.background = solved_result.background_tasks - response = raw_response - else: - response_args: dict[str, Any] = { - "background": solved_result.background_tasks - } - # If status_code was set, use it, otherwise use the default from the - # response class, in the case of redirect it's 307 - current_status_code = ( - status_code if status_code else solved_result.response.status_code - ) - if current_status_code is not None: - response_args["status_code"] = current_status_code - if solved_result.response.status_code: - response_args["status_code"] = solved_result.response.status_code - content = await serialize_response( - field=response_field, - response_content=raw_response, - include=response_model_include, - exclude=response_model_exclude, - by_alias=response_model_by_alias, - exclude_unset=response_model_exclude_unset, - exclude_defaults=response_model_exclude_defaults, - exclude_none=response_model_exclude_none, - is_coroutine=is_coroutine, - endpoint_ctx=endpoint_ctx, - ) - response = actual_response_class(content, **response_args) - if not is_body_allowed_for_status_code(response.status_code): - response.body = b"" - response.headers.raw.extend(solved_result.response.headers.raw) - if errors: - validation_error = RequestValidationError( - errors, body=body, endpoint_ctx=endpoint_ctx - ) - raise validation_error - - # Return response - assert response - return response - - return app - - -def get_websocket_app( - dependant: Dependant, - dependency_overrides_provider: Optional[Any] = None, - embed_body_fields: bool = False, -) -> Callable[[WebSocket], Coroutine[Any, Any, Any]]: - async def app(websocket: WebSocket) -> None: - endpoint_ctx = ( - _extract_endpoint_context(dependant.call) - if dependant.call - else EndpointContext() - ) - if dependant.path: - # For mounted sub-apps, include the mount path prefix - mount_path = websocket.scope.get("root_path", "").rstrip("/") - endpoint_ctx["path"] = f"WS {mount_path}{dependant.path}" - async_exit_stack = websocket.scope.get("fastapi_inner_astack") - assert isinstance(async_exit_stack, AsyncExitStack), ( - "fastapi_inner_astack not found in request scope" - ) - solved_result = await solve_dependencies( - request=websocket, - dependant=dependant, - dependency_overrides_provider=dependency_overrides_provider, - async_exit_stack=async_exit_stack, - embed_body_fields=embed_body_fields, - ) - if solved_result.errors: - raise WebSocketRequestValidationError( - solved_result.errors, - endpoint_ctx=endpoint_ctx, - ) - assert dependant.call is not None, "dependant.call must be a function" - await dependant.call(**solved_result.values) - - return app - - -class APIWebSocketRoute(routing.WebSocketRoute): - def __init__( - self, - path: str, - endpoint: Callable[..., Any], - *, - name: Optional[str] = None, - dependencies: Optional[Sequence[params.Depends]] = None, - dependency_overrides_provider: Optional[Any] = None, - ) -> None: - self.path = path - self.endpoint = endpoint - self.name = get_name(endpoint) if name is None else name - self.dependencies = list(dependencies or []) - self.path_regex, self.path_format, self.param_convertors = compile_path(path) - self.dependant = get_dependant( - path=self.path_format, call=self.endpoint, scope="function" - ) - for depends in self.dependencies[::-1]: - self.dependant.dependencies.insert( - 0, - get_parameterless_sub_dependant(depends=depends, path=self.path_format), - ) - self._flat_dependant = get_flat_dependant(self.dependant) - self._embed_body_fields = _should_embed_body_fields( - self._flat_dependant.body_params - ) - self.app = websocket_session( - get_websocket_app( - dependant=self.dependant, - dependency_overrides_provider=dependency_overrides_provider, - embed_body_fields=self._embed_body_fields, - ) - ) - - def matches(self, scope: Scope) -> tuple[Match, Scope]: - match, child_scope = super().matches(scope) - if match != Match.NONE: - child_scope["route"] = self - return match, child_scope - - -class APIRoute(routing.Route): - def __init__( - self, - path: str, - endpoint: Callable[..., Any], - *, - response_model: Any = Default(None), - status_code: Optional[int] = None, - tags: Optional[list[Union[str, Enum]]] = None, - dependencies: Optional[Sequence[params.Depends]] = None, - summary: Optional[str] = None, - description: Optional[str] = None, - response_description: str = "Successful Response", - responses: Optional[dict[Union[int, str], dict[str, Any]]] = None, - deprecated: Optional[bool] = None, - name: Optional[str] = None, - methods: Optional[Union[set[str], list[str]]] = None, - operation_id: Optional[str] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - include_in_schema: bool = True, - response_class: Union[type[Response], DefaultPlaceholder] = Default( - JSONResponse - ), - dependency_overrides_provider: Optional[Any] = None, - callbacks: Optional[list[BaseRoute]] = None, - openapi_extra: Optional[dict[str, Any]] = None, - generate_unique_id_function: Union[ - Callable[["APIRoute"], str], DefaultPlaceholder - ] = Default(generate_unique_id), - ) -> None: - self.path = path - self.endpoint = endpoint - if isinstance(response_model, DefaultPlaceholder): - return_annotation = get_typed_return_annotation(endpoint) - if lenient_issubclass(return_annotation, Response): - response_model = None - else: - response_model = return_annotation - self.response_model = response_model - self.summary = summary - self.response_description = response_description - self.deprecated = deprecated - self.operation_id = operation_id - self.response_model_include = response_model_include - self.response_model_exclude = response_model_exclude - self.response_model_by_alias = response_model_by_alias - self.response_model_exclude_unset = response_model_exclude_unset - self.response_model_exclude_defaults = response_model_exclude_defaults - self.response_model_exclude_none = response_model_exclude_none - self.include_in_schema = include_in_schema - self.response_class = response_class - self.dependency_overrides_provider = dependency_overrides_provider - self.callbacks = callbacks - self.openapi_extra = openapi_extra - self.generate_unique_id_function = generate_unique_id_function - self.tags = tags or [] - self.responses = responses or {} - self.name = get_name(endpoint) if name is None else name - self.path_regex, self.path_format, self.param_convertors = compile_path(path) - if methods is None: - methods = ["GET"] - self.methods: set[str] = {method.upper() for method in methods} - if isinstance(generate_unique_id_function, DefaultPlaceholder): - current_generate_unique_id: Callable[[APIRoute], str] = ( - generate_unique_id_function.value - ) - else: - current_generate_unique_id = generate_unique_id_function - self.unique_id = self.operation_id or current_generate_unique_id(self) - # normalize enums e.g. http.HTTPStatus - if isinstance(status_code, IntEnum): - status_code = int(status_code) - self.status_code = status_code - if self.response_model: - assert is_body_allowed_for_status_code(status_code), ( - f"Status code {status_code} must not have a response body" - ) - response_name = "Response_" + self.unique_id - if annotation_is_pydantic_v1(self.response_model): - raise PydanticV1NotSupportedError( - "pydantic.v1 models are no longer supported by FastAPI." - f" Please update the response model {self.response_model!r}." - ) - self.response_field = create_model_field( - name=response_name, - type_=self.response_model, - mode="serialization", - ) - # Create a clone of the field, so that a Pydantic submodel is not returned - # as is just because it's an instance of a subclass of a more limited class - # e.g. UserInDB (containing hashed_password) could be a subclass of User - # that doesn't have the hashed_password. But because it's a subclass, it - # would pass the validation and be returned as is. - # By being a new field, no inheritance will be passed as is. A new model - # will always be created. - # TODO: remove when deprecating Pydantic v1 - self.secure_cloned_response_field: Optional[ModelField] = ( - create_cloned_field(self.response_field) - ) - else: - self.response_field = None # type: ignore - self.secure_cloned_response_field = None - self.dependencies = list(dependencies or []) - self.description = description or inspect.cleandoc(self.endpoint.__doc__ or "") - # if a "form feed" character (page break) is found in the description text, - # truncate description text to the content preceding the first "form feed" - self.description = self.description.split("\f")[0].strip() - response_fields = {} - for additional_status_code, response in self.responses.items(): - assert isinstance(response, dict), "An additional response must be a dict" - model = response.get("model") - if model: - assert is_body_allowed_for_status_code(additional_status_code), ( - f"Status code {additional_status_code} must not have a response body" - ) - response_name = f"Response_{additional_status_code}_{self.unique_id}" - if annotation_is_pydantic_v1(model): - raise PydanticV1NotSupportedError( - "pydantic.v1 models are no longer supported by FastAPI." - f" In responses={{}}, please update {model}." - ) - response_field = create_model_field( - name=response_name, type_=model, mode="serialization" - ) - response_fields[additional_status_code] = response_field - if response_fields: - self.response_fields: dict[Union[int, str], ModelField] = response_fields - else: - self.response_fields = {} - - assert callable(endpoint), "An endpoint must be a callable" - self.dependant = get_dependant( - path=self.path_format, call=self.endpoint, scope="function" - ) - for depends in self.dependencies[::-1]: - self.dependant.dependencies.insert( - 0, - get_parameterless_sub_dependant(depends=depends, path=self.path_format), - ) - self._flat_dependant = get_flat_dependant(self.dependant) - self._embed_body_fields = _should_embed_body_fields( - self._flat_dependant.body_params - ) - self.body_field = get_body_field( - flat_dependant=self._flat_dependant, - name=self.unique_id, - embed_body_fields=self._embed_body_fields, - ) - self.app = request_response(self.get_route_handler()) - - def get_route_handler(self) -> Callable[[Request], Coroutine[Any, Any, Response]]: - return get_request_handler( - dependant=self.dependant, - body_field=self.body_field, - status_code=self.status_code, - response_class=self.response_class, - response_field=self.secure_cloned_response_field, - response_model_include=self.response_model_include, - response_model_exclude=self.response_model_exclude, - response_model_by_alias=self.response_model_by_alias, - response_model_exclude_unset=self.response_model_exclude_unset, - response_model_exclude_defaults=self.response_model_exclude_defaults, - response_model_exclude_none=self.response_model_exclude_none, - dependency_overrides_provider=self.dependency_overrides_provider, - embed_body_fields=self._embed_body_fields, - ) - - def matches(self, scope: Scope) -> tuple[Match, Scope]: - match, child_scope = super().matches(scope) - if match != Match.NONE: - child_scope["route"] = self - return match, child_scope - - -class APIRouter(routing.Router): - """ - `APIRouter` class, used to group *path operations*, for example to structure - an app in multiple files. It would then be included in the `FastAPI` app, or - in another `APIRouter` (ultimately included in the app). - - Read more about it in the - [FastAPI docs for Bigger Applications - Multiple Files](https://fastapi.tiangolo.com/tutorial/bigger-applications/). - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - - app = FastAPI() - router = APIRouter() - - - @router.get("/users/", tags=["users"]) - async def read_users(): - return [{"username": "Rick"}, {"username": "Morty"}] - - - app.include_router(router) - ``` - """ - - def __init__( - self, - *, - prefix: Annotated[str, Doc("An optional path prefix for the router.")] = "", - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to all the *path operations* in this - router. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to all the - *path operations* in this router. - - Read more about it in the - [FastAPI docs for Bigger Applications - Multiple Files](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - default_response_class: Annotated[ - type[Response], - Doc( - """ - The default response class to be used. - - Read more in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#default-response-class). - """ - ), - ] = Default(JSONResponse), - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses to be shown in OpenAPI. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Additional Responses in OpenAPI](https://fastapi.tiangolo.com/advanced/additional-responses/). - - And in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - OpenAPI callbacks that should apply to all *path operations* in this - router. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - routes: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - **Note**: you probably shouldn't use this parameter, it is inherited - from Starlette and supported for compatibility. - - --- - - A list of routes to serve incoming HTTP and WebSocket requests. - """ - ), - deprecated( - """ - You normally wouldn't use this parameter with FastAPI, it is inherited - from Starlette and supported for compatibility. - - In FastAPI, you normally would use the *path operation methods*, - like `router.get()`, `router.post()`, etc. - """ - ), - ] = None, - redirect_slashes: Annotated[ - bool, - Doc( - """ - Whether to detect and redirect slashes in URLs when the client doesn't - use the same format. - """ - ), - ] = True, - default: Annotated[ - Optional[ASGIApp], - Doc( - """ - Default function handler for this router. Used to handle - 404 Not Found errors. - """ - ), - ] = None, - dependency_overrides_provider: Annotated[ - Optional[Any], - Doc( - """ - Only used internally by FastAPI to handle dependency overrides. - - You shouldn't need to use it. It normally points to the `FastAPI` app - object. - """ - ), - ] = None, - route_class: Annotated[ - type[APIRoute], - Doc( - """ - Custom route (*path operation*) class to be used by this router. - - Read more about it in the - [FastAPI docs for Custom Request and APIRoute class](https://fastapi.tiangolo.com/how-to/custom-request-and-route/#custom-apiroute-class-in-a-router). - """ - ), - ] = APIRoute, - on_startup: Annotated[ - Optional[Sequence[Callable[[], Any]]], - Doc( - """ - A list of startup event handler functions. - - You should instead use the `lifespan` handlers. - - Read more in the [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - on_shutdown: Annotated[ - Optional[Sequence[Callable[[], Any]]], - Doc( - """ - A list of shutdown event handler functions. - - You should instead use the `lifespan` handlers. - - Read more in the - [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - # the generic to Lifespan[AppType] is the type of the top level application - # which the router cannot know statically, so we use typing.Any - lifespan: Annotated[ - Optional[Lifespan[Any]], - Doc( - """ - A `Lifespan` context manager handler. This replaces `startup` and - `shutdown` functions with a single context manager. - - Read more in the - [FastAPI docs for `lifespan`](https://fastapi.tiangolo.com/advanced/events/). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark all *path operations* in this router as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - To include (or not) all the *path operations* in this router in the - generated OpenAPI. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> None: - # Handle on_startup/on_shutdown locally since Starlette removed support - # Ref: https://github.com/Kludex/starlette/pull/3117 - # TODO: deprecate this once the lifespan (or alternative) interface is improved - self.on_startup: list[Callable[[], Any]] = ( - [] if on_startup is None else list(on_startup) - ) - self.on_shutdown: list[Callable[[], Any]] = ( - [] if on_shutdown is None else list(on_shutdown) - ) - - # Determine the lifespan context to use - if lifespan is None: - # Use the default lifespan that runs on_startup/on_shutdown handlers - lifespan_context: Lifespan[Any] = _DefaultLifespan(self) - elif inspect.isasyncgenfunction(lifespan): - lifespan_context = asynccontextmanager(lifespan) - elif inspect.isgeneratorfunction(lifespan): - lifespan_context = _wrap_gen_lifespan_context(lifespan) - else: - lifespan_context = lifespan - self.lifespan_context = lifespan_context - - super().__init__( - routes=routes, - redirect_slashes=redirect_slashes, - default=default, - lifespan=lifespan_context, - ) - if prefix: - assert prefix.startswith("/"), "A path prefix must start with '/'" - assert not prefix.endswith("/"), ( - "A path prefix must not end with '/', as the routes will start with '/'" - ) - self.prefix = prefix - self.tags: list[Union[str, Enum]] = tags or [] - self.dependencies = list(dependencies or []) - self.deprecated = deprecated - self.include_in_schema = include_in_schema - self.responses = responses or {} - self.callbacks = callbacks or [] - self.dependency_overrides_provider = dependency_overrides_provider - self.route_class = route_class - self.default_response_class = default_response_class - self.generate_unique_id_function = generate_unique_id_function - - def route( - self, - path: str, - methods: Optional[Collection[str]] = None, - name: Optional[str] = None, - include_in_schema: bool = True, - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_route( - path, - func, - methods=methods, - name=name, - include_in_schema=include_in_schema, - ) - return func - - return decorator - - def add_api_route( - self, - path: str, - endpoint: Callable[..., Any], - *, - response_model: Any = Default(None), - status_code: Optional[int] = None, - tags: Optional[list[Union[str, Enum]]] = None, - dependencies: Optional[Sequence[params.Depends]] = None, - summary: Optional[str] = None, - description: Optional[str] = None, - response_description: str = "Successful Response", - responses: Optional[dict[Union[int, str], dict[str, Any]]] = None, - deprecated: Optional[bool] = None, - methods: Optional[Union[set[str], list[str]]] = None, - operation_id: Optional[str] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - include_in_schema: bool = True, - response_class: Union[type[Response], DefaultPlaceholder] = Default( - JSONResponse - ), - name: Optional[str] = None, - route_class_override: Optional[type[APIRoute]] = None, - callbacks: Optional[list[BaseRoute]] = None, - openapi_extra: Optional[dict[str, Any]] = None, - generate_unique_id_function: Union[ - Callable[[APIRoute], str], DefaultPlaceholder - ] = Default(generate_unique_id), - ) -> None: - route_class = route_class_override or self.route_class - responses = responses or {} - combined_responses = {**self.responses, **responses} - current_response_class = get_value_or_default( - response_class, self.default_response_class - ) - current_tags = self.tags.copy() - if tags: - current_tags.extend(tags) - current_dependencies = self.dependencies.copy() - if dependencies: - current_dependencies.extend(dependencies) - current_callbacks = self.callbacks.copy() - if callbacks: - current_callbacks.extend(callbacks) - current_generate_unique_id = get_value_or_default( - generate_unique_id_function, self.generate_unique_id_function - ) - route = route_class( - self.prefix + path, - endpoint=endpoint, - response_model=response_model, - status_code=status_code, - tags=current_tags, - dependencies=current_dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=combined_responses, - deprecated=deprecated or self.deprecated, - methods=methods, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema and self.include_in_schema, - response_class=current_response_class, - name=name, - dependency_overrides_provider=self.dependency_overrides_provider, - callbacks=current_callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=current_generate_unique_id, - ) - self.routes.append(route) - - def api_route( - self, - path: str, - *, - response_model: Any = Default(None), - status_code: Optional[int] = None, - tags: Optional[list[Union[str, Enum]]] = None, - dependencies: Optional[Sequence[params.Depends]] = None, - summary: Optional[str] = None, - description: Optional[str] = None, - response_description: str = "Successful Response", - responses: Optional[dict[Union[int, str], dict[str, Any]]] = None, - deprecated: Optional[bool] = None, - methods: Optional[list[str]] = None, - operation_id: Optional[str] = None, - response_model_include: Optional[IncEx] = None, - response_model_exclude: Optional[IncEx] = None, - response_model_by_alias: bool = True, - response_model_exclude_unset: bool = False, - response_model_exclude_defaults: bool = False, - response_model_exclude_none: bool = False, - include_in_schema: bool = True, - response_class: type[Response] = Default(JSONResponse), - name: Optional[str] = None, - callbacks: Optional[list[BaseRoute]] = None, - openapi_extra: Optional[dict[str, Any]] = None, - generate_unique_id_function: Callable[[APIRoute], str] = Default( - generate_unique_id - ), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_api_route( - path, - func, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=methods, - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - return func - - return decorator - - def add_api_websocket_route( - self, - path: str, - endpoint: Callable[..., Any], - name: Optional[str] = None, - *, - dependencies: Optional[Sequence[params.Depends]] = None, - ) -> None: - current_dependencies = self.dependencies.copy() - if dependencies: - current_dependencies.extend(dependencies) - - route = APIWebSocketRoute( - self.prefix + path, - endpoint=endpoint, - name=name, - dependencies=current_dependencies, - dependency_overrides_provider=self.dependency_overrides_provider, - ) - self.routes.append(route) - - def websocket( - self, - path: Annotated[ - str, - Doc( - """ - WebSocket path. - """ - ), - ], - name: Annotated[ - Optional[str], - Doc( - """ - A name for the WebSocket. Only used internally. - """ - ), - ] = None, - *, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be used for this - WebSocket. - - Read more about it in the - [FastAPI docs for WebSockets](https://fastapi.tiangolo.com/advanced/websockets/). - """ - ), - ] = None, - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Decorate a WebSocket function. - - Read more about it in the - [FastAPI docs for WebSockets](https://fastapi.tiangolo.com/advanced/websockets/). - - **Example** - - ## Example - - ```python - from fastapi import APIRouter, FastAPI, WebSocket - - app = FastAPI() - router = APIRouter() - - @router.websocket("/ws") - async def websocket_endpoint(websocket: WebSocket): - await websocket.accept() - while True: - data = await websocket.receive_text() - await websocket.send_text(f"Message text was: {data}") - - app.include_router(router) - ``` - """ - - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_api_websocket_route( - path, func, name=name, dependencies=dependencies - ) - return func - - return decorator - - def websocket_route( - self, path: str, name: Union[str, None] = None - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_websocket_route(path, func, name=name) - return func - - return decorator - - def include_router( - self, - router: Annotated["APIRouter", Doc("The `APIRouter` to include.")], - *, - prefix: Annotated[str, Doc("An optional path prefix for the router.")] = "", - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to all the *path operations* in this - router. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to all the - *path operations* in this router. - - Read more about it in the - [FastAPI docs for Bigger Applications - Multiple Files](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - default_response_class: Annotated[ - type[Response], - Doc( - """ - The default response class to be used. - - Read more in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#default-response-class). - """ - ), - ] = Default(JSONResponse), - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses to be shown in OpenAPI. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Additional Responses in OpenAPI](https://fastapi.tiangolo.com/advanced/additional-responses/). - - And in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/#include-an-apirouter-with-a-custom-prefix-tags-responses-and-dependencies). - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - OpenAPI callbacks that should apply to all *path operations* in this - router. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark all *path operations* in this router as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include (or not) all the *path operations* in this router in the - generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = True, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> None: - """ - Include another `APIRouter` in the same current `APIRouter`. - - Read more about it in the - [FastAPI docs for Bigger Applications](https://fastapi.tiangolo.com/tutorial/bigger-applications/). - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - - app = FastAPI() - internal_router = APIRouter() - users_router = APIRouter() - - @users_router.get("/users/") - def read_users(): - return [{"name": "Rick"}, {"name": "Morty"}] - - internal_router.include_router(users_router) - app.include_router(internal_router) - ``` - """ - if prefix: - assert prefix.startswith("/"), "A path prefix must start with '/'" - assert not prefix.endswith("/"), ( - "A path prefix must not end with '/', as the routes will start with '/'" - ) - else: - for r in router.routes: - path = getattr(r, "path") # noqa: B009 - name = getattr(r, "name", "unknown") - if path is not None and not path: - raise FastAPIError( - f"Prefix and path cannot be both empty (path operation: {name})" - ) - if responses is None: - responses = {} - for route in router.routes: - if isinstance(route, APIRoute): - combined_responses = {**responses, **route.responses} - use_response_class = get_value_or_default( - route.response_class, - router.default_response_class, - default_response_class, - self.default_response_class, - ) - current_tags = [] - if tags: - current_tags.extend(tags) - if route.tags: - current_tags.extend(route.tags) - current_dependencies: list[params.Depends] = [] - if dependencies: - current_dependencies.extend(dependencies) - if route.dependencies: - current_dependencies.extend(route.dependencies) - current_callbacks = [] - if callbacks: - current_callbacks.extend(callbacks) - if route.callbacks: - current_callbacks.extend(route.callbacks) - current_generate_unique_id = get_value_or_default( - route.generate_unique_id_function, - router.generate_unique_id_function, - generate_unique_id_function, - self.generate_unique_id_function, - ) - self.add_api_route( - prefix + route.path, - route.endpoint, - response_model=route.response_model, - status_code=route.status_code, - tags=current_tags, - dependencies=current_dependencies, - summary=route.summary, - description=route.description, - response_description=route.response_description, - responses=combined_responses, - deprecated=route.deprecated or deprecated or self.deprecated, - methods=route.methods, - operation_id=route.operation_id, - response_model_include=route.response_model_include, - response_model_exclude=route.response_model_exclude, - response_model_by_alias=route.response_model_by_alias, - response_model_exclude_unset=route.response_model_exclude_unset, - response_model_exclude_defaults=route.response_model_exclude_defaults, - response_model_exclude_none=route.response_model_exclude_none, - include_in_schema=route.include_in_schema - and self.include_in_schema - and include_in_schema, - response_class=use_response_class, - name=route.name, - route_class_override=type(route), - callbacks=current_callbacks, - openapi_extra=route.openapi_extra, - generate_unique_id_function=current_generate_unique_id, - ) - elif isinstance(route, routing.Route): - methods = list(route.methods or []) - self.add_route( - prefix + route.path, - route.endpoint, - methods=methods, - include_in_schema=route.include_in_schema, - name=route.name, - ) - elif isinstance(route, APIWebSocketRoute): - current_dependencies = [] - if dependencies: - current_dependencies.extend(dependencies) - if route.dependencies: - current_dependencies.extend(route.dependencies) - self.add_api_websocket_route( - prefix + route.path, - route.endpoint, - dependencies=current_dependencies, - name=route.name, - ) - elif isinstance(route, routing.WebSocketRoute): - self.add_websocket_route( - prefix + route.path, route.endpoint, name=route.name - ) - for handler in router.on_startup: - self.add_event_handler("startup", handler) - for handler in router.on_shutdown: - self.add_event_handler("shutdown", handler) - self.lifespan_context = _merge_lifespan_context( - self.lifespan_context, - router.lifespan_context, - ) - - def get( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP GET operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - - app = FastAPI() - router = APIRouter() - - @router.get("/items/") - def read_items(): - return [{"name": "Empanada"}, {"name": "Arepa"}] - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["GET"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def put( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP PUT operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - router = APIRouter() - - @router.put("/items/{item_id}") - def replace_item(item_id: str, item: Item): - return {"message": "Item replaced", "id": item_id} - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["PUT"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def post( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP POST operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - router = APIRouter() - - @router.post("/items/") - def create_item(item: Item): - return {"message": "Item created"} - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["POST"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def delete( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP DELETE operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - - app = FastAPI() - router = APIRouter() - - @router.delete("/items/{item_id}") - def delete_item(item_id: str): - return {"message": "Item deleted"} - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["DELETE"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def options( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP OPTIONS operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - - app = FastAPI() - router = APIRouter() - - @router.options("/items/") - def get_item_options(): - return {"additions": ["Aji", "Guacamole"]} - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["OPTIONS"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def head( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP HEAD operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - router = APIRouter() - - @router.head("/items/", status_code=204) - def get_items_headers(response: Response): - response.headers["X-Cat-Dog"] = "Alone in the world" - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["HEAD"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def patch( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP PATCH operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - router = APIRouter() - - @router.patch("/items/") - def update_item(item: Item): - return {"message": "Item updated in place"} - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["PATCH"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - def trace( - self, - path: Annotated[ - str, - Doc( - """ - The URL path to be used for this *path operation*. - - For example, in `http://example.com/items`, the path is `/items`. - """ - ), - ], - *, - response_model: Annotated[ - Any, - Doc( - """ - The type to use for the response. - - It could be any valid Pydantic *field* type. So, it doesn't have to - be a Pydantic model, it could be other things, like a `list`, `dict`, - etc. - - It will be used for: - - * Documentation: the generated OpenAPI (and the UI at `/docs`) will - show it as the response (JSON Schema). - * Serialization: you could return an arbitrary object and the - `response_model` would be used to serialize that object into the - corresponding JSON. - * Filtering: the JSON sent to the client will only contain the data - (fields) defined in the `response_model`. If you returned an object - that contains an attribute `password` but the `response_model` does - not include that field, the JSON sent to the client would not have - that `password`. - * Validation: whatever you return will be serialized with the - `response_model`, converting any data as necessary to generate the - corresponding JSON. But if the data in the object returned is not - valid, that would mean a violation of the contract with the client, - so it's an error from the API developer. So, FastAPI will raise an - error and return a 500 error code (Internal Server Error). - - Read more about it in the - [FastAPI docs for Response Model](https://fastapi.tiangolo.com/tutorial/response-model/). - """ - ), - ] = Default(None), - status_code: Annotated[ - Optional[int], - Doc( - """ - The default status code to be used for the response. - - You could override the status code by returning a response directly. - - Read more about it in the - [FastAPI docs for Response Status Code](https://fastapi.tiangolo.com/tutorial/response-status-code/). - """ - ), - ] = None, - tags: Annotated[ - Optional[list[Union[str, Enum]]], - Doc( - """ - A list of tags to be applied to the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/#tags). - """ - ), - ] = None, - dependencies: Annotated[ - Optional[Sequence[params.Depends]], - Doc( - """ - A list of dependencies (using `Depends()`) to be applied to the - *path operation*. - - Read more about it in the - [FastAPI docs for Dependencies in path operation decorators](https://fastapi.tiangolo.com/tutorial/dependencies/dependencies-in-path-operation-decorators/). - """ - ), - ] = None, - summary: Annotated[ - Optional[str], - Doc( - """ - A summary for the *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - A description for the *path operation*. - - If not provided, it will be extracted automatically from the docstring - of the *path operation function*. - - It can contain Markdown. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Path Operation Configuration](https://fastapi.tiangolo.com/tutorial/path-operation-configuration/). - """ - ), - ] = None, - response_description: Annotated[ - str, - Doc( - """ - The description for the default response. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = "Successful Response", - responses: Annotated[ - Optional[dict[Union[int, str], dict[str, Any]]], - Doc( - """ - Additional responses that could be returned by this *path operation*. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - deprecated: Annotated[ - Optional[bool], - Doc( - """ - Mark this *path operation* as deprecated. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - operation_id: Annotated[ - Optional[str], - Doc( - """ - Custom operation ID to be used by this *path operation*. - - By default, it is generated automatically. - - If you provide a custom operation ID, you need to make sure it is - unique for the whole API. - - You can customize the - operation ID generation with the parameter - `generate_unique_id_function` in the `FastAPI` class. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = None, - response_model_include: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to include only certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_exclude: Annotated[ - Optional[IncEx], - Doc( - """ - Configuration passed to Pydantic to exclude certain fields in the - response data. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = None, - response_model_by_alias: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response model - should be serialized by alias when an alias is used. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_include-and-response_model_exclude). - """ - ), - ] = True, - response_model_exclude_unset: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that were not set and - have their default values. This is different from - `response_model_exclude_defaults` in that if the fields are set, - they will be included in the response, even if the value is the same - as the default. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_defaults: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data - should have all the fields, including the ones that have the same value - as the default. This is different from `response_model_exclude_unset` - in that if the fields are set but contain the same default values, - they will be excluded from the response. - - When `True`, default values are omitted from the response. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#use-the-response_model_exclude_unset-parameter). - """ - ), - ] = False, - response_model_exclude_none: Annotated[ - bool, - Doc( - """ - Configuration passed to Pydantic to define if the response data should - exclude fields set to `None`. - - This is much simpler (less smart) than `response_model_exclude_unset` - and `response_model_exclude_defaults`. You probably want to use one of - those two instead of this one, as those allow returning `None` values - when it makes sense. - - Read more about it in the - [FastAPI docs for Response Model - Return Type](https://fastapi.tiangolo.com/tutorial/response-model/#response_model_exclude_none). - """ - ), - ] = False, - include_in_schema: Annotated[ - bool, - Doc( - """ - Include this *path operation* in the generated OpenAPI schema. - - This affects the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for Query Parameters and String Validations](https://fastapi.tiangolo.com/tutorial/query-params-str-validations/#exclude-parameters-from-openapi). - """ - ), - ] = True, - response_class: Annotated[ - type[Response], - Doc( - """ - Response class to be used for this *path operation*. - - This will not be used if you return a response directly. - - Read more about it in the - [FastAPI docs for Custom Response - HTML, Stream, File, others](https://fastapi.tiangolo.com/advanced/custom-response/#redirectresponse). - """ - ), - ] = Default(JSONResponse), - name: Annotated[ - Optional[str], - Doc( - """ - Name for this *path operation*. Only used internally. - """ - ), - ] = None, - callbacks: Annotated[ - Optional[list[BaseRoute]], - Doc( - """ - List of *path operations* that will be used as OpenAPI callbacks. - - This is only for OpenAPI documentation, the callbacks won't be used - directly. - - It will be added to the generated OpenAPI (e.g. visible at `/docs`). - - Read more about it in the - [FastAPI docs for OpenAPI Callbacks](https://fastapi.tiangolo.com/advanced/openapi-callbacks/). - """ - ), - ] = None, - openapi_extra: Annotated[ - Optional[dict[str, Any]], - Doc( - """ - Extra metadata to be included in the OpenAPI schema for this *path - operation*. - - Read more about it in the - [FastAPI docs for Path Operation Advanced Configuration](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#custom-openapi-path-operation-schema). - """ - ), - ] = None, - generate_unique_id_function: Annotated[ - Callable[[APIRoute], str], - Doc( - """ - Customize the function used to generate unique IDs for the *path - operations* shown in the generated OpenAPI. - - This is particularly useful when automatically generating clients or - SDKs for your API. - - Read more about it in the - [FastAPI docs about how to Generate Clients](https://fastapi.tiangolo.com/advanced/generate-clients/#custom-generate-unique-id-function). - """ - ), - ] = Default(generate_unique_id), - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add a *path operation* using an HTTP TRACE operation. - - ## Example - - ```python - from fastapi import APIRouter, FastAPI - from pydantic import BaseModel - - class Item(BaseModel): - name: str - description: str | None = None - - app = FastAPI() - router = APIRouter() - - @router.trace("/items/{item_id}") - def trace_item(item_id: str): - return None - - app.include_router(router) - ``` - """ - return self.api_route( - path=path, - response_model=response_model, - status_code=status_code, - tags=tags, - dependencies=dependencies, - summary=summary, - description=description, - response_description=response_description, - responses=responses, - deprecated=deprecated, - methods=["TRACE"], - operation_id=operation_id, - response_model_include=response_model_include, - response_model_exclude=response_model_exclude, - response_model_by_alias=response_model_by_alias, - response_model_exclude_unset=response_model_exclude_unset, - response_model_exclude_defaults=response_model_exclude_defaults, - response_model_exclude_none=response_model_exclude_none, - include_in_schema=include_in_schema, - response_class=response_class, - name=name, - callbacks=callbacks, - openapi_extra=openapi_extra, - generate_unique_id_function=generate_unique_id_function, - ) - - # TODO: remove this once the lifespan (or alternative) interface is improved - async def _startup(self) -> None: - """ - Run any `.on_startup` event handlers. - - This method is kept for backward compatibility after Starlette removed - support for on_startup/on_shutdown handlers. - - Ref: https://github.com/Kludex/starlette/pull/3117 - """ - for handler in self.on_startup: - if is_async_callable(handler): - await handler() - else: - handler() - - # TODO: remove this once the lifespan (or alternative) interface is improved - async def _shutdown(self) -> None: - """ - Run any `.on_shutdown` event handlers. - - This method is kept for backward compatibility after Starlette removed - support for on_startup/on_shutdown handlers. - - Ref: https://github.com/Kludex/starlette/pull/3117 - """ - for handler in self.on_shutdown: - if is_async_callable(handler): - await handler() - else: - handler() - - # TODO: remove this once the lifespan (or alternative) interface is improved - def add_event_handler( - self, - event_type: str, - func: Callable[[], Any], - ) -> None: - """ - Add an event handler function for startup or shutdown. - - This method is kept for backward compatibility after Starlette removed - support for on_startup/on_shutdown handlers. - - Ref: https://github.com/Kludex/starlette/pull/3117 - """ - assert event_type in ("startup", "shutdown") - if event_type == "startup": - self.on_startup.append(func) - else: - self.on_shutdown.append(func) - - @deprecated( - """ - on_event is deprecated, use lifespan event handlers instead. - - Read more about it in the - [FastAPI docs for Lifespan Events](https://fastapi.tiangolo.com/advanced/events/). - """ - ) - def on_event( - self, - event_type: Annotated[ - str, - Doc( - """ - The type of event. `startup` or `shutdown`. - """ - ), - ], - ) -> Callable[[DecoratedCallable], DecoratedCallable]: - """ - Add an event handler for the router. - - `on_event` is deprecated, use `lifespan` event handlers instead. - - Read more about it in the - [FastAPI docs for Lifespan Events](https://fastapi.tiangolo.com/advanced/events/#alternative-events-deprecated). - """ - - def decorator(func: DecoratedCallable) -> DecoratedCallable: - self.add_event_handler(event_type, func) - return func - - return decorator diff --git a/.env/Lib/site-packages/fastapi/security/__init__.py b/.env/Lib/site-packages/fastapi/security/__init__.py deleted file mode 100644 index 3aa6bf2..0000000 --- a/.env/Lib/site-packages/fastapi/security/__init__.py +++ /dev/null @@ -1,15 +0,0 @@ -from .api_key import APIKeyCookie as APIKeyCookie -from .api_key import APIKeyHeader as APIKeyHeader -from .api_key import APIKeyQuery as APIKeyQuery -from .http import HTTPAuthorizationCredentials as HTTPAuthorizationCredentials -from .http import HTTPBasic as HTTPBasic -from .http import HTTPBasicCredentials as HTTPBasicCredentials -from .http import HTTPBearer as HTTPBearer -from .http import HTTPDigest as HTTPDigest -from .oauth2 import OAuth2 as OAuth2 -from .oauth2 import OAuth2AuthorizationCodeBearer as OAuth2AuthorizationCodeBearer -from .oauth2 import OAuth2PasswordBearer as OAuth2PasswordBearer -from .oauth2 import OAuth2PasswordRequestForm as OAuth2PasswordRequestForm -from .oauth2 import OAuth2PasswordRequestFormStrict as OAuth2PasswordRequestFormStrict -from .oauth2 import SecurityScopes as SecurityScopes -from .open_id_connect_url import OpenIdConnect as OpenIdConnect diff --git a/.env/Lib/site-packages/fastapi/security/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/fastapi/security/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 7fa44d4c44cb0c5a847db0602f76bd89fa36a34c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 845 zcmZ{i%Wl*#6o#F-WUiS@TM1PH!GZ-vq9ZC+2o>5vs0G>%Glhi}DGJGfu_lRA+bIeU zz$@?!ya7v=NW8&FVZ(|YQg>Z(>}iW;fo1*r{&Q>}AIm=+r%7;p|MrQUlnMEz&E+Uv zz}Z)X&m<(F6_MDYmZiCbT%skzW#lp~8?Ho^xJs*rtH?IB4co{yS~FZjuG6~VI&yN9x`Wx1~ zy))W>0r`;g8H0a*5Bv}W`qm=O1MR*loUyhIA zB*U|xaS`LGH~*WQD8T}Kh0X+mOfZ#C0-l2I$~j6Q*$;<2Ngz;%8>{o0pE7R-xxnv1 zL`6##WnP&ol?q$?nwGkjh8F#!<~H|rccP^uZ7q(Lj+U;L9?E@x`FZ_O%)=~#ZLyBU^goeXD1XY9WfkPUnS+8H7UWJr dwhMCC%-aPS7vxDno)_dzK_2`hkHt0I_80U#@Javx diff --git a/.env/Lib/site-packages/fastapi/security/__pycache__/api_key.cpython-312.pyc b/.env/Lib/site-packages/fastapi/security/__pycache__/api_key.cpython-312.pyc deleted file mode 100644 index 7a252231807bbf766a72e832e53c0c884b472313..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10482 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 MJ!i4yFCZ{Izil0a_y7O^ diff --git a/.env/Lib/site-packages/fastapi/security/__pycache__/http.cpython-312.pyc b/.env/Lib/site-packages/fastapi/security/__pycache__/http.cpython-312.pyc deleted file mode 100644 index 71c42ac7240c42a88103a8d9f17ae7d9ae7151b6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14535 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 Ig#ncR1!~S9X#fBK diff --git a/.env/Lib/site-packages/fastapi/security/__pycache__/oauth2.cpython-312.pyc b/.env/Lib/site-packages/fastapi/security/__pycache__/oauth2.cpython-312.pyc deleted file mode 100644 index 2578e62aaa7938e892f62683c175533de030650a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 20667 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 diff --git a/.env/Lib/site-packages/fastapi/security/__pycache__/open_id_connect_url.cpython-312.pyc b/.env/Lib/site-packages/fastapi/security/__pycache__/open_id_connect_url.cpython-312.pyc deleted file mode 100644 index da6f13cd4d1b546cf8a3b0127cbd8ed11281d1f3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3761 zcmcgvO>7&-6`uX!uSk)yC0dQ8cn*#&$`aiqMbo5i0y&Nwp|)&TQiT!%R;%5iwDxkB znORDvpdyBA6mHR?z8L`uv_}U@(d6c1j|F8l@YE@Nmth!oV zuj;a|xrutRnv#9pP1iHkjO-`eY`v%2Bl}4=SI<}TvY&E$>xF7TAu9PHvD0^mone^` zE&66P*~*rC{)vOK8fGRu&kxLi*0*V7rZ$$bM=jz7L!|HB6W?u)no$K4xr5V&bOa8W~U}*XWqPWZL<2t1uL>5 zYV+}6^C5KZ6Gp0vO{%J`R5e?LAI;W(tJw*rvxG`m(oSwD(TE$G-Okfc}w`%<3X zC(rDYXZOi_K%SO6b9*JVw4J|`tY++9maVCF;f`MIv5PDR8dDMW=%c+~MkV0sp#evLW-UG!)0~YCY98ht{XYltp+NrWsr~2 zn$Ib_Y2xy+-3q`BsTXaCLeh=M?6pf3`V+qyyX-OW7mS8q(Oi`AK|Q8t?=k`=z~GkO z?t=<1dvv!5%|7l$>0^}n#21qv;_&(pF zHmjLUH-O@)QO~DH8*~Cd<8x;%!pt%=ZN{bY(73Ik4CPy;ze1;1~hRbr2q=M`d{(7S9@8*r~q-bU-iwlus8f znyz3AV{{d`L2CO-w+|QrT7a{uxner536%l<0Fyw7kVvE{Kn~PzLPW8yb_fJwCLqwV z0uN|~id>iio>(}RLbjX;b-~A1m)x zo{UaB9-X+W|1CY(e&*y|{nv#}*h#`9XF-#D+gZg)%z@16D0FsbHHh;Gw-E(zEROP? z-o0u~Ljb4N`lb-9JEhF*wc^~|-1x+eI{wy1_r~n&)?P)$!_eF!pA?2Z860mP zJ=GpK{b2A>{|j3wy_oF~J(J~w_|a5MmezJ8U>0NbGx*!Q4b6{;0x?@xu0wzSa~)Ui zdu(MR#>P+8byW->jtgFd`P9CxFx-fzZYwRau1=FbXsZffBFtmJx6hQa&Yy*ap|->V zS6s^`mmZg1`mpr!qvFedKX$xb7{oGQexjFy3mKE?zWXlrjvs8vuCG%uo{n=Lk@a8bW>1_g#Jr z$Fh9TRi44-IcVM`GLh0Td6kt-4@g8CnmgnRHJwOz&XIgyryrXUX#T~N?_P>R zJytyI*_+Zo;L)StrNx8GfFOACwz8h~Vx@s@6GomZ@v$e$}9rBfA^d(VuZdL z@f+@xj4lG{qZq|FK`p$AF^VmQnRN>Mm@vXDX4NOEZpZdBd($o*-~DFBeXM8h=89)2 zP4sD6sAeYk6KagG$!N?(*rrLrv@4h@L>fVeKOn%&89jv3M;#<_i92ZP@1ALMZ0$|m z04BP)YjsHncgU8R{||dpUAOKdpKSP+M&b~7W9$JxoKvB=38)>Z1kZh2*l?>|88ump zX&hn{xj|0gS zyO7gpn>N6%Q>kdqmnDk|!Bx3jD4xirywqnKOojjM?N^R)l0nQv29rasLfJvzU36q| zFuyRUu6(N24yv`cYagpO5A2gWw|lpG#bECE?&W^%MeXJF{W)*n_C_#;Hhf3Bs?2%X z5N2PRz{9npiacSHAG|^082I8Sl<~~9Ru&ftyCD`pnypJ8%8+1;Kch24*Tv@s-uc%j ZUR8z!_!nokcWIaOt~?EYIG`t=`UzG0qiz5I diff --git a/.env/Lib/site-packages/fastapi/security/api_key.py b/.env/Lib/site-packages/fastapi/security/api_key.py deleted file mode 100644 index 18dfb8e..0000000 --- a/.env/Lib/site-packages/fastapi/security/api_key.py +++ /dev/null @@ -1,318 +0,0 @@ -from typing import Annotated, Optional, Union - -from annotated_doc import Doc -from fastapi.openapi.models import APIKey, APIKeyIn -from fastapi.security.base import SecurityBase -from starlette.exceptions import HTTPException -from starlette.requests import Request -from starlette.status import HTTP_401_UNAUTHORIZED - - -class APIKeyBase(SecurityBase): - def __init__( - self, - location: APIKeyIn, - name: str, - description: Union[str, None], - scheme_name: Union[str, None], - auto_error: bool, - ): - self.auto_error = auto_error - - self.model: APIKey = APIKey( - **{"in": location}, - name=name, - description=description, - ) - self.scheme_name = scheme_name or self.__class__.__name__ - - def make_not_authenticated_error(self) -> HTTPException: - """ - The WWW-Authenticate header is not standardized for API Key authentication but - the HTTP specification requires that an error of 401 "Unauthorized" must - include a WWW-Authenticate header. - - Ref: https://datatracker.ietf.org/doc/html/rfc9110#name-401-unauthorized - - For this, this method sends a custom challenge `APIKey`. - """ - return HTTPException( - status_code=HTTP_401_UNAUTHORIZED, - detail="Not authenticated", - headers={"WWW-Authenticate": "APIKey"}, - ) - - def check_api_key(self, api_key: Optional[str]) -> Optional[str]: - if not api_key: - if self.auto_error: - raise self.make_not_authenticated_error() - return None - return api_key - - -class APIKeyQuery(APIKeyBase): - """ - API key authentication using a query parameter. - - This defines the name of the query parameter that should be provided in the request - with the API key and integrates that into the OpenAPI documentation. It extracts - the key value sent in the query parameter automatically and provides it as the - dependency result. But it doesn't define how to send that API key to the client. - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be a string containing the key value. - - ## Example - - ```python - from fastapi import Depends, FastAPI - from fastapi.security import APIKeyQuery - - app = FastAPI() - - query_scheme = APIKeyQuery(name="api_key") - - - @app.get("/items/") - async def read_items(api_key: str = Depends(query_scheme)): - return {"api_key": api_key} - ``` - """ - - def __init__( - self, - *, - name: Annotated[ - str, - Doc("Query parameter name."), - ], - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the query parameter is not provided, `APIKeyQuery` will - automatically cancel the request and send the client an error. - - If `auto_error` is set to `False`, when the query parameter is not - available, instead of erroring out, the dependency result will be - `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in a query - parameter or in an HTTP Bearer token). - """ - ), - ] = True, - ): - super().__init__( - location=APIKeyIn.query, - name=name, - scheme_name=scheme_name, - description=description, - auto_error=auto_error, - ) - - async def __call__(self, request: Request) -> Optional[str]: - api_key = request.query_params.get(self.model.name) - return self.check_api_key(api_key) - - -class APIKeyHeader(APIKeyBase): - """ - API key authentication using a header. - - This defines the name of the header that should be provided in the request with - the API key and integrates that into the OpenAPI documentation. It extracts - the key value sent in the header automatically and provides it as the dependency - result. But it doesn't define how to send that key to the client. - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be a string containing the key value. - - ## Example - - ```python - from fastapi import Depends, FastAPI - from fastapi.security import APIKeyHeader - - app = FastAPI() - - header_scheme = APIKeyHeader(name="x-key") - - - @app.get("/items/") - async def read_items(key: str = Depends(header_scheme)): - return {"key": key} - ``` - """ - - def __init__( - self, - *, - name: Annotated[str, Doc("Header name.")], - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the header is not provided, `APIKeyHeader` will - automatically cancel the request and send the client an error. - - If `auto_error` is set to `False`, when the header is not available, - instead of erroring out, the dependency result will be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in a header or - in an HTTP Bearer token). - """ - ), - ] = True, - ): - super().__init__( - location=APIKeyIn.header, - name=name, - scheme_name=scheme_name, - description=description, - auto_error=auto_error, - ) - - async def __call__(self, request: Request) -> Optional[str]: - api_key = request.headers.get(self.model.name) - return self.check_api_key(api_key) - - -class APIKeyCookie(APIKeyBase): - """ - API key authentication using a cookie. - - This defines the name of the cookie that should be provided in the request with - the API key and integrates that into the OpenAPI documentation. It extracts - the key value sent in the cookie automatically and provides it as the dependency - result. But it doesn't define how to set that cookie. - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be a string containing the key value. - - ## Example - - ```python - from fastapi import Depends, FastAPI - from fastapi.security import APIKeyCookie - - app = FastAPI() - - cookie_scheme = APIKeyCookie(name="session") - - - @app.get("/items/") - async def read_items(session: str = Depends(cookie_scheme)): - return {"session": session} - ``` - """ - - def __init__( - self, - *, - name: Annotated[str, Doc("Cookie name.")], - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the cookie is not provided, `APIKeyCookie` will - automatically cancel the request and send the client an error. - - If `auto_error` is set to `False`, when the cookie is not available, - instead of erroring out, the dependency result will be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in a cookie or - in an HTTP Bearer token). - """ - ), - ] = True, - ): - super().__init__( - location=APIKeyIn.cookie, - name=name, - scheme_name=scheme_name, - description=description, - auto_error=auto_error, - ) - - async def __call__(self, request: Request) -> Optional[str]: - api_key = request.cookies.get(self.model.name) - return self.check_api_key(api_key) diff --git a/.env/Lib/site-packages/fastapi/security/base.py b/.env/Lib/site-packages/fastapi/security/base.py deleted file mode 100644 index c43555d..0000000 --- a/.env/Lib/site-packages/fastapi/security/base.py +++ /dev/null @@ -1,6 +0,0 @@ -from fastapi.openapi.models import SecurityBase as SecurityBaseModel - - -class SecurityBase: - model: SecurityBaseModel - scheme_name: str diff --git a/.env/Lib/site-packages/fastapi/security/http.py b/.env/Lib/site-packages/fastapi/security/http.py deleted file mode 100644 index b4c3bc6..0000000 --- a/.env/Lib/site-packages/fastapi/security/http.py +++ /dev/null @@ -1,423 +0,0 @@ -import binascii -from base64 import b64decode -from typing import Annotated, Optional - -from annotated_doc import Doc -from fastapi.exceptions import HTTPException -from fastapi.openapi.models import HTTPBase as HTTPBaseModel -from fastapi.openapi.models import HTTPBearer as HTTPBearerModel -from fastapi.security.base import SecurityBase -from fastapi.security.utils import get_authorization_scheme_param -from pydantic import BaseModel -from starlette.requests import Request -from starlette.status import HTTP_401_UNAUTHORIZED - - -class HTTPBasicCredentials(BaseModel): - """ - The HTTP Basic credentials given as the result of using `HTTPBasic` in a - dependency. - - Read more about it in the - [FastAPI docs for HTTP Basic Auth](https://fastapi.tiangolo.com/advanced/security/http-basic-auth/). - """ - - username: Annotated[str, Doc("The HTTP Basic username.")] - password: Annotated[str, Doc("The HTTP Basic password.")] - - -class HTTPAuthorizationCredentials(BaseModel): - """ - The HTTP authorization credentials in the result of using `HTTPBearer` or - `HTTPDigest` in a dependency. - - The HTTP authorization header value is split by the first space. - - The first part is the `scheme`, the second part is the `credentials`. - - For example, in an HTTP Bearer token scheme, the client will send a header - like: - - ``` - Authorization: Bearer deadbeef12346 - ``` - - In this case: - - * `scheme` will have the value `"Bearer"` - * `credentials` will have the value `"deadbeef12346"` - """ - - scheme: Annotated[ - str, - Doc( - """ - The HTTP authorization scheme extracted from the header value. - """ - ), - ] - credentials: Annotated[ - str, - Doc( - """ - The HTTP authorization credentials extracted from the header value. - """ - ), - ] - - -class HTTPBase(SecurityBase): - def __init__( - self, - *, - scheme: str, - scheme_name: Optional[str] = None, - description: Optional[str] = None, - auto_error: bool = True, - ): - self.model: HTTPBaseModel = HTTPBaseModel( - scheme=scheme, description=description - ) - self.scheme_name = scheme_name or self.__class__.__name__ - self.auto_error = auto_error - - def make_authenticate_headers(self) -> dict[str, str]: - return {"WWW-Authenticate": f"{self.model.scheme.title()}"} - - def make_not_authenticated_error(self) -> HTTPException: - return HTTPException( - status_code=HTTP_401_UNAUTHORIZED, - detail="Not authenticated", - headers=self.make_authenticate_headers(), - ) - - async def __call__( - self, request: Request - ) -> Optional[HTTPAuthorizationCredentials]: - authorization = request.headers.get("Authorization") - scheme, credentials = get_authorization_scheme_param(authorization) - if not (authorization and scheme and credentials): - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return HTTPAuthorizationCredentials(scheme=scheme, credentials=credentials) - - -class HTTPBasic(HTTPBase): - """ - HTTP Basic authentication. - - Ref: https://datatracker.ietf.org/doc/html/rfc7617 - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be an `HTTPBasicCredentials` object containing the - `username` and the `password`. - - Read more about it in the - [FastAPI docs for HTTP Basic Auth](https://fastapi.tiangolo.com/advanced/security/http-basic-auth/). - - ## Example - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - from fastapi.security import HTTPBasic, HTTPBasicCredentials - - app = FastAPI() - - security = HTTPBasic() - - - @app.get("/users/me") - def read_current_user(credentials: Annotated[HTTPBasicCredentials, Depends(security)]): - return {"username": credentials.username, "password": credentials.password} - ``` - """ - - def __init__( - self, - *, - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - realm: Annotated[ - Optional[str], - Doc( - """ - HTTP Basic authentication realm. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the HTTP Basic authentication is not provided (a - header), `HTTPBasic` will automatically cancel the request and send the - client an error. - - If `auto_error` is set to `False`, when the HTTP Basic authentication - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in HTTP Basic - authentication or in an HTTP Bearer token). - """ - ), - ] = True, - ): - self.model = HTTPBaseModel(scheme="basic", description=description) - self.scheme_name = scheme_name or self.__class__.__name__ - self.realm = realm - self.auto_error = auto_error - - def make_authenticate_headers(self) -> dict[str, str]: - if self.realm: - return {"WWW-Authenticate": f'Basic realm="{self.realm}"'} - return {"WWW-Authenticate": "Basic"} - - async def __call__( # type: ignore - self, request: Request - ) -> Optional[HTTPBasicCredentials]: - authorization = request.headers.get("Authorization") - scheme, param = get_authorization_scheme_param(authorization) - if not authorization or scheme.lower() != "basic": - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - try: - data = b64decode(param).decode("ascii") - except (ValueError, UnicodeDecodeError, binascii.Error) as e: - raise self.make_not_authenticated_error() from e - username, separator, password = data.partition(":") - if not separator: - raise self.make_not_authenticated_error() - return HTTPBasicCredentials(username=username, password=password) - - -class HTTPBearer(HTTPBase): - """ - HTTP Bearer token authentication. - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be an `HTTPAuthorizationCredentials` object containing - the `scheme` and the `credentials`. - - ## Example - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - from fastapi.security import HTTPAuthorizationCredentials, HTTPBearer - - app = FastAPI() - - security = HTTPBearer() - - - @app.get("/users/me") - def read_current_user( - credentials: Annotated[HTTPAuthorizationCredentials, Depends(security)] - ): - return {"scheme": credentials.scheme, "credentials": credentials.credentials} - ``` - """ - - def __init__( - self, - *, - bearerFormat: Annotated[Optional[str], Doc("Bearer token format.")] = None, - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the HTTP Bearer token is not provided (in an - `Authorization` header), `HTTPBearer` will automatically cancel the - request and send the client an error. - - If `auto_error` is set to `False`, when the HTTP Bearer token - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in an HTTP - Bearer token or in a cookie). - """ - ), - ] = True, - ): - self.model = HTTPBearerModel(bearerFormat=bearerFormat, description=description) - self.scheme_name = scheme_name or self.__class__.__name__ - self.auto_error = auto_error - - async def __call__( - self, request: Request - ) -> Optional[HTTPAuthorizationCredentials]: - authorization = request.headers.get("Authorization") - scheme, credentials = get_authorization_scheme_param(authorization) - if not (authorization and scheme and credentials): - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - if scheme.lower() != "bearer": - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return HTTPAuthorizationCredentials(scheme=scheme, credentials=credentials) - - -class HTTPDigest(HTTPBase): - """ - HTTP Digest authentication. - - **Warning**: this is only a stub to connect the components with OpenAPI in FastAPI, - but it doesn't implement the full Digest scheme, you would need to to subclass it - and implement it in your code. - - Ref: https://datatracker.ietf.org/doc/html/rfc7616 - - ## Usage - - Create an instance object and use that object as the dependency in `Depends()`. - - The dependency result will be an `HTTPAuthorizationCredentials` object containing - the `scheme` and the `credentials`. - - ## Example - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - from fastapi.security import HTTPAuthorizationCredentials, HTTPDigest - - app = FastAPI() - - security = HTTPDigest() - - - @app.get("/users/me") - def read_current_user( - credentials: Annotated[HTTPAuthorizationCredentials, Depends(security)] - ): - return {"scheme": credentials.scheme, "credentials": credentials.credentials} - ``` - """ - - def __init__( - self, - *, - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if the HTTP Digest is not provided, `HTTPDigest` will - automatically cancel the request and send the client an error. - - If `auto_error` is set to `False`, when the HTTP Digest is not - available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, in HTTP - Digest or in a cookie). - """ - ), - ] = True, - ): - self.model = HTTPBaseModel(scheme="digest", description=description) - self.scheme_name = scheme_name or self.__class__.__name__ - self.auto_error = auto_error - - async def __call__( - self, request: Request - ) -> Optional[HTTPAuthorizationCredentials]: - authorization = request.headers.get("Authorization") - scheme, credentials = get_authorization_scheme_param(authorization) - if not (authorization and scheme and credentials): - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - if scheme.lower() != "digest": - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return HTTPAuthorizationCredentials(scheme=scheme, credentials=credentials) diff --git a/.env/Lib/site-packages/fastapi/security/oauth2.py b/.env/Lib/site-packages/fastapi/security/oauth2.py deleted file mode 100644 index 58ffc5c..0000000 --- a/.env/Lib/site-packages/fastapi/security/oauth2.py +++ /dev/null @@ -1,693 +0,0 @@ -from typing import Annotated, Any, Optional, Union, cast - -from annotated_doc import Doc -from fastapi.exceptions import HTTPException -from fastapi.openapi.models import OAuth2 as OAuth2Model -from fastapi.openapi.models import OAuthFlows as OAuthFlowsModel -from fastapi.param_functions import Form -from fastapi.security.base import SecurityBase -from fastapi.security.utils import get_authorization_scheme_param -from starlette.requests import Request -from starlette.status import HTTP_401_UNAUTHORIZED - - -class OAuth2PasswordRequestForm: - """ - This is a dependency class to collect the `username` and `password` as form data - for an OAuth2 password flow. - - The OAuth2 specification dictates that for a password flow the data should be - collected using form data (instead of JSON) and that it should have the specific - fields `username` and `password`. - - All the initialization parameters are extracted from the request. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - - ## Example - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - from fastapi.security import OAuth2PasswordRequestForm - - app = FastAPI() - - - @app.post("/login") - def login(form_data: Annotated[OAuth2PasswordRequestForm, Depends()]): - data = {} - data["scopes"] = [] - for scope in form_data.scopes: - data["scopes"].append(scope) - if form_data.client_id: - data["client_id"] = form_data.client_id - if form_data.client_secret: - data["client_secret"] = form_data.client_secret - return data - ``` - - Note that for OAuth2 the scope `items:read` is a single scope in an opaque string. - You could have custom internal logic to separate it by colon characters (`:`) or - similar, and get the two parts `items` and `read`. Many applications do that to - group and organize permissions, you could do it as well in your application, just - know that that it is application specific, it's not part of the specification. - """ - - def __init__( - self, - *, - grant_type: Annotated[ - Union[str, None], - Form(pattern="^password$"), - Doc( - """ - The OAuth2 spec says it is required and MUST be the fixed string - "password". Nevertheless, this dependency class is permissive and - allows not passing it. If you want to enforce it, use instead the - `OAuth2PasswordRequestFormStrict` dependency. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ] = None, - username: Annotated[ - str, - Form(), - Doc( - """ - `username` string. The OAuth2 spec requires the exact field name - `username`. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - password: Annotated[ - str, - Form(json_schema_extra={"format": "password"}), - Doc( - """ - `password` string. The OAuth2 spec requires the exact field name - `password`. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - scope: Annotated[ - str, - Form(), - Doc( - """ - A single string with actually several scopes separated by spaces. Each - scope is also a string. - - For example, a single string with: - - ```python - "items:read items:write users:read profile openid" - ```` - - would represent the scopes: - - * `items:read` - * `items:write` - * `users:read` - * `profile` - * `openid` - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ] = "", - client_id: Annotated[ - Union[str, None], - Form(), - Doc( - """ - If there's a `client_id`, it can be sent as part of the form fields. - But the OAuth2 specification recommends sending the `client_id` and - `client_secret` (if any) using HTTP Basic auth. - """ - ), - ] = None, - client_secret: Annotated[ - Union[str, None], - Form(json_schema_extra={"format": "password"}), - Doc( - """ - If there's a `client_password` (and a `client_id`), they can be sent - as part of the form fields. But the OAuth2 specification recommends - sending the `client_id` and `client_secret` (if any) using HTTP Basic - auth. - """ - ), - ] = None, - ): - self.grant_type = grant_type - self.username = username - self.password = password - self.scopes = scope.split() - self.client_id = client_id - self.client_secret = client_secret - - -class OAuth2PasswordRequestFormStrict(OAuth2PasswordRequestForm): - """ - This is a dependency class to collect the `username` and `password` as form data - for an OAuth2 password flow. - - The OAuth2 specification dictates that for a password flow the data should be - collected using form data (instead of JSON) and that it should have the specific - fields `username` and `password`. - - All the initialization parameters are extracted from the request. - - The only difference between `OAuth2PasswordRequestFormStrict` and - `OAuth2PasswordRequestForm` is that `OAuth2PasswordRequestFormStrict` requires the - client to send the form field `grant_type` with the value `"password"`, which - is required in the OAuth2 specification (it seems that for no particular reason), - while for `OAuth2PasswordRequestForm` `grant_type` is optional. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - - ## Example - - ```python - from typing import Annotated - - from fastapi import Depends, FastAPI - from fastapi.security import OAuth2PasswordRequestForm - - app = FastAPI() - - - @app.post("/login") - def login(form_data: Annotated[OAuth2PasswordRequestFormStrict, Depends()]): - data = {} - data["scopes"] = [] - for scope in form_data.scopes: - data["scopes"].append(scope) - if form_data.client_id: - data["client_id"] = form_data.client_id - if form_data.client_secret: - data["client_secret"] = form_data.client_secret - return data - ``` - - Note that for OAuth2 the scope `items:read` is a single scope in an opaque string. - You could have custom internal logic to separate it by colon characters (`:`) or - similar, and get the two parts `items` and `read`. Many applications do that to - group and organize permissions, you could do it as well in your application, just - know that that it is application specific, it's not part of the specification. - - - grant_type: the OAuth2 spec says it is required and MUST be the fixed string "password". - This dependency is strict about it. If you want to be permissive, use instead the - OAuth2PasswordRequestForm dependency class. - username: username string. The OAuth2 spec requires the exact field name "username". - password: password string. The OAuth2 spec requires the exact field name "password". - scope: Optional string. Several scopes (each one a string) separated by spaces. E.g. - "items:read items:write users:read profile openid" - client_id: optional string. OAuth2 recommends sending the client_id and client_secret (if any) - using HTTP Basic auth, as: client_id:client_secret - client_secret: optional string. OAuth2 recommends sending the client_id and client_secret (if any) - using HTTP Basic auth, as: client_id:client_secret - """ - - def __init__( - self, - grant_type: Annotated[ - str, - Form(pattern="^password$"), - Doc( - """ - The OAuth2 spec says it is required and MUST be the fixed string - "password". This dependency is strict about it. If you want to be - permissive, use instead the `OAuth2PasswordRequestForm` dependency - class. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - username: Annotated[ - str, - Form(), - Doc( - """ - `username` string. The OAuth2 spec requires the exact field name - `username`. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - password: Annotated[ - str, - Form(), - Doc( - """ - `password` string. The OAuth2 spec requires the exact field name - `password`. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - scope: Annotated[ - str, - Form(), - Doc( - """ - A single string with actually several scopes separated by spaces. Each - scope is also a string. - - For example, a single string with: - - ```python - "items:read items:write users:read profile openid" - ```` - - would represent the scopes: - - * `items:read` - * `items:write` - * `users:read` - * `profile` - * `openid` - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ] = "", - client_id: Annotated[ - Union[str, None], - Form(), - Doc( - """ - If there's a `client_id`, it can be sent as part of the form fields. - But the OAuth2 specification recommends sending the `client_id` and - `client_secret` (if any) using HTTP Basic auth. - """ - ), - ] = None, - client_secret: Annotated[ - Union[str, None], - Form(), - Doc( - """ - If there's a `client_password` (and a `client_id`), they can be sent - as part of the form fields. But the OAuth2 specification recommends - sending the `client_id` and `client_secret` (if any) using HTTP Basic - auth. - """ - ), - ] = None, - ): - super().__init__( - grant_type=grant_type, - username=username, - password=password, - scope=scope, - client_id=client_id, - client_secret=client_secret, - ) - - -class OAuth2(SecurityBase): - """ - This is the base class for OAuth2 authentication, an instance of it would be used - as a dependency. All other OAuth2 classes inherit from it and customize it for - each OAuth2 flow. - - You normally would not create a new class inheriting from it but use one of the - existing subclasses, and maybe compose them if you want to support multiple flows. - - Read more about it in the - [FastAPI docs for Security](https://fastapi.tiangolo.com/tutorial/security/). - """ - - def __init__( - self, - *, - flows: Annotated[ - Union[OAuthFlowsModel, dict[str, dict[str, Any]]], - Doc( - """ - The dictionary of OAuth2 flows. - """ - ), - ] = OAuthFlowsModel(), - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if no HTTP Authorization header is provided, required for - OAuth2 authentication, it will automatically cancel the request and - send the client an error. - - If `auto_error` is set to `False`, when the HTTP Authorization header - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, with OAuth2 - or in a cookie). - """ - ), - ] = True, - ): - self.model = OAuth2Model( - flows=cast(OAuthFlowsModel, flows), description=description - ) - self.scheme_name = scheme_name or self.__class__.__name__ - self.auto_error = auto_error - - def make_not_authenticated_error(self) -> HTTPException: - """ - The OAuth 2 specification doesn't define the challenge that should be used, - because a `Bearer` token is not really the only option to authenticate. - - But declaring any other authentication challenge would be application-specific - as it's not defined in the specification. - - For practical reasons, this method uses the `Bearer` challenge by default, as - it's probably the most common one. - - If you are implementing an OAuth2 authentication scheme other than the provided - ones in FastAPI (based on bearer tokens), you might want to override this. - - Ref: https://datatracker.ietf.org/doc/html/rfc6749 - """ - return HTTPException( - status_code=HTTP_401_UNAUTHORIZED, - detail="Not authenticated", - headers={"WWW-Authenticate": "Bearer"}, - ) - - async def __call__(self, request: Request) -> Optional[str]: - authorization = request.headers.get("Authorization") - if not authorization: - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return authorization - - -class OAuth2PasswordBearer(OAuth2): - """ - OAuth2 flow for authentication using a bearer token obtained with a password. - An instance of it would be used as a dependency. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - - def __init__( - self, - tokenUrl: Annotated[ - str, - Doc( - """ - The URL to obtain the OAuth2 token. This would be the *path operation* - that has `OAuth2PasswordRequestForm` as a dependency. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ], - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - scopes: Annotated[ - Optional[dict[str, str]], - Doc( - """ - The OAuth2 scopes that would be required by the *path operations* that - use this dependency. - - Read more about it in the - [FastAPI docs for Simple OAuth2 with Password and Bearer](https://fastapi.tiangolo.com/tutorial/security/simple-oauth2/). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if no HTTP Authorization header is provided, required for - OAuth2 authentication, it will automatically cancel the request and - send the client an error. - - If `auto_error` is set to `False`, when the HTTP Authorization header - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, with OAuth2 - or in a cookie). - """ - ), - ] = True, - refreshUrl: Annotated[ - Optional[str], - Doc( - """ - The URL to refresh the token and obtain a new one. - """ - ), - ] = None, - ): - if not scopes: - scopes = {} - flows = OAuthFlowsModel( - password=cast( - Any, - { - "tokenUrl": tokenUrl, - "refreshUrl": refreshUrl, - "scopes": scopes, - }, - ) - ) - super().__init__( - flows=flows, - scheme_name=scheme_name, - description=description, - auto_error=auto_error, - ) - - async def __call__(self, request: Request) -> Optional[str]: - authorization = request.headers.get("Authorization") - scheme, param = get_authorization_scheme_param(authorization) - if not authorization or scheme.lower() != "bearer": - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return param - - -class OAuth2AuthorizationCodeBearer(OAuth2): - """ - OAuth2 flow for authentication using a bearer token obtained with an OAuth2 code - flow. An instance of it would be used as a dependency. - """ - - def __init__( - self, - authorizationUrl: str, - tokenUrl: Annotated[ - str, - Doc( - """ - The URL to obtain the OAuth2 token. - """ - ), - ], - refreshUrl: Annotated[ - Optional[str], - Doc( - """ - The URL to refresh the token and obtain a new one. - """ - ), - ] = None, - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - scopes: Annotated[ - Optional[dict[str, str]], - Doc( - """ - The OAuth2 scopes that would be required by the *path operations* that - use this dependency. - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if no HTTP Authorization header is provided, required for - OAuth2 authentication, it will automatically cancel the request and - send the client an error. - - If `auto_error` is set to `False`, when the HTTP Authorization header - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, with OAuth2 - or in a cookie). - """ - ), - ] = True, - ): - if not scopes: - scopes = {} - flows = OAuthFlowsModel( - authorizationCode=cast( - Any, - { - "authorizationUrl": authorizationUrl, - "tokenUrl": tokenUrl, - "refreshUrl": refreshUrl, - "scopes": scopes, - }, - ) - ) - super().__init__( - flows=flows, - scheme_name=scheme_name, - description=description, - auto_error=auto_error, - ) - - async def __call__(self, request: Request) -> Optional[str]: - authorization = request.headers.get("Authorization") - scheme, param = get_authorization_scheme_param(authorization) - if not authorization or scheme.lower() != "bearer": - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None # pragma: nocover - return param - - -class SecurityScopes: - """ - This is a special class that you can define in a parameter in a dependency to - obtain the OAuth2 scopes required by all the dependencies in the same chain. - - This way, multiple dependencies can have different scopes, even when used in the - same *path operation*. And with this, you can access all the scopes required in - all those dependencies in a single place. - - Read more about it in the - [FastAPI docs for OAuth2 scopes](https://fastapi.tiangolo.com/advanced/security/oauth2-scopes/). - """ - - def __init__( - self, - scopes: Annotated[ - Optional[list[str]], - Doc( - """ - This will be filled by FastAPI. - """ - ), - ] = None, - ): - self.scopes: Annotated[ - list[str], - Doc( - """ - The list of all the scopes required by dependencies. - """ - ), - ] = scopes or [] - self.scope_str: Annotated[ - str, - Doc( - """ - All the scopes required by all the dependencies in a single string - separated by spaces, as defined in the OAuth2 specification. - """ - ), - ] = " ".join(self.scopes) diff --git a/.env/Lib/site-packages/fastapi/security/open_id_connect_url.py b/.env/Lib/site-packages/fastapi/security/open_id_connect_url.py deleted file mode 100644 index f4d9533..0000000 --- a/.env/Lib/site-packages/fastapi/security/open_id_connect_url.py +++ /dev/null @@ -1,94 +0,0 @@ -from typing import Annotated, Optional - -from annotated_doc import Doc -from fastapi.openapi.models import OpenIdConnect as OpenIdConnectModel -from fastapi.security.base import SecurityBase -from starlette.exceptions import HTTPException -from starlette.requests import Request -from starlette.status import HTTP_401_UNAUTHORIZED - - -class OpenIdConnect(SecurityBase): - """ - OpenID Connect authentication class. An instance of it would be used as a - dependency. - - **Warning**: this is only a stub to connect the components with OpenAPI in FastAPI, - but it doesn't implement the full OpenIdConnect scheme, for example, it doesn't use - the OpenIDConnect URL. You would need to to subclass it and implement it in your - code. - """ - - def __init__( - self, - *, - openIdConnectUrl: Annotated[ - str, - Doc( - """ - The OpenID Connect URL. - """ - ), - ], - scheme_name: Annotated[ - Optional[str], - Doc( - """ - Security scheme name. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - description: Annotated[ - Optional[str], - Doc( - """ - Security scheme description. - - It will be included in the generated OpenAPI (e.g. visible at `/docs`). - """ - ), - ] = None, - auto_error: Annotated[ - bool, - Doc( - """ - By default, if no HTTP Authorization header is provided, required for - OpenID Connect authentication, it will automatically cancel the request - and send the client an error. - - If `auto_error` is set to `False`, when the HTTP Authorization header - is not available, instead of erroring out, the dependency result will - be `None`. - - This is useful when you want to have optional authentication. - - It is also useful when you want to have authentication that can be - provided in one of multiple optional ways (for example, with OpenID - Connect or in a cookie). - """ - ), - ] = True, - ): - self.model = OpenIdConnectModel( - openIdConnectUrl=openIdConnectUrl, description=description - ) - self.scheme_name = scheme_name or self.__class__.__name__ - self.auto_error = auto_error - - def make_not_authenticated_error(self) -> HTTPException: - return HTTPException( - status_code=HTTP_401_UNAUTHORIZED, - detail="Not authenticated", - headers={"WWW-Authenticate": "Bearer"}, - ) - - async def __call__(self, request: Request) -> Optional[str]: - authorization = request.headers.get("Authorization") - if not authorization: - if self.auto_error: - raise self.make_not_authenticated_error() - else: - return None - return authorization diff --git a/.env/Lib/site-packages/fastapi/security/utils.py b/.env/Lib/site-packages/fastapi/security/utils.py deleted file mode 100644 index fd349ae..0000000 --- a/.env/Lib/site-packages/fastapi/security/utils.py +++ /dev/null @@ -1,10 +0,0 @@ -from typing import Optional - - -def get_authorization_scheme_param( - authorization_header_value: Optional[str], -) -> tuple[str, str]: - if not authorization_header_value: - return "", "" - scheme, _, param = authorization_header_value.partition(" ") - return scheme, param.strip() diff --git a/.env/Lib/site-packages/fastapi/staticfiles.py b/.env/Lib/site-packages/fastapi/staticfiles.py deleted file mode 100644 index 299015d..0000000 --- a/.env/Lib/site-packages/fastapi/staticfiles.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.staticfiles import StaticFiles as StaticFiles # noqa diff --git a/.env/Lib/site-packages/fastapi/templating.py b/.env/Lib/site-packages/fastapi/templating.py deleted file mode 100644 index 0cb8684..0000000 --- a/.env/Lib/site-packages/fastapi/templating.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.templating import Jinja2Templates as Jinja2Templates # noqa diff --git a/.env/Lib/site-packages/fastapi/testclient.py b/.env/Lib/site-packages/fastapi/testclient.py deleted file mode 100644 index 4012406..0000000 --- a/.env/Lib/site-packages/fastapi/testclient.py +++ /dev/null @@ -1 +0,0 @@ -from starlette.testclient import TestClient as TestClient # noqa diff --git a/.env/Lib/site-packages/fastapi/types.py b/.env/Lib/site-packages/fastapi/types.py deleted file mode 100644 index 1c3a6de..0000000 --- a/.env/Lib/site-packages/fastapi/types.py +++ /dev/null @@ -1,11 +0,0 @@ -import types -from enum import Enum -from typing import Any, Callable, Optional, TypeVar, Union - -from pydantic import BaseModel -from pydantic.main import IncEx as IncEx - -DecoratedCallable = TypeVar("DecoratedCallable", bound=Callable[..., Any]) -UnionType = getattr(types, "UnionType", Union) -ModelNameMap = dict[Union[type[BaseModel], type[Enum]], str] -DependencyCacheKey = tuple[Optional[Callable[..., Any]], tuple[str, ...], str] diff --git a/.env/Lib/site-packages/fastapi/utils.py b/.env/Lib/site-packages/fastapi/utils.py deleted file mode 100644 index 1c3a088..0000000 --- a/.env/Lib/site-packages/fastapi/utils.py +++ /dev/null @@ -1,164 +0,0 @@ -import re -import warnings -from collections.abc import MutableMapping -from typing import ( - TYPE_CHECKING, - Any, - Optional, - Union, -) -from weakref import WeakKeyDictionary - -import fastapi -from fastapi._compat import ( - BaseConfig, - ModelField, - PydanticSchemaGenerationError, - Undefined, - UndefinedType, - Validator, - annotation_is_pydantic_v1, -) -from fastapi.datastructures import DefaultPlaceholder, DefaultType -from fastapi.exceptions import FastAPIDeprecationWarning, PydanticV1NotSupportedError -from pydantic import BaseModel -from pydantic.fields import FieldInfo -from typing_extensions import Literal - -from ._compat import v2 - -if TYPE_CHECKING: # pragma: nocover - from .routing import APIRoute - -# Cache for `create_cloned_field` -_CLONED_TYPES_CACHE: MutableMapping[type[BaseModel], type[BaseModel]] = ( - WeakKeyDictionary() -) - - -def is_body_allowed_for_status_code(status_code: Union[int, str, None]) -> bool: - if status_code is None: - return True - # Ref: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#patterned-fields-1 - if status_code in { - "default", - "1XX", - "2XX", - "3XX", - "4XX", - "5XX", - }: - return True - current_status_code = int(status_code) - return not (current_status_code < 200 or current_status_code in {204, 205, 304}) - - -def get_path_param_names(path: str) -> set[str]: - return set(re.findall("{(.*?)}", path)) - - -_invalid_args_message = ( - "Invalid args for response field! Hint: " - "check that {type_} is a valid Pydantic field type. " - "If you are using a return type annotation that is not a valid Pydantic " - "field (e.g. Union[Response, dict, None]) you can disable generating the " - "response model from the type annotation with the path operation decorator " - "parameter response_model=None. Read more: " - "https://fastapi.tiangolo.com/tutorial/response-model/" -) - - -def create_model_field( - name: str, - type_: Any, - class_validators: Optional[dict[str, Validator]] = None, - default: Optional[Any] = Undefined, - required: Union[bool, UndefinedType] = Undefined, - model_config: Union[type[BaseConfig], None] = None, - field_info: Optional[FieldInfo] = None, - alias: Optional[str] = None, - mode: Literal["validation", "serialization"] = "validation", - version: Literal["1", "auto"] = "auto", -) -> ModelField: - if annotation_is_pydantic_v1(type_): - raise PydanticV1NotSupportedError( - "pydantic.v1 models are no longer supported by FastAPI." - f" Please update the response model {type_!r}." - ) - class_validators = class_validators or {} - - field_info = field_info or FieldInfo(annotation=type_, default=default, alias=alias) - kwargs = {"mode": mode, "name": name, "field_info": field_info} - try: - return v2.ModelField(**kwargs) # type: ignore[arg-type] - except PydanticSchemaGenerationError: - raise fastapi.exceptions.FastAPIError( - _invalid_args_message.format(type_=type_) - ) from None - - -def create_cloned_field( - field: ModelField, - *, - cloned_types: Optional[MutableMapping[type[BaseModel], type[BaseModel]]] = None, -) -> ModelField: - return field - - -def generate_operation_id_for_path( - *, name: str, path: str, method: str -) -> str: # pragma: nocover - warnings.warn( - message="fastapi.utils.generate_operation_id_for_path() was deprecated, " - "it is not used internally, and will be removed soon", - category=FastAPIDeprecationWarning, - stacklevel=2, - ) - operation_id = f"{name}{path}" - operation_id = re.sub(r"\W", "_", operation_id) - operation_id = f"{operation_id}_{method.lower()}" - return operation_id - - -def generate_unique_id(route: "APIRoute") -> str: - operation_id = f"{route.name}{route.path_format}" - operation_id = re.sub(r"\W", "_", operation_id) - assert route.methods - operation_id = f"{operation_id}_{list(route.methods)[0].lower()}" - return operation_id - - -def deep_dict_update(main_dict: dict[Any, Any], update_dict: dict[Any, Any]) -> None: - for key, value in update_dict.items(): - if ( - key in main_dict - and isinstance(main_dict[key], dict) - and isinstance(value, dict) - ): - deep_dict_update(main_dict[key], value) - elif ( - key in main_dict - and isinstance(main_dict[key], list) - and isinstance(update_dict[key], list) - ): - main_dict[key] = main_dict[key] + update_dict[key] - else: - main_dict[key] = value - - -def get_value_or_default( - first_item: Union[DefaultPlaceholder, DefaultType], - *extra_items: Union[DefaultPlaceholder, DefaultType], -) -> Union[DefaultPlaceholder, DefaultType]: - """ - Pass items or `DefaultPlaceholder`s by descending priority. - - The first one to _not_ be a `DefaultPlaceholder` will be returned. - - Otherwise, the first item (a `DefaultPlaceholder`) will be returned. - """ - items = (first_item,) + extra_items - for item in items: - if not isinstance(item, DefaultPlaceholder): - return item - return first_item diff --git a/.env/Lib/site-packages/fastapi/websockets.py b/.env/Lib/site-packages/fastapi/websockets.py deleted file mode 100644 index 55a4ac4..0000000 --- a/.env/Lib/site-packages/fastapi/websockets.py +++ /dev/null @@ -1,3 +0,0 @@ -from starlette.websockets import WebSocket as WebSocket # noqa -from starlette.websockets import WebSocketDisconnect as WebSocketDisconnect # noqa -from starlette.websockets import WebSocketState as WebSocketState # noqa diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/INSTALLER b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/METADATA b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/METADATA deleted file mode 100644 index 78655dd..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/METADATA +++ /dev/null @@ -1,98 +0,0 @@ -Metadata-Version: 2.4 -Name: greenlet -Version: 3.3.1 -Summary: Lightweight in-process concurrent programming -Author-email: Alexey Borzenkov -Maintainer-email: Jason Madden -License-Expression: MIT AND Python-2.0 -Project-URL: Homepage, https://greenlet.readthedocs.io -Project-URL: Documentation, https://greenlet.readthedocs.io -Project-URL: Repository, https://github.com/python-greenlet/greenlet -Project-URL: Issues, https://github.com/python-greenlet/greenlet/issues -Project-URL: Changelog, https://greenlet.readthedocs.io/en/latest/changes.html -Keywords: greenlet,coroutine,concurrency,threads,cooperative -Classifier: Development Status :: 5 - Production/Stable -Classifier: Intended Audience :: Developers -Classifier: Natural Language :: English -Classifier: Programming Language :: C -Classifier: Programming Language :: Python -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: 3.13 -Classifier: Programming Language :: Python :: 3.14 -Classifier: Operating System :: OS Independent -Classifier: Topic :: Software Development :: Libraries :: Python Modules -Requires-Python: >=3.10 -Description-Content-Type: text/x-rst -License-File: LICENSE -License-File: LICENSE.PSF -Provides-Extra: docs -Requires-Dist: Sphinx; extra == "docs" -Requires-Dist: furo; extra == "docs" -Provides-Extra: test -Requires-Dist: objgraph; extra == "test" -Requires-Dist: psutil; extra == "test" -Requires-Dist: setuptools; extra == "test" -Dynamic: license-file - -.. This file is included into docs/history.rst - - -Greenlets are lightweight coroutines for in-process concurrent -programming. - -The "greenlet" package is a spin-off of `Stackless`_, a version of -CPython that supports micro-threads called "tasklets". Tasklets run -pseudo-concurrently (typically in a single or a few OS-level threads) -and are synchronized with data exchanges on "channels". - -A "greenlet", on the other hand, is a still more primitive notion of -micro-thread with no implicit scheduling; coroutines, in other words. -This is useful when you want to control exactly when your code runs. -You can build custom scheduled micro-threads on top of greenlet; -however, it seems that greenlets are useful on their own as a way to -make advanced control flow structures. For example, we can recreate -generators; the difference with Python's own generators is that our -generators can call nested functions and the nested functions can -yield values too. (Additionally, you don't need a "yield" keyword. See -the example in `test_generator.py -`_). - -Greenlets are provided as a C extension module for the regular unmodified -interpreter. - -.. _`Stackless`: http://www.stackless.com - - -Who is using Greenlet? -====================== - -There are several libraries that use Greenlet as a more flexible -alternative to Python's built in coroutine support: - - - `Concurrence`_ - - `Eventlet`_ - - `Gevent`_ - -.. _Concurrence: http://opensource.hyves.org/concurrence/ -.. _Eventlet: http://eventlet.net/ -.. _Gevent: http://www.gevent.org/ - -Getting Greenlet -================ - -The easiest way to get Greenlet is to install it with pip:: - - pip install greenlet - - -Source code archives and binary distributions are available on the -python package index at https://pypi.org/project/greenlet - -The source code repository is hosted on github: -https://github.com/python-greenlet/greenlet - -Documentation is available on readthedocs.org: -https://greenlet.readthedocs.io diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/RECORD b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/RECORD deleted file mode 100644 index 8b66a7d..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/RECORD +++ /dev/null @@ -1,121 +0,0 @@ -../../include/site/python3.12/greenlet/greenlet.h,sha256=goR9ZprGoq71mBRhzBuexFvTNryvf4uceMHMIaSxh3o,4919 -greenlet-3.3.1.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -greenlet-3.3.1.dist-info/METADATA,sha256=inkHwUnYCTTYTwPPNKP0QwPtb-E3W2vO5Wr6VT5fM6I,3832 -greenlet-3.3.1.dist-info/RECORD,, -greenlet-3.3.1.dist-info/WHEEL,sha256=-0WtXfa1FcFqYulduII3R48HVP8SNWtKBAbj4Y-usSY,102 -greenlet-3.3.1.dist-info/licenses/LICENSE,sha256=q5d7ZUYFZwMZUJSB7OSRmPQFoSk3KsOMlvUc2WwvRS4,1464 -greenlet-3.3.1.dist-info/licenses/LICENSE.PSF,sha256=ae-micx67Hc2I20wOcHWZeSwxzSlv-pav2MwI6uy__Q,2471 -greenlet-3.3.1.dist-info/top_level.txt,sha256=YSnRsCRoO61JGlP57o8iKL6rdLWDWuiyKD8ekpWUsDc,9 -greenlet/CObjects.cpp,sha256=jdL4suQ4l3rNX4T-quyF9jnXBMFhfAMqikjzYCwP5TM,3665 -greenlet/PyGreenlet.cpp,sha256=_7ts9jcGImX0anUjLVpWYfKAohnCAp1eViPUmiouFIo,26088 -greenlet/PyGreenlet.hpp,sha256=LhUVgLOiWGOlnfjHfcpjjqhYuGrOdOsuKjhyVyRT0f0,1498 -greenlet/PyGreenletUnswitchable.cpp,sha256=XGAYX6LkaJAywv7yN6AJEJtbeC5XpuwvIUVF7XBBWzg,4522 -greenlet/PyModule.cpp,sha256=Hcg7v335rvVQYe8znjzIjLkZT5qLCFDIgkN9tfJfOkA,8941 -greenlet/TBrokenGreenlet.cpp,sha256=_hLV4vPhUsiwb5rnNnrUNbyCBHe0NPbWvlB-UtyNTOI,1066 -greenlet/TExceptionState.cpp,sha256=GKPJWNgvUtUVicGW2gl5nlT3qud-GeBNlKxSiCoWALU,1421 -greenlet/TGreenlet.cpp,sha256=Z5_8INRQ04l68pgQTyRQJRrRn2fMVCTSw39bIioLx8k,26634 -greenlet/TGreenlet.hpp,sha256=C2N-jJJoH37mnQOZdU6dpHzVk8Jt6ZrtpyafebQ1xLI,29537 -greenlet/TGreenletGlobals.cpp,sha256=bD_S4-irJj_kmEwWfyUrFfNXMLx65-ALpNyuLdnDjP8,3358 -greenlet/TMainGreenlet.cpp,sha256=aUtJqc9eQVtjIe9ueqlDl2KmpLVErfeawtu1PKOmoC0,3580 -greenlet/TPythonState.cpp,sha256=65znGlG30s6voSXVIsvYNE42L9x6I4_TxvqYIZjGTA8,17594 -greenlet/TStackState.cpp,sha256=T295c_VRn3ov2yQ8n0evJN73YlieXX_8Tg2ddR4XRZc,7646 -greenlet/TThreadState.hpp,sha256=drV2HVk65ynQJVnIn8lQ7N9L5xKpiVHxACBhkIaRtGc,20159 -greenlet/TThreadStateCreator.hpp,sha256=uWnKS-OXOutt8lcDDTCxtZ1hOnJjDtMZM_whep_qtko,2722 -greenlet/TThreadStateDestroy.cpp,sha256=tVsIR4bCsCbWOG28YWbWOCuCeYqN3p8PQTDievFxwKM,8618 -greenlet/TUserGreenlet.cpp,sha256=ZN7_FxmN5IAb95wpcP6OwL87iugXYHSptjosHRVZ8Jw,24215 -greenlet/__init__.py,sha256=9mB_r1P36BeOTTycD3UScXN9oxVq3emfYdAjBijKbEE,1794 -greenlet/__pycache__/__init__.cpython-312.pyc,, -greenlet/_greenlet.cp312-win_amd64.pyd,sha256=sUn4L3TWA0SDwPrWr5CEttYYk_j88KfkAVo7Jb4PR6g,73216 -greenlet/greenlet.cpp,sha256=WLJ7tB8udLGkLn18GPSUVYZdms1i02omumzZBlxTvlI,11416 -greenlet/greenlet.h,sha256=goR9ZprGoq71mBRhzBuexFvTNryvf4uceMHMIaSxh3o,4919 -greenlet/greenlet_allocator.hpp,sha256=Ymeor5KOG_MmiBRSpJY6InFsatIJHnBPBrO7ETzliFE,1911 -greenlet/greenlet_compiler_compat.hpp,sha256=tP2AfsUu3zjkSqh5QUS9MWgI2675-wMpgwesmli4po0,4444 -greenlet/greenlet_cpython_compat.hpp,sha256=26BZd7HijX5G2AKg5fqYTbPVCne9d2eqxIxCBAfQWwM,4409 -greenlet/greenlet_exceptions.hpp,sha256=E6M80eSszo5dfEfvfm0iHJlEOSNMY0mepzif64Z6mfY,4674 -greenlet/greenlet_internal.hpp,sha256=Z-S_NZgq4ykMtWjfsfjgt6TuS_tmK7-BmRt3KBG3CvE,2816 -greenlet/greenlet_msvc_compat.hpp,sha256=6zttIzyIoiSREOGt2zGEos7sirU4_Bqxo-7UU6v7L4Y,3295 -greenlet/greenlet_refs.hpp,sha256=7HyYNspkZNg7iEHNj90Lmg9M782mgg1735ww5ufO1mo,35554 -greenlet/greenlet_slp_switch.hpp,sha256=YRz-KWeqi8E9_6ysufslzEVbus6EM0WtceVWXm5A0PQ,3401 -greenlet/greenlet_thread_support.hpp,sha256=KoLKoz1rG2NT37w1bA_GEAz370VLUPGKTfUQHqtKwFA,898 -greenlet/platform/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -greenlet/platform/__pycache__/__init__.cpython-312.pyc,, -greenlet/platform/setup_switch_x64_masm.cmd,sha256=ZpClUJeU0ujEPSTWNSepP0W2f9XiYQKA8QKSoVou8EU,143 -greenlet/platform/switch_aarch64_gcc.h,sha256=Bzt3HLXqPpBeXMFqPHv_iFrl3qQ-hLbg5U1-BLxhvDA,4431 -greenlet/platform/switch_alpha_unix.h,sha256=k9LA3BAtojTCGqMzmYyWg5I73jjHDqDE_UKZuWrEITE,701 -greenlet/platform/switch_amd64_unix.h,sha256=YvMD530xkEG_S4zBSgR3IlvARJMAiIP5lO85kH-Ixpw,2835 -greenlet/platform/switch_arm32_gcc.h,sha256=89_taedd3VelvYc_-TywNEVqAV77x_zDYaAPtOVGPP0,2558 -greenlet/platform/switch_arm32_ios.h,sha256=GEi3DbjVYwuSKsA6PUBtwV27oWfbtFpj_nZ-Dx2Yrz8,1959 -greenlet/platform/switch_arm64_masm.asm,sha256=RtKHfJETzcd6RVUv5UL1jhXCuzVc5IIES-NtPT-nPFo,1298 -greenlet/platform/switch_arm64_masm.obj,sha256=DmLnIB_icoEHAz1naue_pJPTZgR9ElM7-Nmztr-o9_U,746 -greenlet/platform/switch_arm64_msvc.h,sha256=CnFzBbEqaIliXwzNTxBtbG5WNpfphuZatpHTXVUfylI,414 -greenlet/platform/switch_csky_gcc.h,sha256=dJx1cCkiqgNari_cOY_KfnjzR3G-J2b_Mf6TKNzaXvk,1379 -greenlet/platform/switch_loongarch64_linux.h,sha256=mydHgN6yFNQzbDRWTYTYXM03rU_vQngeJyLTrGbY4qs,810 -greenlet/platform/switch_m68k_gcc.h,sha256=s8A58AzT0_csU9H7j5FlR9A8myKP3CJLWfDij756Eiw,966 -greenlet/platform/switch_mips_unix.h,sha256=rPH5D4Rn2IzUBBHHJwHF7Ga9QHvtvoiWz_LJtJVDEN8,1527 -greenlet/platform/switch_ppc64_aix.h,sha256=0kpzjH-gFk6l7J0qjNfFALAVydm-Gb-3xQEgeFHC5TQ,3963 -greenlet/platform/switch_ppc64_linux.h,sha256=J2H7nQdiZoFhkVWGWEFES6758VFspQ7PkUbHshZf8yI,3920 -greenlet/platform/switch_ppc_aix.h,sha256=7akkI638_kXcTj-z33z4Hp-yXCzBKZCgc9oywjgj2sI,3028 -greenlet/platform/switch_ppc_linux.h,sha256=z1G0G3sFmVeuweZpbfnW4T4YsqXLsGpAGkLhN1SzETg,2843 -greenlet/platform/switch_ppc_macosx.h,sha256=x6eeMN4JlCumd6oL2RnbTO24YvEi5Is1IwANClgcxbQ,2706 -greenlet/platform/switch_ppc_unix.h,sha256=B4e2xvK37ixz7E3eXUyafm4oJYwnSStbBk4HI-egl7M,2734 -greenlet/platform/switch_riscv_unix.h,sha256=MBhaF319hLczUNtfuyuBQFlw3fP9cXtv8fTnLbroBTc,990 -greenlet/platform/switch_s390_unix.h,sha256=vlduFQw6kpkalkx5mreVugwD77_3JZZmEGgLUUbCZg8,2850 -greenlet/platform/switch_sh_gcc.h,sha256=Salb5aBXHT_C6myIAB9QzoVgjtD4T4b1BsDOoGbRCIg,937 -greenlet/platform/switch_sparc_sun_gcc.h,sha256=A7T5AMMzTQaW7ZTLNAVl0RwJYc1ON_ZagcRR1dRU43g,2889 -greenlet/platform/switch_x32_unix.h,sha256=brLdkEECmpmMql9k4RNj1SzeAeat77LumkJy8O9G62Y,1572 -greenlet/platform/switch_x64_masm.asm,sha256=krAM4iTcFFUccNlAptXgx41oevPO5zsGOAdMdjU0rAY,1951 -greenlet/platform/switch_x64_masm.obj,sha256=GNtTNxYdo7idFUYsQv-mrXWgyT5EJ93-9q90lN6svtQ,1078 -greenlet/platform/switch_x64_msvc.h,sha256=LH4ZgQTC0PRuG38H7nRCUR5dDLQrz61bxaG4-5mZ9S0,1864 -greenlet/platform/switch_x86_msvc.h,sha256=nWtRXFfHb9rE3pjCe5o5tdoJyctklHBD4kVq7LX9fLs,13164 -greenlet/platform/switch_x86_unix.h,sha256=KqakzLbWleoUwgtpjDGt_4o63bOy5Q0Cxq8Re5n5OyA,3164 -greenlet/slp_platformselect.h,sha256=8tpogbpy7l8mS88NsVokrVpy8rGQWJQX_X3wjGGKsGQ,4036 -greenlet/tests/__init__.py,sha256=-Ev1aQIYbndrP0Dfc_P8g6JlOKnJWMOJFSrLOl3wHrQ,9984 -greenlet/tests/__pycache__/__init__.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_clearing_run_switches.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_cpp_exception.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_initialstub_already_started.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_slp_switch.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_switch_three_greenlets.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_switch_three_greenlets2.cpython-312.pyc,, -greenlet/tests/__pycache__/fail_switch_two_greenlets.cpython-312.pyc,, -greenlet/tests/__pycache__/leakcheck.cpython-312.pyc,, -greenlet/tests/__pycache__/test_contextvars.cpython-312.pyc,, -greenlet/tests/__pycache__/test_cpp.cpython-312.pyc,, -greenlet/tests/__pycache__/test_extension_interface.cpython-312.pyc,, -greenlet/tests/__pycache__/test_gc.cpython-312.pyc,, -greenlet/tests/__pycache__/test_generator.cpython-312.pyc,, -greenlet/tests/__pycache__/test_generator_nested.cpython-312.pyc,, -greenlet/tests/__pycache__/test_greenlet.cpython-312.pyc,, -greenlet/tests/__pycache__/test_greenlet_trash.cpython-312.pyc,, -greenlet/tests/__pycache__/test_leaks.cpython-312.pyc,, -greenlet/tests/__pycache__/test_stack_saved.cpython-312.pyc,, -greenlet/tests/__pycache__/test_throw.cpython-312.pyc,, -greenlet/tests/__pycache__/test_tracing.cpython-312.pyc,, -greenlet/tests/__pycache__/test_version.cpython-312.pyc,, -greenlet/tests/__pycache__/test_weakref.cpython-312.pyc,, -greenlet/tests/_test_extension.c,sha256=NGj8mIpnvlfLi4JU2rHSGo4LNZLD9UThOfHmvSZj2is,7179 -greenlet/tests/_test_extension.cp312-win_amd64.pyd,sha256=0dIcnhsUT_cXLGpCXtqWVrRPvBHVZ6TWQAftpfSlzSs,14336 -greenlet/tests/_test_extension_cpp.cp312-win_amd64.pyd,sha256=Gtaq-Xv3syUblEdpHZ_HGfbgcBYxEnQoXCxRpT8RXuQ,15872 -greenlet/tests/_test_extension_cpp.cpp,sha256=BQN0BP-yyNUP7mEHTDNLxOLCvPKfvWR-g-2gYIHKbPs,6915 -greenlet/tests/fail_clearing_run_switches.py,sha256=v6lj1G7GTRml7CrUPRuzxkk68p1MfMmNwykba1mFc3c,1310 -greenlet/tests/fail_cpp_exception.py,sha256=-nUH8slJOxgTVfFx2uE7HwDlVh-dEskr6zR-jyjsJgM,1018 -greenlet/tests/fail_initialstub_already_started.py,sha256=Ku8SxVRpwE5TK8-7Netnd3i6_nrkUGtKpTR8TisQDzE,2039 -greenlet/tests/fail_slp_switch.py,sha256=FmUo8YsGmXt1HqAu9nsJZwH5OKjgmgzH2nkc8iExCRs,553 -greenlet/tests/fail_switch_three_greenlets.py,sha256=KXffSSGYpKFrfLzF9gDuVIvSs-GKVuDQw5cxdnHF14s,1000 -greenlet/tests/fail_switch_three_greenlets2.py,sha256=JDXE9CQdVeJ1rw4YGC1wA2AC3hN3N3JosLcUfjX1rm4,1340 -greenlet/tests/fail_switch_two_greenlets.py,sha256=UuUCa6YqV7Lshm3Ra7MzuLnIpw_1NVZtT-Num0DVMBg,858 -greenlet/tests/leakcheck.py,sha256=Tli9zbAN_saaCZWYgKbbsrSRctowqIcZKH4GzoimzGs,12988 -greenlet/tests/test_contextvars.py,sha256=J8WpLsWaDnXbNVTl6OSEKhyoPF3T5DooBLpbRMf0qv0,10853 -greenlet/tests/test_cpp.py,sha256=Kn_fZONxXA_C5xRkFZALGTsAeFJDtRAIhR7z7IPCaHg,2809 -greenlet/tests/test_extension_interface.py,sha256=TlayHEP63tRrrjmBdTBvI9rW4ZBBp5ZxMSmuXHOGq4s,3944 -greenlet/tests/test_gc.py,sha256=lWYPr4MfFMyAt7zSyUnYxK4Yb_HWsdagOrnSMXZC6BM,3008 -greenlet/tests/test_generator.py,sha256=46DOhdQ61iyAv4Du_j5BVVrwlEoZwmlro6j5DKASQ7Y,1299 -greenlet/tests/test_generator_nested.py,sha256=MjgSMvUtJ2sd9zXSVMTBobsRXQp64rHTWS3xyoqsAbg,3886 -greenlet/tests/test_greenlet.py,sha256=r-qC8A0LfybrxABAG51cZdeoeRsMmGCGNNn8zetwdXU,49804 -greenlet/tests/test_greenlet_trash.py,sha256=f3YdPXMO-S5EcrYzpfv97M3a-kYwgYTO_6hfyXsC_BQ,8134 -greenlet/tests/test_leaks.py,sha256=NJwgzhos-2U-GsBB4K1v0E866vDbYOrTgd6CWT-6vLI,18533 -greenlet/tests/test_stack_saved.py,sha256=2r2x_sTVoMTOjzTGVQHhYFRKyM1Foto4k04RsO9Gda4,465 -greenlet/tests/test_throw.py,sha256=hRbOM7chvmejDOyK99_9tVXZtLIVXb8dGeNCE1Whslk,3840 -greenlet/tests/test_tracing.py,sha256=uNItNmw1ZuJSW0qQ5uyQM6eMeT1TOAfmqfrhoSfrWZY,8890 -greenlet/tests/test_version.py,sha256=GsqWNSUf9oXAduSzenycdWX9WTcDqz3sbY5RL8eVe-s,1380 -greenlet/tests/test_weakref.py,sha256=vg44gv40Ip1zSfgu-NPWFKDUvc-Z_ew4UFjszBQXf0k,918 diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/WHEEL b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/WHEEL deleted file mode 100644 index 977ac93..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/WHEEL +++ /dev/null @@ -1,5 +0,0 @@ -Wheel-Version: 1.0 -Generator: setuptools (80.10.1) -Root-Is-Purelib: false -Tag: cp312-cp312-win_amd64 - diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE deleted file mode 100644 index b73a4a1..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE +++ /dev/null @@ -1,30 +0,0 @@ -The following files are derived from Stackless Python and are subject to the -same license as Stackless Python: - - src/greenlet/slp_platformselect.h - files in src/greenlet/platform/ directory - -See LICENSE.PSF and http://www.stackless.com/ for details. - -Unless otherwise noted, the files in greenlet have been released under the -following MIT license: - -Copyright (c) Armin Rigo, Christian Tismer and contributors - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE.PSF b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE.PSF deleted file mode 100644 index d3b509a..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/licenses/LICENSE.PSF +++ /dev/null @@ -1,47 +0,0 @@ -PYTHON SOFTWARE FOUNDATION LICENSE VERSION 2 --------------------------------------------- - -1. This LICENSE AGREEMENT is between the Python Software Foundation -("PSF"), and the Individual or Organization ("Licensee") accessing and -otherwise using this software ("Python") in source or binary form and -its associated documentation. - -2. Subject to the terms and conditions of this License Agreement, PSF hereby -grants Licensee a nonexclusive, royalty-free, world-wide license to reproduce, -analyze, test, perform and/or display publicly, prepare derivative works, -distribute, and otherwise use Python alone or in any derivative version, -provided, however, that PSF's License Agreement and PSF's notice of copyright, -i.e., "Copyright (c) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, -2011 Python Software Foundation; All Rights Reserved" are retained in Python -alone or in any derivative version prepared by Licensee. - -3. In the event Licensee prepares a derivative work that is based on -or incorporates Python or any part thereof, and wants to make -the derivative work available to others as provided herein, then -Licensee hereby agrees to include in any such work a brief summary of -the changes made to Python. - -4. PSF is making Python available to Licensee on an "AS IS" -basis. PSF MAKES NO REPRESENTATIONS OR WARRANTIES, EXPRESS OR -IMPLIED. BY WAY OF EXAMPLE, BUT NOT LIMITATION, PSF MAKES NO AND -DISCLAIMS ANY REPRESENTATION OR WARRANTY OF MERCHANTABILITY OR FITNESS -FOR ANY PARTICULAR PURPOSE OR THAT THE USE OF PYTHON WILL NOT -INFRINGE ANY THIRD PARTY RIGHTS. - -5. PSF SHALL NOT BE LIABLE TO LICENSEE OR ANY OTHER USERS OF PYTHON -FOR ANY INCIDENTAL, SPECIAL, OR CONSEQUENTIAL DAMAGES OR LOSS AS -A RESULT OF MODIFYING, DISTRIBUTING, OR OTHERWISE USING PYTHON, -OR ANY DERIVATIVE THEREOF, EVEN IF ADVISED OF THE POSSIBILITY THEREOF. - -6. This License Agreement will automatically terminate upon a material -breach of its terms and conditions. - -7. Nothing in this License Agreement shall be deemed to create any -relationship of agency, partnership, or joint venture between PSF and -Licensee. This License Agreement does not grant permission to use PSF -trademarks or trade name in a trademark sense to endorse or promote -products or services of Licensee, or any third party. - -8. By copying, installing or otherwise using Python, Licensee -agrees to be bound by the terms and conditions of this License -Agreement. diff --git a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/top_level.txt b/.env/Lib/site-packages/greenlet-3.3.1.dist-info/top_level.txt deleted file mode 100644 index 46725be..0000000 --- a/.env/Lib/site-packages/greenlet-3.3.1.dist-info/top_level.txt +++ /dev/null @@ -1 +0,0 @@ -greenlet diff --git a/.env/Lib/site-packages/greenlet/CObjects.cpp b/.env/Lib/site-packages/greenlet/CObjects.cpp deleted file mode 100644 index c135995..0000000 --- a/.env/Lib/site-packages/greenlet/CObjects.cpp +++ /dev/null @@ -1,157 +0,0 @@ -#ifndef COBJECTS_CPP -#define COBJECTS_CPP -/***************************************************************************** - * C interface - * - * These are exported using the CObject API - */ -#ifdef __clang__ -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wunused-function" -#endif - -#include "greenlet_exceptions.hpp" - -#include "greenlet_internal.hpp" -#include "greenlet_refs.hpp" - - -#include "TThreadStateDestroy.cpp" - -#include "PyGreenlet.hpp" - -using greenlet::PyErrOccurred; -using greenlet::Require; - - - -extern "C" { -static PyGreenlet* -PyGreenlet_GetCurrent(void) -{ - return GET_THREAD_STATE().state().get_current().relinquish_ownership(); -} - -static int -PyGreenlet_SetParent(PyGreenlet* g, PyGreenlet* nparent) -{ - return green_setparent((PyGreenlet*)g, (PyObject*)nparent, NULL); -} - -static PyGreenlet* -PyGreenlet_New(PyObject* run, PyGreenlet* parent) -{ - using greenlet::refs::NewDictReference; - // In the past, we didn't use green_new and green_init, but that - // was a maintenance issue because we duplicated code. This way is - // much safer, but slightly slower. If that's a problem, we could - // refactor green_init to separate argument parsing from initialization. - OwnedGreenlet g = OwnedGreenlet::consuming(green_new(&PyGreenlet_Type, nullptr, nullptr)); - if (!g) { - return NULL; - } - - try { - NewDictReference kwargs; - if (run) { - kwargs.SetItem(mod_globs->str_run, run); - } - if (parent) { - kwargs.SetItem("parent", (PyObject*)parent); - } - - Require(green_init(g.borrow(), mod_globs->empty_tuple, kwargs.borrow())); - } - catch (const PyErrOccurred&) { - return nullptr; - } - - return g.relinquish_ownership(); -} - -static PyObject* -PyGreenlet_Switch(PyGreenlet* self, PyObject* args, PyObject* kwargs) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return NULL; - } - - if (args == NULL) { - args = mod_globs->empty_tuple; - } - - if (kwargs == NULL || !PyDict_Check(kwargs)) { - kwargs = NULL; - } - - return green_switch(self, args, kwargs); -} - -static PyObject* -PyGreenlet_Throw(PyGreenlet* self, PyObject* typ, PyObject* val, PyObject* tb) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return nullptr; - } - try { - PyErrPieces err_pieces(typ, val, tb); - return internal_green_throw(self, err_pieces).relinquish_ownership(); - } - catch (const PyErrOccurred&) { - return nullptr; - } -} - - - -static int -Extern_PyGreenlet_MAIN(PyGreenlet* self) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return -1; - } - return self->pimpl->main(); -} - -static int -Extern_PyGreenlet_ACTIVE(PyGreenlet* self) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return -1; - } - return self->pimpl->active(); -} - -static int -Extern_PyGreenlet_STARTED(PyGreenlet* self) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return -1; - } - return self->pimpl->started(); -} - -static PyGreenlet* -Extern_PyGreenlet_GET_PARENT(PyGreenlet* self) -{ - if (!PyGreenlet_Check(self)) { - PyErr_BadArgument(); - return NULL; - } - // This can return NULL even if there is no exception - return self->pimpl->parent().acquire(); -} -} // extern C. - -/** End C API ****************************************************************/ -#ifdef __clang__ -# pragma clang diagnostic pop -#endif - - -#endif diff --git a/.env/Lib/site-packages/greenlet/PyGreenlet.cpp b/.env/Lib/site-packages/greenlet/PyGreenlet.cpp deleted file mode 100644 index fd62241..0000000 --- a/.env/Lib/site-packages/greenlet/PyGreenlet.cpp +++ /dev/null @@ -1,774 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -#ifndef PYGREENLET_CPP -#define PYGREENLET_CPP -/***************** -The Python slot functions for TGreenlet. - */ - - -#define PY_SSIZE_T_CLEAN -#include -#include "structmember.h" // PyMemberDef - -#include "greenlet_internal.hpp" -#include "TThreadStateDestroy.cpp" -#include "TGreenlet.hpp" -// #include "TUserGreenlet.cpp" -// #include "TMainGreenlet.cpp" -// #include "TBrokenGreenlet.cpp" - - -#include "greenlet_refs.hpp" -#include "greenlet_slp_switch.hpp" - -#include "greenlet_thread_support.hpp" -#include "TGreenlet.hpp" - -#include "TGreenletGlobals.cpp" -#include "TThreadStateDestroy.cpp" -#include "PyGreenlet.hpp" -// #include "TGreenlet.cpp" - -// #include "TExceptionState.cpp" -// #include "TPythonState.cpp" -// #include "TStackState.cpp" - -using greenlet::LockGuard; -using greenlet::LockInitError; -using greenlet::PyErrOccurred; -using greenlet::Require; - -using greenlet::g_handle_exit; -using greenlet::single_result; - -using greenlet::Greenlet; -using greenlet::UserGreenlet; -using greenlet::MainGreenlet; -using greenlet::BrokenGreenlet; -using greenlet::ThreadState; -using greenlet::PythonState; - - - -static PyGreenlet* -green_new(PyTypeObject* type, PyObject* UNUSED(args), PyObject* UNUSED(kwds)) -{ - PyGreenlet* o = - (PyGreenlet*)PyBaseObject_Type.tp_new(type, mod_globs->empty_tuple, mod_globs->empty_dict); - if (o) { - // Recall: borrowing or getting the current greenlet - // causes the "deleteme list" to get cleared. So constructing a greenlet - // can do things like cause other greenlets to get finalized. - UserGreenlet* c = new UserGreenlet(o, GET_THREAD_STATE().state().borrow_current()); - assert(Py_REFCNT(o) == 1); - // Also: This looks like a memory leak, but isn't. Constructing the - // C++ object assigns it to the pimpl pointer of the Python object (o); - // we'll need that later. - assert(c == o->pimpl); - } - return o; -} - - -// green_init is used in the tp_init slot. So it's important that -// it can be called directly from CPython. Thus, we don't use -// BorrowedGreenlet and BorrowedObject --- although in theory -// these should be binary layout compatible, that may not be -// guaranteed to be the case (32-bit linux ppc possibly). -static int -green_init(PyGreenlet* self, PyObject* args, PyObject* kwargs) -{ - PyArgParseParam run; - PyArgParseParam nparent; - static const char* kwlist[] = { - "run", - "parent", - NULL - }; - - // recall: The O specifier does NOT increase the reference count. - if (!PyArg_ParseTupleAndKeywords( - args, kwargs, "|OO:green", (char**)kwlist, &run, &nparent)) { - return -1; - } - - if (run) { - if (green_setrun(self, run, NULL)) { - return -1; - } - } - if (nparent && !nparent.is_None()) { - return green_setparent(self, nparent, NULL); - } - return 0; -} - - - -static int -green_traverse(PyGreenlet* self, visitproc visit, void* arg) -{ - // We must only visit referenced objects, i.e. only objects - // Py_INCREF'ed by this greenlet (directly or indirectly): - // - // - stack_prev is not visited: holds previous stack pointer, but it's not - // referenced - // - frames are not visited as we don't strongly reference them; - // alive greenlets are not garbage collected - // anyway. This can be a problem, however, if this greenlet is - // never allowed to finish, and is referenced from the frame: we - // have an uncollectible cycle in that case. Note that the - // frame object itself is also frequently not even tracked by the GC - // starting with Python 3.7 (frames are allocated by the - // interpreter untracked, and only become tracked when their - // evaluation is finished if they have a refcount > 1). All of - // this is to say that we should probably strongly reference - // the frame object. Doing so, while always allowing GC on a - // greenlet, solves several leaks for us. - - Py_VISIT(self->dict); - if (!self->pimpl) { - // Hmm. I have seen this at interpreter shutdown time, - // I think. That's very odd because this doesn't go away until - // we're ``green_dealloc()``, at which point we shouldn't be - // traversed anymore. - return 0; - } - - return self->pimpl->tp_traverse(visit, arg); -} - -static int -green_is_gc(PyObject* _self) -{ - BorrowedGreenlet self(_self); - int result = 0; - /* Main greenlet can be garbage collected since it can only - become unreachable if the underlying thread exited. - Active greenlets --- including those that are suspended --- - cannot be garbage collected, however. - */ - if (self->main() || !self->active()) { - result = 1; - } - // The main greenlet pointer will eventually go away after the thread dies. - if (self->was_running_in_dead_thread()) { - // Our thread is dead! We can never run again. Might as well - // GC us. Note that if a tuple containing only us and other - // immutable objects had been scanned before this, when we - // would have returned 0, the tuple will take itself out of GC - // tracking and never be investigated again. So that could - // result in both us and the tuple leaking due to an - // unreachable/uncollectible reference. The same goes for - // dictionaries. - // - // It's not a great idea to be changing our GC state on the - // fly. - result = 1; - } - return result; -} - - -static int -green_clear(PyGreenlet* self) -{ - /* Greenlet is only cleared if it is about to be collected. - Since active greenlets are not garbage collectable, we can - be sure that, even if they are deallocated during clear, - nothing they reference is in unreachable or finalizers, - so even if it switches we are relatively safe. */ - // XXX: Are we responsible for clearing weakrefs here? - Py_CLEAR(self->dict); - return self->pimpl->tp_clear(); -} - -/** - * Returns 0 on failure (the object was resurrected) or 1 on success. - **/ -static int -_green_dealloc_kill_started_non_main_greenlet(BorrowedGreenlet self) -{ - /* Hacks hacks hacks copied from instance_dealloc() */ - /* Temporarily resurrect the greenlet. */ - assert(self.REFCNT() == 0); - Py_SET_REFCNT(self.borrow(), 1); - /* Save the current exception, if any. */ - PyErrPieces saved_err; - try { - // BY THE TIME WE GET HERE, the state may actually be going - // away - // if we're shutting down the interpreter and freeing thread - // entries, - // this could result in freeing greenlets that were leaked. So - // we can't try to read the state. - self->deallocing_greenlet_in_thread( - self->thread_state() - ? static_cast(GET_THREAD_STATE()) - : nullptr); - } - catch (const PyErrOccurred&) { - PyErr_WriteUnraisable(self.borrow_o()); - /* XXX what else should we do? */ - } - /* Check for no resurrection must be done while we keep - * our internal reference, otherwise PyFile_WriteObject - * causes recursion if using Py_INCREF/Py_DECREF - */ - if (self.REFCNT() == 1 && self->active()) { - /* Not resurrected, but still not dead! - XXX what else should we do? we complain. */ - PyObject* f = PySys_GetObject("stderr"); - Py_INCREF(self.borrow_o()); /* leak! */ - if (f != NULL) { - PyFile_WriteString("GreenletExit did not kill ", f); - PyFile_WriteObject(self.borrow_o(), f, 0); - PyFile_WriteString("\n", f); - } - } - /* Restore the saved exception. */ - saved_err.PyErrRestore(); - /* Undo the temporary resurrection; can't use DECREF here, - * it would cause a recursive call. - */ - assert(self.REFCNT() > 0); - - Py_ssize_t refcnt = self.REFCNT() - 1; - Py_SET_REFCNT(self.borrow_o(), refcnt); - if (refcnt != 0) { - /* Resurrected! */ - _Py_NewReference(self.borrow_o()); - Py_SET_REFCNT(self.borrow_o(), refcnt); - /* Better to use tp_finalizer slot (PEP 442) - * and call ``PyObject_CallFinalizerFromDealloc``, - * but that's only supported in Python 3.4+; see - * Modules/_io/iobase.c for an example. - * TODO: We no longer run on anything that old, switch to finalizers. - * - * The following approach is copied from iobase.c in CPython 2.7. - * (along with much of this function in general). Here's their - * comment: - * - * When called from a heap type's dealloc, the type will be - * decref'ed on return (see e.g. subtype_dealloc in typeobject.c). - * - * On free-threaded builds of CPython, the type is meant to be immortal - * so we probably shouldn't mess with this? See - * test_issue_245_reference_counting_subclass_no_threads - */ - if (PyType_HasFeature(self.TYPE(), Py_TPFLAGS_HEAPTYPE)) { - Py_INCREF(self.TYPE()); - } - - PyObject_GC_Track((PyObject*)self); - - GREENLET_Py_DEC_REFTOTAL; -#ifdef COUNT_ALLOCS - --Py_TYPE(self)->tp_frees; - --Py_TYPE(self)->tp_allocs; -#endif /* COUNT_ALLOCS */ - return 0; - } - return 1; -} - - -static void -green_dealloc(PyGreenlet* self) -{ - PyObject_GC_UnTrack(self); - BorrowedGreenlet me(self); - if (me->active() - && me->started() - && !me->main()) { - if (!_green_dealloc_kill_started_non_main_greenlet(me)) { - return; - } - } - - if (self->weakreflist != NULL) { - PyObject_ClearWeakRefs((PyObject*)self); - } - Py_CLEAR(self->dict); - - if (self->pimpl) { - // In case deleting this, which frees some memory, - // somehow winds up calling back into us. That's usually a - //bug in our code. - Greenlet* p = self->pimpl; - self->pimpl = nullptr; - delete p; - } - // and finally we're done. self is now invalid. - Py_TYPE(self)->tp_free((PyObject*)self); -} - - - -static OwnedObject -internal_green_throw(BorrowedGreenlet self, PyErrPieces& err_pieces) -{ - PyObject* result = nullptr; - err_pieces.PyErrRestore(); - assert(PyErr_Occurred()); - if (self->started() && !self->active()) { - /* dead greenlet: turn GreenletExit into a regular return */ - result = g_handle_exit(OwnedObject()).relinquish_ownership(); - } - self->args() <<= result; - - return single_result(self->g_switch()); -} - - - -PyDoc_STRVAR( - green_switch_doc, - "switch(*args, **kwargs)\n" - "\n" - "Switch execution to this greenlet.\n" - "\n" - "If this greenlet has never been run, then this greenlet\n" - "will be switched to using the body of ``self.run(*args, **kwargs)``.\n" - "\n" - "If the greenlet is active (has been run, but was switch()'ed\n" - "out before leaving its run function), then this greenlet will\n" - "be resumed and the return value to its switch call will be\n" - "None if no arguments are given, the given argument if one\n" - "argument is given, or the args tuple and keyword args dict if\n" - "multiple arguments are given.\n" - "\n" - "If the greenlet is dead, or is the current greenlet then this\n" - "function will simply return the arguments using the same rules as\n" - "above.\n"); - -static PyObject* -green_switch(PyGreenlet* self, PyObject* args, PyObject* kwargs) -{ - using greenlet::SwitchingArgs; - SwitchingArgs switch_args(OwnedObject::owning(args), OwnedObject::owning(kwargs)); - self->pimpl->may_switch_away(); - self->pimpl->args() <<= switch_args; - - // If we're switching out of a greenlet, and that switch is the - // last thing the greenlet does, the greenlet ought to be able to - // go ahead and die at that point. Currently, someone else must - // manually switch back to the greenlet so that we "fall off the - // end" and can perform cleanup. You'd think we'd be able to - // figure out that this is happening using the frame's ``f_lasti`` - // member, which is supposed to be an index into - // ``frame->f_code->co_code``, the bytecode string. However, in - // recent interpreters, ``f_lasti`` tends not to be updated thanks - // to things like the PREDICT() macros in ceval.c. So it doesn't - // really work to do that in many cases. For example, the Python - // code: - // def run(): - // greenlet.getcurrent().parent.switch() - // produces bytecode of len 16, with the actual call to switch() - // being at index 10 (in Python 3.10). However, the reported - // ``f_lasti`` we actually see is...5! (Which happens to be the - // second byte of the CALL_METHOD op for ``getcurrent()``). - - try { - //OwnedObject result = single_result(self->pimpl->g_switch()); - OwnedObject result(single_result(self->pimpl->g_switch())); -#ifndef NDEBUG - // Note that the current greenlet isn't necessarily self. If self - // finished, we went to one of its parents. - assert(!self->pimpl->args()); - - const BorrowedGreenlet& current = GET_THREAD_STATE().state().borrow_current(); - // It's possible it's never been switched to. - assert(!current->args()); -#endif - PyObject* p = result.relinquish_ownership(); - - if (!p && !PyErr_Occurred()) { - // This shouldn't be happening anymore, so the asserts - // are there for debug builds. Non-debug builds - // crash "gracefully" in this case, although there is an - // argument to be made for killing the process in all - // cases --- for this to be the case, our switches - // probably nested in an incorrect way, so the state is - // suspicious. Nothing should be corrupt though, just - // confused at the Python level. Letting this propagate is - // probably good enough. - assert(p || PyErr_Occurred()); - throw PyErrOccurred( - mod_globs->PyExc_GreenletError, - "Greenlet.switch() returned NULL without an exception set." - ); - } - return p; - } - catch(const PyErrOccurred&) { - return nullptr; - } -} - -PyDoc_STRVAR( - green_throw_doc, - "Switches execution to this greenlet, but immediately raises the\n" - "given exception in this greenlet. If no argument is provided, the " - "exception\n" - "defaults to `greenlet.GreenletExit`. The normal exception\n" - "propagation rules apply, as described for `switch`. Note that calling " - "this\n" - "method is almost equivalent to the following::\n" - "\n" - " def raiser():\n" - " raise typ, val, tb\n" - " g_raiser = greenlet(raiser, parent=g)\n" - " g_raiser.switch()\n" - "\n" - "except that this trick does not work for the\n" - "`greenlet.GreenletExit` exception, which would not propagate\n" - "from ``g_raiser`` to ``g``.\n"); - -static PyObject* -green_throw(PyGreenlet* self, PyObject* args) -{ - PyArgParseParam typ(mod_globs->PyExc_GreenletExit); - PyArgParseParam val; - PyArgParseParam tb; - - if (!PyArg_ParseTuple(args, "|OOO:throw", &typ, &val, &tb)) { - return nullptr; - } - - assert(typ.borrow() || val.borrow()); - - self->pimpl->may_switch_away(); - try { - // Both normalizing the error and the actual throw_greenlet - // could throw PyErrOccurred. - PyErrPieces err_pieces(typ.borrow(), val.borrow(), tb.borrow()); - - return internal_green_throw(self, err_pieces).relinquish_ownership(); - } - catch (const PyErrOccurred&) { - return nullptr; - } -} - -static int -green_bool(PyGreenlet* self) -{ - return self->pimpl->active(); -} - -/** - * CAUTION: Allocates memory, may run GC and arbitrary Python code. - */ -static PyObject* -green_getdict(PyGreenlet* self, void* UNUSED(context)) -{ - if (self->dict == NULL) { - self->dict = PyDict_New(); - if (self->dict == NULL) { - return NULL; - } - } - Py_INCREF(self->dict); - return self->dict; -} - -static int -green_setdict(PyGreenlet* self, PyObject* val, void* UNUSED(context)) -{ - PyObject* tmp; - - if (val == NULL) { - PyErr_SetString(PyExc_TypeError, "__dict__ may not be deleted"); - return -1; - } - if (!PyDict_Check(val)) { - PyErr_SetString(PyExc_TypeError, "__dict__ must be a dictionary"); - return -1; - } - tmp = self->dict; - Py_INCREF(val); - self->dict = val; - Py_XDECREF(tmp); - return 0; -} - -static bool -_green_not_dead(BorrowedGreenlet self) -{ - // XXX: Where else should we do this? - // Probably on entry to most Python-facing functions? - if (self->was_running_in_dead_thread()) { - self->deactivate_and_free(); - return false; - } - return self->active() || !self->started(); -} - - -static PyObject* -green_getdead(PyGreenlet* self, void* UNUSED(context)) -{ - if (_green_not_dead(self)) { - Py_RETURN_FALSE; - } - else { - Py_RETURN_TRUE; - } -} - -static PyObject* -green_get_stack_saved(PyGreenlet* self, void* UNUSED(context)) -{ - return PyLong_FromSsize_t(self->pimpl->stack_saved()); -} - - -static PyObject* -green_getrun(PyGreenlet* self, void* UNUSED(context)) -{ - try { - OwnedObject result(BorrowedGreenlet(self)->run()); - return result.relinquish_ownership(); - } - catch(const PyErrOccurred&) { - return nullptr; - } -} - - -static int -green_setrun(PyGreenlet* self, PyObject* nrun, void* UNUSED(context)) -{ - try { - BorrowedGreenlet(self)->run(nrun); - return 0; - } - catch(const PyErrOccurred&) { - return -1; - } -} - -static PyObject* -green_getparent(PyGreenlet* self, void* UNUSED(context)) -{ - return BorrowedGreenlet(self)->parent().acquire_or_None(); -} - - -static int -green_setparent(PyGreenlet* self, PyObject* nparent, void* UNUSED(context)) -{ - try { - BorrowedGreenlet(self)->parent(nparent); - } - catch(const PyErrOccurred&) { - return -1; - } - return 0; -} - - -static PyObject* -green_getcontext(const PyGreenlet* self, void* UNUSED(context)) -{ - const Greenlet *const g = self->pimpl; - try { - OwnedObject result(g->context()); - return result.relinquish_ownership(); - } - catch(const PyErrOccurred&) { - return nullptr; - } -} - -static int -green_setcontext(PyGreenlet* self, PyObject* nctx, void* UNUSED(context)) -{ - try { - BorrowedGreenlet(self)->context(nctx); - return 0; - } - catch(const PyErrOccurred&) { - return -1; - } -} - - -static PyObject* -green_getframe(PyGreenlet* self, void* UNUSED(context)) -{ - const PythonState::OwnedFrame& top_frame = BorrowedGreenlet(self)->top_frame(); - return top_frame.acquire_or_None(); -} - - -static PyObject* -green_getstate(PyGreenlet* self) -{ - PyErr_Format(PyExc_TypeError, - "cannot serialize '%s' object", - Py_TYPE(self)->tp_name); - return nullptr; -} - -static PyObject* -green_repr(PyGreenlet* _self) -{ - BorrowedGreenlet self(_self); - /* - Return a string like - - - The handling of greenlets across threads is not super good. - We mostly use the internal definitions of these terms, but they - generally should make sense to users as well. - */ - PyObject* result; - int never_started = !self->started() && !self->active(); - - const char* const tp_name = Py_TYPE(self)->tp_name; - - if (_green_not_dead(self)) { - /* XXX: The otid= is almost useless because you can't correlate it to - any thread identifier exposed to Python. We could use - PyThreadState_GET()->thread_id, but we'd need to save that in the - greenlet, or save the whole PyThreadState object itself. - - As it stands, its only useful for identifying greenlets from the same thread. - */ - const char* state_in_thread; - if (self->was_running_in_dead_thread()) { - // The thread it was running in is dead! - // This can happen, especially at interpreter shut down. - // It complicates debugging output because it may be - // impossible to access the current thread state at that - // time. Thus, don't access the current thread state. - state_in_thread = " (thread exited)"; - } - else { - state_in_thread = GET_THREAD_STATE().state().is_current(self) - ? " current" - : (self->started() ? " suspended" : ""); - } - result = PyUnicode_FromFormat( - "<%s object at %p (otid=%p)%s%s%s%s>", - tp_name, - self.borrow_o(), - self->thread_state(), - state_in_thread, - self->active() ? " active" : "", - never_started ? " pending" : " started", - self->main() ? " main" : "" - ); - } - else { - result = PyUnicode_FromFormat( - "<%s object at %p (otid=%p) %sdead>", - tp_name, - self.borrow_o(), - self->thread_state(), - self->was_running_in_dead_thread() - ? "(thread exited) " - : "" - ); - } - - return result; -} - - -static PyMethodDef green_methods[] = { - { - .ml_name="switch", - .ml_meth=reinterpret_cast(green_switch), - .ml_flags=METH_VARARGS | METH_KEYWORDS, - .ml_doc=green_switch_doc - }, - {.ml_name="throw", .ml_meth=(PyCFunction)green_throw, .ml_flags=METH_VARARGS, .ml_doc=green_throw_doc}, - {.ml_name="__getstate__", .ml_meth=(PyCFunction)green_getstate, .ml_flags=METH_NOARGS, .ml_doc=NULL}, - {.ml_name=NULL, .ml_meth=NULL} /* sentinel */ -}; - -static PyGetSetDef green_getsets[] = { - /* name, getter, setter, doc, context pointer */ - {.name="__dict__", .get=(getter)green_getdict, .set=(setter)green_setdict}, - {.name="run", .get=(getter)green_getrun, .set=(setter)green_setrun}, - {.name="parent", .get=(getter)green_getparent, .set=(setter)green_setparent}, - {.name="gr_frame", .get=(getter)green_getframe }, - { - .name="gr_context", - .get=(getter)green_getcontext, - .set=(setter)green_setcontext - }, - {.name="dead", .get=(getter)green_getdead}, - {.name="_stack_saved", .get=(getter)green_get_stack_saved}, - {.name=NULL} -}; - -static PyMemberDef green_members[] = { - {.name=NULL} -}; - -static PyNumberMethods green_as_number = { - .nb_bool=(inquiry)green_bool, -}; - - -PyTypeObject PyGreenlet_Type = { - .ob_base=PyVarObject_HEAD_INIT(NULL, 0) - .tp_name="greenlet.greenlet", /* tp_name */ - .tp_basicsize=sizeof(PyGreenlet), /* tp_basicsize */ - /* methods */ - .tp_dealloc=(destructor)green_dealloc, /* tp_dealloc */ - .tp_repr=(reprfunc)green_repr, /* tp_repr */ - .tp_as_number=&green_as_number, /* tp_as _number*/ - .tp_flags=G_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /* tp_flags */ - .tp_doc="greenlet(run=None, parent=None) -> greenlet\n\n" - "Creates a new greenlet object (without running it).\n\n" - " - *run* -- The callable to invoke.\n" - " - *parent* -- The parent greenlet. The default is the current " - "greenlet.", /* tp_doc */ - .tp_traverse=(traverseproc)green_traverse, /* tp_traverse */ - .tp_clear=(inquiry)green_clear, /* tp_clear */ - .tp_weaklistoffset=offsetof(PyGreenlet, weakreflist), /* tp_weaklistoffset */ - - .tp_methods=green_methods, /* tp_methods */ - .tp_members=green_members, /* tp_members */ - .tp_getset=green_getsets, /* tp_getset */ - .tp_dictoffset=offsetof(PyGreenlet, dict), /* tp_dictoffset */ - .tp_init=(initproc)green_init, /* tp_init */ - .tp_alloc=PyType_GenericAlloc, /* tp_alloc */ - .tp_new=(newfunc)green_new, /* tp_new */ - .tp_free=PyObject_GC_Del, /* tp_free */ -#ifndef Py_GIL_DISABLED -/* - We may have been handling this wrong all along. - - It shows as a problem with the GIL disabled. In builds of 3.14 with - assertions enabled, we break the garbage collector if we *ever* - return false from this function. The docs say this is to distinguish - some objects that are collectable vs some that are not, specifically - giving the example of PyTypeObject as the only place this is done, - where it distinguishes between static types like this one (allocated - by the C runtime at load time) and dynamic heap types (created at - runtime as objects). With the GIL disabled, all allocations that are - potentially collectable go in the mimalloc heap, and the collector - asserts that tp_is_gc() is true for them as it walks through the - heap object by object. Since we set the Py_TPFLAGS_HAS_GC bit, we - are always allocated in that mimalloc heap, so we must always be - collectable. - - XXX: TODO: Could this be responsible for some apparent leaks, even - on GIL builds, at least in 3.14? See if we can catch an assertion - failure in the GC on regular 3.14 as well. - */ - .tp_is_gc=(inquiry)green_is_gc, /* tp_is_gc */ -#endif -}; - -#endif - -// Local Variables: -// flycheck-clang-include-path: ("/opt/local/Library/Frameworks/Python.framework/Versions/3.8/include/python3.8") -// End: diff --git a/.env/Lib/site-packages/greenlet/PyGreenlet.hpp b/.env/Lib/site-packages/greenlet/PyGreenlet.hpp deleted file mode 100644 index df6cd80..0000000 --- a/.env/Lib/site-packages/greenlet/PyGreenlet.hpp +++ /dev/null @@ -1,35 +0,0 @@ -#ifndef PYGREENLET_HPP -#define PYGREENLET_HPP - - -#include "greenlet.h" -#include "greenlet_compiler_compat.hpp" -#include "greenlet_refs.hpp" - - -using greenlet::refs::OwnedGreenlet; -using greenlet::refs::BorrowedGreenlet; -using greenlet::refs::BorrowedObject;; -using greenlet::refs::OwnedObject; -using greenlet::refs::PyErrPieces; - - -// XXX: These doesn't really belong here, it's not a Python slot. -static OwnedObject internal_green_throw(BorrowedGreenlet self, PyErrPieces& err_pieces); - -static PyGreenlet* green_new(PyTypeObject* type, PyObject* UNUSED(args), PyObject* UNUSED(kwds)); -static int green_clear(PyGreenlet* self); -static int green_init(PyGreenlet* self, PyObject* args, PyObject* kwargs); -static int green_setparent(PyGreenlet* self, PyObject* nparent, void* UNUSED(context)); -static int green_setrun(PyGreenlet* self, PyObject* nrun, void* UNUSED(context)); -static int green_traverse(PyGreenlet* self, visitproc visit, void* arg); -static void green_dealloc(PyGreenlet* self); -static PyObject* green_getparent(PyGreenlet* self, void* UNUSED(context)); - -static int green_is_gc(PyObject* self); -static PyObject* green_getdead(PyGreenlet* self, void* UNUSED(context)); -static PyObject* green_getrun(PyGreenlet* self, void* UNUSED(context)); -static int green_setcontext(PyGreenlet* self, PyObject* nctx, void* UNUSED(context)); -static PyObject* green_getframe(PyGreenlet* self, void* UNUSED(context)); -static PyObject* green_repr(PyGreenlet* self); -#endif diff --git a/.env/Lib/site-packages/greenlet/PyGreenletUnswitchable.cpp b/.env/Lib/site-packages/greenlet/PyGreenletUnswitchable.cpp deleted file mode 100644 index 1b768ee..0000000 --- a/.env/Lib/site-packages/greenlet/PyGreenletUnswitchable.cpp +++ /dev/null @@ -1,147 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - Implementation of the Python slots for PyGreenletUnswitchable_Type -*/ -#ifndef PY_GREENLET_UNSWITCHABLE_CPP -#define PY_GREENLET_UNSWITCHABLE_CPP - - - -#define PY_SSIZE_T_CLEAN -#include -#include "structmember.h" // PyMemberDef - -#include "greenlet_internal.hpp" -// Code after this point can assume access to things declared in stdint.h, -// including the fixed-width types. This goes for the platform-specific switch functions -// as well. -#include "greenlet_refs.hpp" -#include "greenlet_slp_switch.hpp" - -#include "greenlet_thread_support.hpp" -#include "TGreenlet.hpp" - -#include "TGreenlet.cpp" -#include "TGreenletGlobals.cpp" -#include "TThreadStateDestroy.cpp" - - -using greenlet::LockGuard; -using greenlet::LockInitError; -using greenlet::PyErrOccurred; -using greenlet::Require; - -using greenlet::g_handle_exit; -using greenlet::single_result; - -using greenlet::Greenlet; -using greenlet::UserGreenlet; -using greenlet::MainGreenlet; -using greenlet::BrokenGreenlet; -using greenlet::ThreadState; -using greenlet::PythonState; - - -#include "PyGreenlet.hpp" - -static PyGreenlet* -green_unswitchable_new(PyTypeObject* type, PyObject* UNUSED(args), PyObject* UNUSED(kwds)) -{ - PyGreenlet* o = - (PyGreenlet*)PyBaseObject_Type.tp_new(type, mod_globs->empty_tuple, mod_globs->empty_dict); - if (o) { - new BrokenGreenlet(o, GET_THREAD_STATE().state().borrow_current()); - assert(Py_REFCNT(o) == 1); - } - return o; -} - -static PyObject* -green_unswitchable_getforce(PyGreenlet* self, void* UNUSED(context)) -{ - BrokenGreenlet* broken = dynamic_cast(self->pimpl); - return PyBool_FromLong(broken->_force_switch_error); -} - -static int -green_unswitchable_setforce(PyGreenlet* self, PyObject* nforce, void* UNUSED(context)) -{ - if (!nforce) { - PyErr_SetString( - PyExc_AttributeError, - "Cannot delete force_switch_error" - ); - return -1; - } - BrokenGreenlet* broken = dynamic_cast(self->pimpl); - int is_true = PyObject_IsTrue(nforce); - if (is_true == -1) { - return -1; - } - broken->_force_switch_error = is_true; - return 0; -} - -static PyObject* -green_unswitchable_getforceslp(PyGreenlet* self, void* UNUSED(context)) -{ - BrokenGreenlet* broken = dynamic_cast(self->pimpl); - return PyBool_FromLong(broken->_force_slp_switch_error); -} - -static int -green_unswitchable_setforceslp(PyGreenlet* self, PyObject* nforce, void* UNUSED(context)) -{ - if (!nforce) { - PyErr_SetString( - PyExc_AttributeError, - "Cannot delete force_slp_switch_error" - ); - return -1; - } - BrokenGreenlet* broken = dynamic_cast(self->pimpl); - int is_true = PyObject_IsTrue(nforce); - if (is_true == -1) { - return -1; - } - broken->_force_slp_switch_error = is_true; - return 0; -} - -static PyGetSetDef green_unswitchable_getsets[] = { - /* name, getter, setter, doc, closure (context pointer) */ - { - .name="force_switch_error", - .get=(getter)green_unswitchable_getforce, - .set=(setter)green_unswitchable_setforce, - .doc=NULL - }, - { - .name="force_slp_switch_error", - .get=(getter)green_unswitchable_getforceslp, - .set=(setter)green_unswitchable_setforceslp, - .doc=nullptr - }, - {.name=nullptr} -}; - -PyTypeObject PyGreenletUnswitchable_Type = { - .ob_base=PyVarObject_HEAD_INIT(NULL, 0) - .tp_name="greenlet._greenlet.UnswitchableGreenlet", - .tp_dealloc= (destructor)green_dealloc, /* tp_dealloc */ - .tp_flags=G_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /* tp_flags */ - .tp_doc="Undocumented internal class", /* tp_doc */ - .tp_traverse=(traverseproc)green_traverse, /* tp_traverse */ - .tp_clear=(inquiry)green_clear, /* tp_clear */ - - .tp_getset=green_unswitchable_getsets, /* tp_getset */ - .tp_base=&PyGreenlet_Type, /* tp_base */ - .tp_init=(initproc)green_init, /* tp_init */ - .tp_alloc=PyType_GenericAlloc, /* tp_alloc */ - .tp_new=(newfunc)green_unswitchable_new, /* tp_new */ - .tp_free=PyObject_GC_Del, /* tp_free */ - .tp_is_gc=(inquiry)green_is_gc, /* tp_is_gc */ -}; - - -#endif diff --git a/.env/Lib/site-packages/greenlet/PyModule.cpp b/.env/Lib/site-packages/greenlet/PyModule.cpp deleted file mode 100644 index a999dc9..0000000 --- a/.env/Lib/site-packages/greenlet/PyModule.cpp +++ /dev/null @@ -1,292 +0,0 @@ -#ifndef PY_MODULE_CPP -#define PY_MODULE_CPP - -#include "greenlet_internal.hpp" - - -#include "TGreenletGlobals.cpp" -#include "TMainGreenlet.cpp" -#include "TThreadStateDestroy.cpp" - -using greenlet::LockGuard; -using greenlet::ThreadState; - -#ifdef __clang__ -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wunused-function" -# pragma clang diagnostic ignored "-Wunused-variable" -#endif - -PyDoc_STRVAR(mod_getcurrent_doc, - "getcurrent() -> greenlet\n" - "\n" - "Returns the current greenlet (i.e. the one which called this " - "function).\n"); - -static PyObject* -mod_getcurrent(PyObject* UNUSED(module)) -{ - return GET_THREAD_STATE().state().get_current().relinquish_ownership_o(); -} - -PyDoc_STRVAR(mod_settrace_doc, - "settrace(callback) -> object\n" - "\n" - "Sets a new tracing function and returns the previous one.\n"); -static PyObject* -mod_settrace(PyObject* UNUSED(module), PyObject* args) -{ - PyArgParseParam tracefunc; - if (!PyArg_ParseTuple(args, "O", &tracefunc)) { - return NULL; - } - ThreadState& state = GET_THREAD_STATE(); - OwnedObject previous = state.get_tracefunc(); - if (!previous) { - previous = Py_None; - } - - state.set_tracefunc(tracefunc); - - return previous.relinquish_ownership(); -} - -PyDoc_STRVAR(mod_gettrace_doc, - "gettrace() -> object\n" - "\n" - "Returns the currently set tracing function, or None.\n"); - -static PyObject* -mod_gettrace(PyObject* UNUSED(module)) -{ - OwnedObject tracefunc = GET_THREAD_STATE().state().get_tracefunc(); - if (!tracefunc) { - tracefunc = Py_None; - } - return tracefunc.relinquish_ownership(); -} - - - -PyDoc_STRVAR(mod_set_thread_local_doc, - "set_thread_local(key, value) -> None\n" - "\n" - "Set a value in the current thread-local dictionary. Debugging only.\n"); - -static PyObject* -mod_set_thread_local(PyObject* UNUSED(module), PyObject* args) -{ - PyArgParseParam key; - PyArgParseParam value; - PyObject* result = NULL; - - if (PyArg_UnpackTuple(args, "set_thread_local", 2, 2, &key, &value)) { - if(PyDict_SetItem( - PyThreadState_GetDict(), // borrow - key, - value) == 0 ) { - // success - Py_INCREF(Py_None); - result = Py_None; - } - } - return result; -} - -PyDoc_STRVAR(mod_get_pending_cleanup_count_doc, - "get_pending_cleanup_count() -> Integer\n" - "\n" - "Get the number of greenlet cleanup operations pending. Testing only.\n"); - - -static PyObject* -mod_get_pending_cleanup_count(PyObject* UNUSED(module)) -{ - LockGuard cleanup_lock(*mod_globs->thread_states_to_destroy_lock); - return PyLong_FromSize_t(mod_globs->thread_states_to_destroy.size()); -} - -PyDoc_STRVAR(mod_get_total_main_greenlets_doc, - "get_total_main_greenlets() -> Integer\n" - "\n" - "Quickly return the number of main greenlets that exist. Testing only.\n"); - -static PyObject* -mod_get_total_main_greenlets(PyObject* UNUSED(module)) -{ - return PyLong_FromSize_t(G_TOTAL_MAIN_GREENLETS); -} - - - -PyDoc_STRVAR(mod_get_clocks_used_doing_optional_cleanup_doc, - "get_clocks_used_doing_optional_cleanup() -> Integer\n" - "\n" - "Get the number of clock ticks the program has used doing optional " - "greenlet cleanup.\n" - "Beginning in greenlet 2.0, greenlet tries to find and dispose of greenlets\n" - "that leaked after a thread exited. This requires invoking Python's garbage collector,\n" - "which may have a performance cost proportional to the number of live objects.\n" - "This function returns the amount of processor time\n" - "greenlet has used to do this. In programs that run with very large amounts of live\n" - "objects, this metric can be used to decide whether the cost of doing this cleanup\n" - "is worth the memory leak being corrected. If not, you can disable the cleanup\n" - "using ``enable_optional_cleanup(False)``.\n" - "The units are arbitrary and can only be compared to themselves (similarly to ``time.clock()``);\n" - "for example, to see how it scales with your heap. You can attempt to convert them into seconds\n" - "by dividing by the value of CLOCKS_PER_SEC." - "If cleanup has been disabled, returns None." - "\n" - "This is an implementation specific, provisional API. It may be changed or removed\n" - "in the future.\n" - ".. versionadded:: 2.0" - ); -static PyObject* -mod_get_clocks_used_doing_optional_cleanup(PyObject* UNUSED(module)) -{ - std::clock_t clocks = ThreadState::clocks_used_doing_gc(); - - if (clocks == std::clock_t(-1)) { - Py_RETURN_NONE; - } - // This might not actually work on some implementations; clock_t - // is an opaque type. - return PyLong_FromSsize_t(clocks); -} - -PyDoc_STRVAR(mod_enable_optional_cleanup_doc, - "mod_enable_optional_cleanup(bool) -> None\n" - "\n" - "Enable or disable optional cleanup operations.\n" - "See ``get_clocks_used_doing_optional_cleanup()`` for details.\n" - ); -static PyObject* -mod_enable_optional_cleanup(PyObject* UNUSED(module), PyObject* flag) -{ - int is_true = PyObject_IsTrue(flag); - if (is_true == -1) { - return nullptr; - } - - if (is_true) { - std::clock_t clocks = ThreadState::clocks_used_doing_gc(); - // If we already have a value, we don't want to lose it. - if (clocks == std::clock_t(-1)) { - ThreadState::set_clocks_used_doing_gc(0); - } - } - else { - ThreadState::set_clocks_used_doing_gc(std::clock_t(-1)); - } - Py_RETURN_NONE; -} - - - - -#if !GREENLET_PY313 -PyDoc_STRVAR(mod_get_tstate_trash_delete_nesting_doc, - "get_tstate_trash_delete_nesting() -> Integer\n" - "\n" - "Return the 'trash can' nesting level. Testing only.\n"); -static PyObject* -mod_get_tstate_trash_delete_nesting(PyObject* UNUSED(module)) -{ - PyThreadState* tstate = PyThreadState_GET(); - -#if GREENLET_PY312 - return PyLong_FromLong(tstate->trash.delete_nesting); -#else - return PyLong_FromLong(tstate->trash_delete_nesting); -#endif -} -#endif - - - - -static PyMethodDef GreenMethods[] = { - { - .ml_name="getcurrent", - .ml_meth=(PyCFunction)mod_getcurrent, - .ml_flags=METH_NOARGS, - .ml_doc=mod_getcurrent_doc - }, - { - .ml_name="settrace", - .ml_meth=(PyCFunction)mod_settrace, - .ml_flags=METH_VARARGS, - .ml_doc=mod_settrace_doc - }, - { - .ml_name="gettrace", - .ml_meth=(PyCFunction)mod_gettrace, - .ml_flags=METH_NOARGS, - .ml_doc=mod_gettrace_doc - }, - { - .ml_name="set_thread_local", - .ml_meth=(PyCFunction)mod_set_thread_local, - .ml_flags=METH_VARARGS, - .ml_doc=mod_set_thread_local_doc - }, - { - .ml_name="get_pending_cleanup_count", - .ml_meth=(PyCFunction)mod_get_pending_cleanup_count, - .ml_flags=METH_NOARGS, - .ml_doc=mod_get_pending_cleanup_count_doc - }, - { - .ml_name="get_total_main_greenlets", - .ml_meth=(PyCFunction)mod_get_total_main_greenlets, - .ml_flags=METH_NOARGS, - .ml_doc=mod_get_total_main_greenlets_doc - }, - { - .ml_name="get_clocks_used_doing_optional_cleanup", - .ml_meth=(PyCFunction)mod_get_clocks_used_doing_optional_cleanup, - .ml_flags=METH_NOARGS, - .ml_doc=mod_get_clocks_used_doing_optional_cleanup_doc - }, - { - .ml_name="enable_optional_cleanup", - .ml_meth=(PyCFunction)mod_enable_optional_cleanup, - .ml_flags=METH_O, - .ml_doc=mod_enable_optional_cleanup_doc - }, -#if !GREENLET_PY313 - { - .ml_name="get_tstate_trash_delete_nesting", - .ml_meth=(PyCFunction)mod_get_tstate_trash_delete_nesting, - .ml_flags=METH_NOARGS, - .ml_doc=mod_get_tstate_trash_delete_nesting_doc - }, -#endif - {.ml_name=NULL, .ml_meth=NULL} /* Sentinel */ -}; - -static const char* const copy_on_greentype[] = { - "getcurrent", - "error", - "GreenletExit", - "settrace", - "gettrace", - NULL -}; - -static struct PyModuleDef greenlet_module_def = { - .m_base=PyModuleDef_HEAD_INIT, - .m_name="greenlet._greenlet", - .m_doc=NULL, - .m_size=-1, - .m_methods=GreenMethods, -}; - - -#endif - -#ifdef __clang__ -# pragma clang diagnostic pop -#elif defined(__GNUC__) -# pragma GCC diagnostic pop -#endif diff --git a/.env/Lib/site-packages/greenlet/TBrokenGreenlet.cpp b/.env/Lib/site-packages/greenlet/TBrokenGreenlet.cpp deleted file mode 100644 index 7e9ab5b..0000000 --- a/.env/Lib/site-packages/greenlet/TBrokenGreenlet.cpp +++ /dev/null @@ -1,45 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of greenlet::UserGreenlet. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ - -#include "TGreenlet.hpp" - -namespace greenlet { - -void* BrokenGreenlet::operator new(size_t UNUSED(count)) -{ - return allocator.allocate(1); -} - - -void BrokenGreenlet::operator delete(void* ptr) -{ - return allocator.deallocate(static_cast(ptr), - 1); -} - -greenlet::PythonAllocator greenlet::BrokenGreenlet::allocator; - -bool -BrokenGreenlet::force_slp_switch_error() const noexcept -{ - return this->_force_slp_switch_error; -} - -UserGreenlet::switchstack_result_t BrokenGreenlet::g_switchstack(void) -{ - if (this->_force_switch_error) { - return switchstack_result_t(-1); - } - return UserGreenlet::g_switchstack(); -} - -}; //namespace greenlet diff --git a/.env/Lib/site-packages/greenlet/TExceptionState.cpp b/.env/Lib/site-packages/greenlet/TExceptionState.cpp deleted file mode 100644 index 08a94ae..0000000 --- a/.env/Lib/site-packages/greenlet/TExceptionState.cpp +++ /dev/null @@ -1,62 +0,0 @@ -#ifndef GREENLET_EXCEPTION_STATE_CPP -#define GREENLET_EXCEPTION_STATE_CPP - -#include -#include "TGreenlet.hpp" - -namespace greenlet { - - -ExceptionState::ExceptionState() -{ - this->clear(); -} - -void ExceptionState::operator<<(const PyThreadState *const tstate) noexcept -{ - this->exc_info = tstate->exc_info; - this->exc_state = tstate->exc_state; -} - -void ExceptionState::operator>>(PyThreadState *const tstate) noexcept -{ - tstate->exc_state = this->exc_state; - tstate->exc_info = - this->exc_info ? this->exc_info : &tstate->exc_state; - this->clear(); -} - -void ExceptionState::clear() noexcept -{ - this->exc_info = nullptr; - this->exc_state.exc_value = nullptr; -#if !GREENLET_PY311 - this->exc_state.exc_type = nullptr; - this->exc_state.exc_traceback = nullptr; -#endif - this->exc_state.previous_item = nullptr; -} - -int ExceptionState::tp_traverse(visitproc visit, void* arg) noexcept -{ - Py_VISIT(this->exc_state.exc_value); -#if !GREENLET_PY311 - Py_VISIT(this->exc_state.exc_type); - Py_VISIT(this->exc_state.exc_traceback); -#endif - return 0; -} - -void ExceptionState::tp_clear() noexcept -{ - Py_CLEAR(this->exc_state.exc_value); -#if !GREENLET_PY311 - Py_CLEAR(this->exc_state.exc_type); - Py_CLEAR(this->exc_state.exc_traceback); -#endif -} - - -}; // namespace greenlet - -#endif // GREENLET_EXCEPTION_STATE_CPP diff --git a/.env/Lib/site-packages/greenlet/TGreenlet.cpp b/.env/Lib/site-packages/greenlet/TGreenlet.cpp deleted file mode 100644 index 1fb056e..0000000 --- a/.env/Lib/site-packages/greenlet/TGreenlet.cpp +++ /dev/null @@ -1,725 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of greenlet::Greenlet. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#ifndef TGREENLET_CPP -#define TGREENLET_CPP -#include "greenlet_internal.hpp" -#include "TGreenlet.hpp" - - -#include "TGreenletGlobals.cpp" -#include "TThreadStateDestroy.cpp" - -namespace greenlet { - -Greenlet::Greenlet(PyGreenlet* p) - : Greenlet(p, StackState()) -{ -} - -Greenlet::Greenlet(PyGreenlet* p, const StackState& initial_stack) - : _self(p), stack_state(initial_stack) -{ - assert(p->pimpl == nullptr); - p->pimpl = this; -} - -Greenlet::~Greenlet() -{ - // XXX: Can't do this. tp_clear is a virtual function, and by the - // time we're here, we've sliced off our child classes. - //this->tp_clear(); - this->_self->pimpl = nullptr; -} - -bool -Greenlet::force_slp_switch_error() const noexcept -{ - return false; -} - -void -Greenlet::release_args() -{ - this->switch_args.CLEAR(); -} - -/** - * CAUTION: This will allocate memory and may trigger garbage - * collection and arbitrary Python code. - */ -OwnedObject -Greenlet::throw_GreenletExit_during_dealloc(const ThreadState& UNUSED(current_thread_state)) -{ - // If we're killed because we lost all references in the - // middle of a switch, that's ok. Don't reset the args/kwargs, - // we still want to pass them to the parent. - PyErr_SetString(mod_globs->PyExc_GreenletExit, - "Killing the greenlet because all references have vanished."); - // To get here it had to have run before - return this->g_switch(); -} - -inline void -Greenlet::slp_restore_state() noexcept -{ -#ifdef SLP_BEFORE_RESTORE_STATE - SLP_BEFORE_RESTORE_STATE(); -#endif - this->stack_state.copy_heap_to_stack( - this->thread_state()->borrow_current()->stack_state); -} - - -inline int -Greenlet::slp_save_state(char *const stackref) noexcept -{ - // XXX: This used to happen in the middle, before saving, but - // after finding the next owner. Does that matter? This is - // only defined for Sparc/GCC where it flushes register - // windows to the stack (I think) -#ifdef SLP_BEFORE_SAVE_STATE - SLP_BEFORE_SAVE_STATE(); -#endif - return this->stack_state.copy_stack_to_heap(stackref, - this->thread_state()->borrow_current()->stack_state); -} - -/** - * CAUTION: This will allocate memory and may trigger garbage - * collection and arbitrary Python code. - */ -OwnedObject -Greenlet::on_switchstack_or_initialstub_failure( - Greenlet* target, - const Greenlet::switchstack_result_t& err, - const bool target_was_me, - const bool was_initial_stub) -{ - // If we get here, either g_initialstub() - // failed, or g_switchstack() failed. Either one of those - // cases SHOULD leave us in the original greenlet with a valid stack. - if (!PyErr_Occurred()) { - PyErr_SetString( - PyExc_SystemError, - was_initial_stub - ? "Failed to switch stacks into a greenlet for the first time." - : "Failed to switch stacks into a running greenlet."); - } - this->release_args(); - - if (target && !target_was_me) { - target->murder_in_place(); - } - - assert(!err.the_new_current_greenlet); - assert(!err.origin_greenlet); - return OwnedObject(); - -} - -OwnedGreenlet -Greenlet::g_switchstack_success() noexcept -{ - PyThreadState* tstate = PyThreadState_GET(); - // restore the saved state - this->python_state >> tstate; - this->exception_state >> tstate; - - // The thread state hasn't been changed yet. - ThreadState* thread_state = this->thread_state(); - OwnedGreenlet result(thread_state->get_current()); - thread_state->set_current(this->self()); - //assert(thread_state->borrow_current().borrow() == this->_self); - return result; -} - -Greenlet::switchstack_result_t -Greenlet::g_switchstack(void) -{ - // if any of these assertions fail, it's likely because we - // switched away and tried to switch back to us. Early stages of - // switching are not reentrant because we re-use ``this->args()``. - // Switching away would happen if we trigger a garbage collection - // (by just using some Python APIs that happen to allocate Python - // objects) and some garbage had weakref callbacks or __del__ that - // switches (people don't write code like that by hand, but with - // gevent it's possible without realizing it) - assert(this->args() || PyErr_Occurred()); - { /* save state */ - if (this->thread_state()->is_current(this->self())) { - // Hmm, nothing to do. - // TODO: Does this bypass trace events that are - // important? - return switchstack_result_t(0, - this, this->thread_state()->borrow_current()); - } - BorrowedGreenlet current = this->thread_state()->borrow_current(); - PyThreadState* tstate = PyThreadState_GET(); - - current->python_state << tstate; - current->exception_state << tstate; - this->python_state.will_switch_from(tstate); - switching_thread_state = this; - current->expose_frames(); - } - assert(this->args() || PyErr_Occurred()); - // If this is the first switch into a greenlet, this will - // return twice, once with 1 in the new greenlet, once with 0 - // in the origin. - int err; - if (this->force_slp_switch_error()) { - err = -1; - } - else { - err = slp_switch(); - } - - if (err < 0) { /* error */ - // Tested by - // test_greenlet.TestBrokenGreenlets.test_failed_to_slp_switch_into_running - // - // It's not clear if it's worth trying to clean up and - // continue here. Failing to switch stacks is a big deal which - // may not be recoverable (who knows what state the stack is in). - // Also, we've stolen references in preparation for calling - // ``g_switchstack_success()`` and we don't have a clean - // mechanism for backing that all out. - Py_FatalError("greenlet: Failed low-level slp_switch(). The stack is probably corrupt."); - } - - // No stack-based variables are valid anymore. - - // But the global is volatile so we can reload it without the - // compiler caching it from earlier. - Greenlet* greenlet_that_switched_in = switching_thread_state; // aka this - switching_thread_state = nullptr; - // except that no stack variables are valid, we would: - // assert(this == greenlet_that_switched_in); - - // switchstack success is where we restore the exception state, - // etc. It returns the origin greenlet because its convenient. - - OwnedGreenlet origin = greenlet_that_switched_in->g_switchstack_success(); - assert(greenlet_that_switched_in->args() || PyErr_Occurred()); - return switchstack_result_t(err, greenlet_that_switched_in, origin); -} - - -inline void -Greenlet::check_switch_allowed() const -{ - // TODO: Make this take a parameter of the current greenlet, - // or current main greenlet, to make the check for - // cross-thread switching cheaper. Surely somewhere up the - // call stack we've already accessed the thread local variable. - - // We expect to always have a main greenlet now; accessing the thread state - // created it. However, if we get here and cleanup has already - // begun because we're a greenlet that was running in a - // (now dead) thread, these invariants will not hold true. In - // fact, accessing `this->thread_state` may not even be possible. - - // If the thread this greenlet was running in is dead, - // we'll still have a reference to a main greenlet, but the - // thread state pointer we have is bogus. - // TODO: Give the objects an API to determine if they belong - // to a dead thread. - - const BorrowedMainGreenlet main_greenlet = this->find_main_greenlet_in_lineage(); - - if (!main_greenlet) { - throw PyErrOccurred(mod_globs->PyExc_GreenletError, - "cannot switch to a garbage collected greenlet"); - } - - if (!main_greenlet->thread_state()) { - throw PyErrOccurred(mod_globs->PyExc_GreenletError, - "cannot switch to a different thread (which happens to have exited)"); - } - - // The main greenlet we found was from the .parent lineage. - // That may or may not have any relationship to the main - // greenlet of the running thread. We can't actually access - // our this->thread_state members to try to check that, - // because it could be in the process of getting destroyed, - // but setting the main_greenlet->thread_state member to NULL - // may not be visible yet. So we need to check against the - // current thread state (once the cheaper checks are out of - // the way) - const BorrowedMainGreenlet current_main_greenlet = GET_THREAD_STATE().state().borrow_main_greenlet(); - if ( - // lineage main greenlet is not this thread's greenlet - current_main_greenlet != main_greenlet - || ( - // atteched to some thread - this->main_greenlet() - // XXX: Same condition as above. Was this supposed to be - // this->main_greenlet()? - && current_main_greenlet != main_greenlet) - // switching into a known dead thread (XXX: which, if we get here, - // is bad, because we just accessed the thread state, which is - // gone!) - || (!current_main_greenlet->thread_state())) { - // CAUTION: This may trigger memory allocations, gc, and - // arbitrary Python code. - throw PyErrOccurred( - mod_globs->PyExc_GreenletError, - "Cannot switch to a different thread\n\tCurrent: %R\n\tExpected: %R", - current_main_greenlet, main_greenlet); - } -} - -const OwnedObject -Greenlet::context() const -{ - using greenlet::PythonStateContext; - OwnedObject result; - - if (this->is_currently_running_in_some_thread()) { - /* Currently running greenlet: context is stored in the thread state, - not the greenlet object. */ - if (GET_THREAD_STATE().state().is_current(this->self())) { - result = PythonStateContext::context(PyThreadState_GET()); - } - else { - throw ValueError( - "cannot get context of a " - "greenlet that is running in a different thread"); - } - } - else { - /* Greenlet is not running: just return context. */ - result = this->python_state.context(); - } - if (!result) { - result = OwnedObject::None(); - } - return result; -} - - -void -Greenlet::context(BorrowedObject given) -{ - using greenlet::PythonStateContext; - if (!given) { - throw AttributeError("can't delete context attribute"); - } - if (given.is_None()) { - /* "Empty context" is stored as NULL, not None. */ - given = nullptr; - } - - //checks type, incrs refcnt - greenlet::refs::OwnedContext context(given); - PyThreadState* tstate = PyThreadState_GET(); - - if (this->is_currently_running_in_some_thread()) { - if (!GET_THREAD_STATE().state().is_current(this->self())) { - throw ValueError("cannot set context of a greenlet" - " that is running in a different thread"); - } - - /* Currently running greenlet: context is stored in the thread state, - not the greenlet object. */ - OwnedObject octx = OwnedObject::consuming(PythonStateContext::context(tstate)); - PythonStateContext::context(tstate, context.relinquish_ownership()); - } - else { - /* Greenlet is not running: just set context. Note that the - greenlet may be dead.*/ - this->python_state.context() = context; - } -} - -/** - * CAUTION: May invoke arbitrary Python code. - * - * Figure out what the result of ``greenlet.switch(arg, kwargs)`` - * should be and transfers ownership of it to the left-hand-side. - * - * If switch() was just passed an arg tuple, then we'll just return that. - * If only keyword arguments were passed, then we'll pass the keyword - * argument dict. Otherwise, we'll create a tuple of (args, kwargs) and - * return both. - * - * CAUTION: This may allocate a new tuple object, which may - * cause the Python garbage collector to run, which in turn may - * run arbitrary Python code that switches. - */ -OwnedObject& operator<<=(OwnedObject& lhs, greenlet::SwitchingArgs& rhs) noexcept -{ - // Because this may invoke arbitrary Python code, which could - // result in switching back to us, we need to get the - // arguments locally on the stack. - assert(rhs); - OwnedObject args = rhs.args(); - OwnedObject kwargs = rhs.kwargs(); - rhs.CLEAR(); - // We shouldn't be called twice for the same switch. - assert(args || kwargs); - assert(!rhs); - - if (!kwargs) { - lhs = args; - } - else if (!PyDict_Size(kwargs.borrow())) { - lhs = args; - } - else if (!PySequence_Length(args.borrow())) { - lhs = kwargs; - } - else { - // PyTuple_Pack allocates memory, may GC, may run arbitrary - // Python code. - lhs = OwnedObject::consuming(PyTuple_Pack(2, args.borrow(), kwargs.borrow())); - } - return lhs; -} - -static OwnedObject -g_handle_exit(const OwnedObject& greenlet_result) -{ - if (!greenlet_result && mod_globs->PyExc_GreenletExit.PyExceptionMatches()) { - /* catch and ignore GreenletExit */ - PyErrFetchParam val; - PyErr_Fetch(PyErrFetchParam(), val, PyErrFetchParam()); - if (!val) { - return OwnedObject::None(); - } - return OwnedObject(val); - } - - if (greenlet_result) { - // package the result into a 1-tuple - // PyTuple_Pack increments the reference of its arguments, - // so we always need to decref the greenlet result; - // the owner will do that. - return OwnedObject::consuming(PyTuple_Pack(1, greenlet_result.borrow())); - } - - return OwnedObject(); -} - - - -/** - * May run arbitrary Python code. - */ -OwnedObject -Greenlet::g_switch_finish(const switchstack_result_t& err) -{ - assert(err.the_new_current_greenlet == this); - - ThreadState& state = *this->thread_state(); - // Because calling the trace function could do arbitrary things, - // including switching away from this greenlet and then maybe - // switching back, we need to capture the arguments now so that - // they don't change. - OwnedObject result; - if (this->args()) { - result <<= this->args(); - } - else { - assert(PyErr_Occurred()); - } - assert(!this->args()); - try { - // Our only caller handles the bad error case - assert(err.status >= 0); - assert(state.borrow_current() == this->self()); - if (OwnedObject tracefunc = state.get_tracefunc()) { - assert(result || PyErr_Occurred()); - g_calltrace(tracefunc, - result ? mod_globs->event_switch : mod_globs->event_throw, - err.origin_greenlet, - this->self()); - } - // The above could have invoked arbitrary Python code, but - // it couldn't switch back to this object and *also* - // throw an exception, so the args won't have changed. - - if (PyErr_Occurred()) { - // We get here if we fell of the end of the run() function - // raising an exception. The switch itself was - // successful, but the function raised. - // valgrind reports that memory allocated here can still - // be reached after a test run. - throw PyErrOccurred::from_current(); - } - return result; - } - catch (const PyErrOccurred&) { - /* Turn switch errors into switch throws */ - /* Turn trace errors into switch throws */ - this->release_args(); - throw; - } -} - -void -Greenlet::g_calltrace(const OwnedObject& tracefunc, - const greenlet::refs::ImmortalEventName& event, - const BorrowedGreenlet& origin, - const BorrowedGreenlet& target) -{ - PyErrPieces saved_exc; - try { - TracingGuard tracing_guard; - // TODO: We have saved the active exception (if any) that's - // about to be raised. In the 'throw' case, we could provide - // the exception to the tracefunction, which seems very helpful. - tracing_guard.CallTraceFunction(tracefunc, event, origin, target); - } - catch (const PyErrOccurred&) { - // In case of exceptions trace function is removed, - // and any existing exception is replaced with the tracing - // exception. - GET_THREAD_STATE().state().set_tracefunc(Py_None); - throw; - } - - saved_exc.PyErrRestore(); - assert( - (event == mod_globs->event_throw && PyErr_Occurred()) - || (event == mod_globs->event_switch && !PyErr_Occurred()) - ); -} - -void -Greenlet::murder_in_place() -{ - if (this->active()) { - assert(!this->is_currently_running_in_some_thread()); - this->deactivate_and_free(); - } -} - -inline void -Greenlet::deactivate_and_free() -{ - if (!this->active()) { - return; - } - // Throw away any saved stack. - this->stack_state = StackState(); - assert(!this->stack_state.active()); - // Throw away any Python references. - // We're holding a borrowed reference to the last - // frame we executed. Since we borrowed it, the - // normal traversal, clear, and dealloc functions - // ignore it, meaning it leaks. (The thread state - // object can't find it to clear it when that's - // deallocated either, because by definition if we - // got an object on this list, it wasn't - // running and the thread state doesn't have - // this frame.) - // So here, we *do* clear it. - this->python_state.tp_clear(true); -} - -bool -Greenlet::belongs_to_thread(const ThreadState* thread_state) const -{ - if (!this->thread_state() // not running anywhere, or thread - // exited - || !thread_state) { // same, or there is no thread state. - return false; - } - return true; -} - - -void -Greenlet::deallocing_greenlet_in_thread(const ThreadState* current_thread_state) -{ - /* Cannot raise an exception to kill the greenlet if - it is not running in the same thread! */ - if (this->belongs_to_thread(current_thread_state)) { - assert(current_thread_state); - // To get here it had to have run before - /* Send the greenlet a GreenletExit exception. */ - - // We don't care about the return value, only whether an - // exception happened. - this->throw_GreenletExit_during_dealloc(*current_thread_state); - return; - } - - // Not the same thread! Temporarily save the greenlet - // into its thread's deleteme list, *if* it exists. - // If that thread has already exited, and processed its pending - // cleanup, we'll never be able to clean everything up: we won't - // be able to raise an exception. - // That's mostly OK! Since we can't add it to a list, our refcount - // won't increase, and we'll go ahead with the DECREFs later. - - ThreadState *const thread_state = this->thread_state(); - if (thread_state) { - thread_state->delete_when_thread_running(this->self()); - } - else { - // The thread is dead, we can't raise an exception. - // We need to make it look non-active, though, so that dealloc - // finishes killing it. - this->deactivate_and_free(); - } - return; -} - - -int -Greenlet::tp_traverse(visitproc visit, void* arg) -{ - - int result; - if ((result = this->exception_state.tp_traverse(visit, arg)) != 0) { - return result; - } - //XXX: This is ugly. But so is handling everything having to do - //with the top frame. - bool visit_top_frame = this->was_running_in_dead_thread(); - // When true, the thread is dead. Our implicit weak reference to the - // frame is now all that's left; we consider ourselves to - // strongly own it now. - if ((result = this->python_state.tp_traverse(visit, arg, visit_top_frame)) != 0) { - return result; - } - return 0; -} - -int -Greenlet::tp_clear() -{ - bool own_top_frame = this->was_running_in_dead_thread(); - this->exception_state.tp_clear(); - this->python_state.tp_clear(own_top_frame); - return 0; -} - -bool Greenlet::is_currently_running_in_some_thread() const -{ - return this->stack_state.active() && !this->python_state.top_frame(); -} - -#if GREENLET_PY312 -void GREENLET_NOINLINE(Greenlet::expose_frames)() -{ - if (!this->python_state.top_frame()) { - return; - } - - _PyInterpreterFrame* last_complete_iframe = nullptr; - _PyInterpreterFrame* iframe = this->python_state.top_frame()->f_frame; - while (iframe) { - // We must make a copy before looking at the iframe contents, - // since iframe might point to a portion of the greenlet's C stack - // that was spilled when switching greenlets. - _PyInterpreterFrame iframe_copy; - this->stack_state.copy_from_stack(&iframe_copy, iframe, sizeof(*iframe)); - if (!_PyFrame_IsIncomplete(&iframe_copy)) { - // If the iframe were OWNED_BY_CSTACK then it would always be - // incomplete. Since it's not incomplete, it's not on the C stack - // and we can access it through the original `iframe` pointer - // directly. This is important since GetFrameObject might - // lazily _create_ the frame object and we don't want the - // interpreter to lose track of it. - // - #if !GREENLET_PY315 - // This enum value was removed in - // https://github.com/python/cpython/pull/141108 - - assert(iframe_copy.owner != FRAME_OWNED_BY_CSTACK); - #endif - - // We really want to just write: - // PyFrameObject* frame = _PyFrame_GetFrameObject(iframe); - // but _PyFrame_GetFrameObject calls _PyFrame_MakeAndSetFrameObject - // which is not a visible symbol in libpython. The easiest - // way to get a public function to call it is using - // PyFrame_GetBack, which is defined as follows: - // assert(frame != NULL); - // assert(!_PyFrame_IsIncomplete(frame->f_frame)); - // PyFrameObject *back = frame->f_back; - // if (back == NULL) { - // _PyInterpreterFrame *prev = frame->f_frame->previous; - // prev = _PyFrame_GetFirstComplete(prev); - // if (prev) { - // back = _PyFrame_GetFrameObject(prev); - // } - // } - // return (PyFrameObject*)Py_XNewRef(back); - if (!iframe->frame_obj) { - PyFrameObject dummy_frame; - _PyInterpreterFrame dummy_iframe; - dummy_frame.f_back = nullptr; - dummy_frame.f_frame = &dummy_iframe; - // force the iframe to be considered complete without - // needing to check its code object: - dummy_iframe.owner = FRAME_OWNED_BY_GENERATOR; - dummy_iframe.previous = iframe; - assert(!_PyFrame_IsIncomplete(&dummy_iframe)); - // Drop the returned reference immediately; the iframe - // continues to hold a strong reference - Py_XDECREF(PyFrame_GetBack(&dummy_frame)); - assert(iframe->frame_obj); - } - - // This is a complete frame, so make the last one of those we saw - // point at it, bypassing any incomplete frames (which may have - // been on the C stack) in between the two. We're overwriting - // last_complete_iframe->previous and need that to be reversible, - // so we store the original previous ptr in the frame object - // (which we must have created on a previous iteration through - // this loop). The frame object has a bunch of storage that is - // only used when its iframe is OWNED_BY_FRAME_OBJECT, which only - // occurs when the frame object outlives the frame's execution, - // which can't have happened yet because the frame is currently - // executing as far as the interpreter is concerned. So, we can - // reuse it for our own purposes. - assert(iframe->owner == FRAME_OWNED_BY_THREAD - || iframe->owner == FRAME_OWNED_BY_GENERATOR); - if (last_complete_iframe) { - assert(last_complete_iframe->frame_obj); - memcpy(&last_complete_iframe->frame_obj->_f_frame_data[0], - &last_complete_iframe->previous, sizeof(void *)); - last_complete_iframe->previous = iframe; - } - last_complete_iframe = iframe; - } - // Frames that are OWNED_BY_FRAME_OBJECT are linked via the - // frame's f_back while all others are linked via the iframe's - // previous ptr. Since all the frames we traverse are running - // as far as the interpreter is concerned, we don't have to - // worry about the OWNED_BY_FRAME_OBJECT case. - iframe = iframe_copy.previous; - } - - // Give the outermost complete iframe a null previous pointer to - // account for any potential incomplete/C-stack iframes between it - // and the actual top-of-stack - if (last_complete_iframe) { - assert(last_complete_iframe->frame_obj); - memcpy(&last_complete_iframe->frame_obj->_f_frame_data[0], - &last_complete_iframe->previous, sizeof(void *)); - last_complete_iframe->previous = nullptr; - } -} -#else -void Greenlet::expose_frames() -{ - -} -#endif - -}; // namespace greenlet -#endif diff --git a/.env/Lib/site-packages/greenlet/TGreenlet.hpp b/.env/Lib/site-packages/greenlet/TGreenlet.hpp deleted file mode 100644 index 32330e9..0000000 --- a/.env/Lib/site-packages/greenlet/TGreenlet.hpp +++ /dev/null @@ -1,837 +0,0 @@ -#ifndef GREENLET_GREENLET_HPP -#define GREENLET_GREENLET_HPP -/* - * Declarations of the core data structures. -*/ - -#define PY_SSIZE_T_CLEAN -#include - -#include "greenlet_compiler_compat.hpp" -#include "greenlet_refs.hpp" -#include "greenlet_cpython_compat.hpp" -#include "greenlet_allocator.hpp" - -using greenlet::refs::OwnedObject; -using greenlet::refs::OwnedGreenlet; -using greenlet::refs::OwnedMainGreenlet; -using greenlet::refs::BorrowedGreenlet; - -#if PY_VERSION_HEX < 0x30B00A6 -# define _PyCFrame CFrame -# define _PyInterpreterFrame _interpreter_frame -#endif - -#if GREENLET_PY312 -# define Py_BUILD_CORE -# include "internal/pycore_frame.h" -#endif - -#if GREENLET_PY314 -# include "internal/pycore_interpframe_structs.h" -#if defined(_MSC_VER) || defined(__MINGW64__) -# include "greenlet_msvc_compat.hpp" -#else -# include "internal/pycore_interpframe.h" -#endif -#ifdef Py_GIL_DISABLED -# include "internal/pycore_tstate.h" -#endif -#endif - -// XXX: TODO: Work to remove all virtual functions -// for speed of calling and size of objects (no vtable). -// One pattern is the Curiously Recurring Template -namespace greenlet -{ - class ExceptionState - { - private: - G_NO_COPIES_OF_CLS(ExceptionState); - - // Even though these are borrowed objects, we actually own - // them, when they're not null. - // XXX: Express that in the API. - private: - _PyErr_StackItem* exc_info; - _PyErr_StackItem exc_state; - public: - ExceptionState(); - void operator<<(const PyThreadState *const tstate) noexcept; - void operator>>(PyThreadState* tstate) noexcept; - void clear() noexcept; - - int tp_traverse(visitproc visit, void* arg) noexcept; - void tp_clear() noexcept; - }; - - template - void operator<<(const PyThreadState *const tstate, T& exc); - - class PythonStateContext - { - protected: - greenlet::refs::OwnedContext _context; - public: - inline const greenlet::refs::OwnedContext& context() const - { - return this->_context; - } - inline greenlet::refs::OwnedContext& context() - { - return this->_context; - } - - inline void tp_clear() - { - this->_context.CLEAR(); - } - - template - inline static PyObject* context(T* tstate) - { - return tstate->context; - } - - template - inline static void context(T* tstate, PyObject* new_context) - { - tstate->context = new_context; - tstate->context_ver++; - } - }; - class SwitchingArgs; - class PythonState : public PythonStateContext - { - public: - typedef greenlet::refs::OwnedReference OwnedFrame; - private: - G_NO_COPIES_OF_CLS(PythonState); - // We own this if we're suspended (although currently we don't - // tp_traverse into it; that's a TODO). If we're running, it's - // empty. If we get deallocated and *still* have a frame, it - // won't be reachable from the place that normally decref's - // it, so we need to do it (hence owning it). - OwnedFrame _top_frame; -#if GREENLET_USE_CFRAME - _PyCFrame* cframe; - int use_tracing; -#endif -#if GREENLET_PY314 - int py_recursion_depth; - // I think this is only used by the JIT. At least, - // we only got errors not switching it when the JIT was enabled. - // Python/generated_cases.c.h:12469: _PyEval_EvalFrameDefault: - // Assertion `tstate->current_executor == NULL' failed. - // see https://github.com/python-greenlet/greenlet/issues/460 - PyObject* current_executor; - _PyStackRef* stackpointer; - #ifdef Py_GIL_DISABLED - _PyCStackRef* c_stack_refs; - #endif -#elif GREENLET_PY312 - int py_recursion_depth; - int c_recursion_depth; -#else - int recursion_depth; -#endif -#if GREENLET_PY313 - PyObject *delete_later; -#else - int trash_delete_nesting; -#endif -#if GREENLET_PY311 - _PyInterpreterFrame* current_frame; - _PyStackChunk* datastack_chunk; - PyObject** datastack_top; - PyObject** datastack_limit; -#endif - // The PyInterpreterFrame list on 3.12+ contains some entries that are - // on the C stack, which can't be directly accessed while a greenlet is - // suspended. In order to keep greenlet gr_frame introspection working, - // we adjust stack switching to rewrite the interpreter frame list - // to skip these C-stack frames; we call this "exposing" the greenlet's - // frames because it makes them valid to work with in Python. Then when - // the greenlet is resumed we need to remember to reverse the operation - // we did. The C-stack frames are "entry frames" which are a low-level - // interpreter detail; they're not needed for introspection, but do - // need to be present for the eval loop to work. - void unexpose_frames(); - - public: - - PythonState(); - // You can use this for testing whether we have a frame - // or not. It returns const so they can't modify it. - const OwnedFrame& top_frame() const noexcept; - - inline void operator<<(const PyThreadState *const tstate) noexcept; - inline void operator>>(PyThreadState* tstate) noexcept; - void clear() noexcept; - - int tp_traverse(visitproc visit, void* arg, bool visit_top_frame) noexcept; - void tp_clear(bool own_top_frame) noexcept; - void set_initial_state(const PyThreadState* const tstate) noexcept; -#if GREENLET_USE_CFRAME - void set_new_cframe(_PyCFrame& frame) noexcept; -#endif - - void may_switch_away() noexcept; - inline void will_switch_from(PyThreadState *const origin_tstate) noexcept; - void did_finish(PyThreadState* tstate) noexcept; - }; - - class StackState - { - // By having only plain C (POD) members, no virtual functions - // or bases, we get a trivial assignment operator generated - // for us. However, that's not safe since we do manage memory. - // So we declare an assignment operator that only works if we - // don't have any memory allocated. (We don't use - // std::shared_ptr for reference counting just to keep this - // object small) - private: - char* _stack_start; - char* stack_stop; - char* stack_copy; - intptr_t _stack_saved; - StackState* stack_prev; - inline int copy_stack_to_heap_up_to(const char* const stop) noexcept; - inline void free_stack_copy() noexcept; - - public: - /** - * Creates a started, but inactive, state, using *current* - * as the previous. - */ - StackState(void* mark, StackState& current); - /** - * Creates an inactive, unstarted, state. - */ - StackState(); - ~StackState(); - StackState(const StackState& other); - StackState& operator=(const StackState& other); - inline void copy_heap_to_stack(const StackState& current) noexcept; - inline int copy_stack_to_heap(char* const stackref, const StackState& current) noexcept; - inline bool started() const noexcept; - inline bool main() const noexcept; - inline bool active() const noexcept; - inline void set_active() noexcept; - inline void set_inactive() noexcept; - inline intptr_t stack_saved() const noexcept; - inline char* stack_start() const noexcept; - static inline StackState make_main() noexcept; -#ifdef GREENLET_USE_STDIO - friend std::ostream& operator<<(std::ostream& os, const StackState& s); -#endif - - // Fill in [dest, dest + n) with the values that would be at - // [src, src + n) while this greenlet is running. This is like memcpy - // except that if the greenlet is suspended it accounts for the portion - // of the greenlet's stack that was spilled to the heap. `src` may - // be on this greenlet's stack, or on the heap, but not on a different - // greenlet's stack. - void copy_from_stack(void* dest, const void* src, size_t n) const; - }; -#ifdef GREENLET_USE_STDIO - std::ostream& operator<<(std::ostream& os, const StackState& s); -#endif - - class SwitchingArgs - { - private: - G_NO_ASSIGNMENT_OF_CLS(SwitchingArgs); - // If args and kwargs are both false (NULL), this is a *throw*, not a - // switch. PyErr_... must have been called already. - OwnedObject _args; - OwnedObject _kwargs; - public: - - SwitchingArgs() - {} - - SwitchingArgs(const OwnedObject& args, const OwnedObject& kwargs) - : _args(args), - _kwargs(kwargs) - {} - - SwitchingArgs(const SwitchingArgs& other) - : _args(other._args), - _kwargs(other._kwargs) - {} - - const OwnedObject& args() - { - return this->_args; - } - - const OwnedObject& kwargs() - { - return this->_kwargs; - } - - /** - * Moves ownership from the argument to this object. - */ - SwitchingArgs& operator<<=(SwitchingArgs& other) - { - if (this != &other) { - this->_args = other._args; - this->_kwargs = other._kwargs; - other.CLEAR(); - } - return *this; - } - - /** - * Acquires ownership of the argument (consumes the reference). - */ - SwitchingArgs& operator<<=(PyObject* args) - { - this->_args = OwnedObject::consuming(args); - this->_kwargs.CLEAR(); - return *this; - } - - /** - * Acquires ownership of the argument. - * - * Sets the args to be the given value; clears the kwargs. - */ - SwitchingArgs& operator<<=(OwnedObject& args) - { - assert(&args != &this->_args); - this->_args = args; - this->_kwargs.CLEAR(); - args.CLEAR(); - - return *this; - } - - explicit operator bool() const noexcept - { - return this->_args || this->_kwargs; - } - - inline void CLEAR() - { - this->_args.CLEAR(); - this->_kwargs.CLEAR(); - } - - const std::string as_str() const noexcept - { - return PyUnicode_AsUTF8( - OwnedObject::consuming( - PyUnicode_FromFormat( - "SwitchingArgs(args=%R, kwargs=%R)", - this->_args.borrow(), - this->_kwargs.borrow() - ) - ).borrow() - ); - } - }; - - class ThreadState; - - class UserGreenlet; - class MainGreenlet; - - class Greenlet - { - private: - G_NO_COPIES_OF_CLS(Greenlet); - PyGreenlet* const _self; - private: - // XXX: Work to remove these. - friend class ThreadState; - friend class UserGreenlet; - friend class MainGreenlet; - protected: - ExceptionState exception_state; - SwitchingArgs switch_args; - StackState stack_state; - PythonState python_state; - Greenlet(PyGreenlet* p, const StackState& initial_state); - public: - // This constructor takes ownership of the PyGreenlet, by - // setting ``p->pimpl = this;``. - Greenlet(PyGreenlet* p); - virtual ~Greenlet(); - - const OwnedObject context() const; - - // You MUST call this _very_ early in the switching process to - // prepare anything that may need prepared. This might perform - // garbage collections or otherwise run arbitrary Python code. - // - // One specific use of it is for Python 3.11+, preventing - // running arbitrary code at unsafe times. See - // PythonState::may_switch_away(). - inline void may_switch_away() - { - this->python_state.may_switch_away(); - } - - inline void context(refs::BorrowedObject new_context); - - inline SwitchingArgs& args() - { - return this->switch_args; - } - - virtual const refs::BorrowedMainGreenlet main_greenlet() const = 0; - - inline intptr_t stack_saved() const noexcept - { - return this->stack_state.stack_saved(); - } - - // This is used by the macro SLP_SAVE_STATE to compute the - // difference in stack sizes. It might be nice to handle the - // computation ourself, but the type of the result - // varies by platform, so doing it in the macro is the - // simplest way. - inline const char* stack_start() const noexcept - { - return this->stack_state.stack_start(); - } - - virtual OwnedObject throw_GreenletExit_during_dealloc(const ThreadState& current_thread_state); - virtual OwnedObject g_switch() = 0; - /** - * Force the greenlet to appear dead. Used when it's not - * possible to throw an exception into a greenlet anymore. - * - * This losses access to the thread state and the main greenlet. - */ - virtual void murder_in_place(); - - /** - * Called when somebody notices we were running in a dead - * thread to allow cleaning up resources (because we can't - * raise GreenletExit into it anymore). - * This is very similar to ``murder_in_place()``, except that - * it DOES NOT lose the main greenlet or thread state. - */ - inline void deactivate_and_free(); - - - // Called when some thread wants to deallocate a greenlet - // object. - // The thread may or may not be the same thread the greenlet - // was running in. - // The thread state will be null if the thread the greenlet - // was running in was known to have exited. - void deallocing_greenlet_in_thread(const ThreadState* current_state); - - // Must be called on 3.12+ before exposing a suspended greenlet's - // frames to user code. This rewrites the linked list of interpreter - // frames to skip the ones that are being stored on the C stack (which - // can't be safely accessed while the greenlet is suspended because - // that stack space might be hosting a different greenlet), and - // sets PythonState::frames_were_exposed so we remember to restore - // the original list before resuming the greenlet. The C-stack frames - // are a low-level interpreter implementation detail; while they're - // important to the bytecode eval loop, they're superfluous for - // introspection purposes. - void expose_frames(); - - - // TODO: Figure out how to make these non-public. - inline void slp_restore_state() noexcept; - inline int slp_save_state(char *const stackref) noexcept; - - inline bool is_currently_running_in_some_thread() const; - virtual bool belongs_to_thread(const ThreadState* state) const; - - inline bool started() const - { - return this->stack_state.started(); - } - inline bool active() const - { - return this->stack_state.active(); - } - inline bool main() const - { - return this->stack_state.main(); - } - virtual refs::BorrowedMainGreenlet find_main_greenlet_in_lineage() const = 0; - - virtual const OwnedGreenlet parent() const = 0; - virtual void parent(const refs::BorrowedObject new_parent) = 0; - - inline const PythonState::OwnedFrame& top_frame() - { - return this->python_state.top_frame(); - } - - virtual const OwnedObject& run() const = 0; - virtual void run(const refs::BorrowedObject nrun) = 0; - - - virtual int tp_traverse(visitproc visit, void* arg); - virtual int tp_clear(); - - - // Return the thread state that the greenlet is running in, or - // null if the greenlet is not running or the thread is known - // to have exited. - virtual ThreadState* thread_state() const noexcept = 0; - - // Return true if the greenlet is known to have been running - // (active) in a thread that has now exited. - virtual bool was_running_in_dead_thread() const noexcept = 0; - - // Return a borrowed greenlet that is the Python object - // this object represents. - inline BorrowedGreenlet self() const noexcept - { - return BorrowedGreenlet(this->_self); - } - - // For testing. If this returns true, we should pretend that - // slp_switch() failed. - virtual bool force_slp_switch_error() const noexcept; - - protected: - inline void release_args(); - - // The functions that must not be inlined are declared virtual. - // We also mark them as protected, not private, so that the - // compiler is forced to call them through a function pointer. - // (A sufficiently smart compiler could directly call a private - // virtual function since it can never be overridden in a - // subclass). - - // Also TODO: Switch away from integer error codes and to enums, - // or throw exceptions when possible. - struct switchstack_result_t - { - int status; - Greenlet* the_new_current_greenlet; - OwnedGreenlet origin_greenlet; - - switchstack_result_t() - : status(0), - the_new_current_greenlet(nullptr) - {} - - switchstack_result_t(int err) - : status(err), - the_new_current_greenlet(nullptr) - {} - - switchstack_result_t(int err, Greenlet* state, OwnedGreenlet& origin) - : status(err), - the_new_current_greenlet(state), - origin_greenlet(origin) - { - } - - switchstack_result_t(int err, Greenlet* state, const BorrowedGreenlet& origin) - : status(err), - the_new_current_greenlet(state), - origin_greenlet(origin) - { - } - - switchstack_result_t(const switchstack_result_t& other) - : status(other.status), - the_new_current_greenlet(other.the_new_current_greenlet), - origin_greenlet(other.origin_greenlet) - {} - - switchstack_result_t& operator=(const switchstack_result_t& other) - { - this->status = other.status; - this->the_new_current_greenlet = other.the_new_current_greenlet; - this->origin_greenlet = other.origin_greenlet; - return *this; - } - }; - - OwnedObject on_switchstack_or_initialstub_failure( - Greenlet* target, - const switchstack_result_t& err, - const bool target_was_me=false, - const bool was_initial_stub=false); - - // Returns the previous greenlet we just switched away from. - virtual OwnedGreenlet g_switchstack_success() noexcept; - - - // Check the preconditions for switching to this greenlet; if they - // aren't met, throws PyErrOccurred. Most callers will want to - // catch this and clear the arguments - inline void check_switch_allowed() const; - class GreenletStartedWhileInPython : public std::runtime_error - { - public: - GreenletStartedWhileInPython() : std::runtime_error("") - {} - }; - - protected: - - - /** - Perform a stack switch into this greenlet. - - This temporarily sets the global variable - ``switching_thread_state`` to this greenlet; as soon as the - call to ``slp_switch`` completes, this is reset to NULL. - Consequently, this depends on the GIL. - - TODO: Adopt the stackman model and pass ``slp_switch`` a - callback function and context pointer; this eliminates the - need for global variables altogether. - - Because the stack switch happens in this function, this - function can't use its own stack (local) variables, set - before the switch, and then accessed after the switch. - - Further, you con't even access ``g_thread_state_global`` - before and after the switch from the global variable. - Because it is thread local some compilers cache it in a - register/on the stack, notably new versions of MSVC; this - breaks with strange crashes sometime later, because writing - to anything in ``g_thread_state_global`` after the switch - is actually writing to random memory. For this reason, we - call a non-inlined function to finish the operation. (XXX: - The ``/GT`` MSVC compiler argument probably fixes that.) - - It is very important that stack switch is 'atomic', i.e. no - calls into other Python code allowed (except very few that - are safe), because global variables are very fragile. (This - should no longer be the case with thread-local variables.) - - */ - // Made virtual to facilitate subclassing UserGreenlet for testing. - virtual switchstack_result_t g_switchstack(void); - -class TracingGuard -{ -private: - PyThreadState* tstate; -public: - TracingGuard() - : tstate(PyThreadState_GET()) - { - PyThreadState_EnterTracing(this->tstate); - } - - ~TracingGuard() - { - PyThreadState_LeaveTracing(this->tstate); - this->tstate = nullptr; - } - - inline void CallTraceFunction(const OwnedObject& tracefunc, - const greenlet::refs::ImmortalEventName& event, - const BorrowedGreenlet& origin, - const BorrowedGreenlet& target) - { - // TODO: This calls tracefunc(event, (origin, target)). Add a shortcut - // function for that that's specialized to avoid the Py_BuildValue - // string parsing, or start with just using "ON" format with PyTuple_Pack(2, - // origin, target). That seems like what the N format is meant - // for. - // XXX: Why does event not automatically cast back to a PyObject? - // It tries to call the "deleted constructor ImmortalEventName - // const" instead. - assert(tracefunc); - assert(event); - assert(origin); - assert(target); - greenlet::refs::NewReference retval( - PyObject_CallFunction( - tracefunc.borrow(), - "O(OO)", - event.borrow(), - origin.borrow(), - target.borrow() - )); - if (!retval) { - throw PyErrOccurred::from_current(); - } - } -}; - - static void - g_calltrace(const OwnedObject& tracefunc, - const greenlet::refs::ImmortalEventName& event, - const greenlet::refs::BorrowedGreenlet& origin, - const BorrowedGreenlet& target); - private: - OwnedObject g_switch_finish(const switchstack_result_t& err); - - }; - - class UserGreenlet : public Greenlet - { - private: - static greenlet::PythonAllocator allocator; - OwnedMainGreenlet _main_greenlet; - OwnedObject _run_callable; - OwnedGreenlet _parent; - public: - static void* operator new(size_t UNUSED(count)); - static void operator delete(void* ptr); - - UserGreenlet(PyGreenlet* p, BorrowedGreenlet the_parent); - virtual ~UserGreenlet(); - - virtual refs::BorrowedMainGreenlet find_main_greenlet_in_lineage() const; - virtual bool was_running_in_dead_thread() const noexcept; - virtual ThreadState* thread_state() const noexcept; - virtual OwnedObject g_switch(); - virtual const OwnedObject& run() const - { - if (this->started() || !this->_run_callable) { - throw AttributeError("run"); - } - return this->_run_callable; - } - virtual void run(const refs::BorrowedObject nrun); - - virtual const OwnedGreenlet parent() const; - virtual void parent(const refs::BorrowedObject new_parent); - - virtual const refs::BorrowedMainGreenlet main_greenlet() const; - - virtual void murder_in_place(); - virtual bool belongs_to_thread(const ThreadState* state) const; - virtual int tp_traverse(visitproc visit, void* arg); - virtual int tp_clear(); - class ParentIsCurrentGuard - { - private: - OwnedGreenlet oldparent; - UserGreenlet* greenlet; - G_NO_COPIES_OF_CLS(ParentIsCurrentGuard); - public: - ParentIsCurrentGuard(UserGreenlet* p, const ThreadState& thread_state); - ~ParentIsCurrentGuard(); - }; - virtual OwnedObject throw_GreenletExit_during_dealloc(const ThreadState& current_thread_state); - protected: - virtual switchstack_result_t g_initialstub(void* mark); - private: - // This function isn't meant to return. - // This accepts raw pointers and the ownership of them at the - // same time. The caller should use ``inner_bootstrap(origin.relinquish_ownership())``. - void inner_bootstrap(PyGreenlet* origin_greenlet, PyObject* run); - }; - - class BrokenGreenlet : public UserGreenlet - { - private: - static greenlet::PythonAllocator allocator; - public: - bool _force_switch_error = false; - bool _force_slp_switch_error = false; - - static void* operator new(size_t UNUSED(count)); - static void operator delete(void* ptr); - BrokenGreenlet(PyGreenlet* p, BorrowedGreenlet the_parent) - : UserGreenlet(p, the_parent) - {} - virtual ~BrokenGreenlet() - {} - - virtual switchstack_result_t g_switchstack(void); - virtual bool force_slp_switch_error() const noexcept; - - }; - - class MainGreenlet : public Greenlet - { - private: - static greenlet::PythonAllocator allocator; - refs::BorrowedMainGreenlet _self; - ThreadState* _thread_state; - G_NO_COPIES_OF_CLS(MainGreenlet); - public: - static void* operator new(size_t UNUSED(count)); - static void operator delete(void* ptr); - - MainGreenlet(refs::BorrowedMainGreenlet::PyType*, ThreadState*); - virtual ~MainGreenlet(); - - - virtual const OwnedObject& run() const; - virtual void run(const refs::BorrowedObject nrun); - - virtual const OwnedGreenlet parent() const; - virtual void parent(const refs::BorrowedObject new_parent); - - virtual const refs::BorrowedMainGreenlet main_greenlet() const; - - virtual refs::BorrowedMainGreenlet find_main_greenlet_in_lineage() const; - virtual bool was_running_in_dead_thread() const noexcept; - virtual ThreadState* thread_state() const noexcept; - void thread_state(ThreadState*) noexcept; - virtual OwnedObject g_switch(); - virtual int tp_traverse(visitproc visit, void* arg); - }; - - // Instantiate one on the stack to save the GC state, - // and then disable GC. When it goes out of scope, GC will be - // restored to its original state. Sadly, these APIs are only - // available on 3.10+; luckily, we only need them on 3.11+. -#if GREENLET_PY310 - class GCDisabledGuard - { - private: - int was_enabled = 0; - public: - GCDisabledGuard() - : was_enabled(PyGC_IsEnabled()) - { - PyGC_Disable(); - } - - ~GCDisabledGuard() - { - if (this->was_enabled) { - PyGC_Enable(); - } - } - }; -#endif - - OwnedObject& operator<<=(OwnedObject& lhs, greenlet::SwitchingArgs& rhs) noexcept; - - //TODO: Greenlet::g_switch() should call this automatically on its - //return value. As it is, the module code is calling it. - static inline OwnedObject - single_result(const OwnedObject& results) - { - if (results - && PyTuple_Check(results.borrow()) - && PyTuple_GET_SIZE(results.borrow()) == 1) { - PyObject* result = PyTuple_GET_ITEM(results.borrow(), 0); - assert(result); - return OwnedObject::owning(result); - } - return results; - } - - - static OwnedObject - g_handle_exit(const OwnedObject& greenlet_result); - - - template - void operator<<(const PyThreadState *const lhs, T& rhs) - { - rhs.operator<<(lhs); - } - -} // namespace greenlet ; - -#endif diff --git a/.env/Lib/site-packages/greenlet/TGreenletGlobals.cpp b/.env/Lib/site-packages/greenlet/TGreenletGlobals.cpp deleted file mode 100644 index 0087d2f..0000000 --- a/.env/Lib/site-packages/greenlet/TGreenletGlobals.cpp +++ /dev/null @@ -1,94 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of GreenletGlobals. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#ifndef T_GREENLET_GLOBALS -#define T_GREENLET_GLOBALS - -#include "greenlet_refs.hpp" -#include "greenlet_exceptions.hpp" -#include "greenlet_thread_support.hpp" -#include "greenlet_internal.hpp" - -namespace greenlet { - -// This encapsulates what were previously module global "constants" -// established at init time. -// This is a step towards Python3 style module state that allows -// reloading. -// -// In an earlier iteration of this code, we used placement new to be -// able to allocate this object statically still, so that references -// to its members don't incur an extra pointer indirection. -// But under some scenarios, that could result in crashes at -// shutdown because apparently the destructor was getting run twice? -class GreenletGlobals -{ - -public: - const greenlet::refs::ImmortalEventName event_switch; - const greenlet::refs::ImmortalEventName event_throw; - const greenlet::refs::ImmortalException PyExc_GreenletError; - const greenlet::refs::ImmortalException PyExc_GreenletExit; - const greenlet::refs::ImmortalObject empty_tuple; - const greenlet::refs::ImmortalObject empty_dict; - const greenlet::refs::ImmortalString str_run; - Mutex* const thread_states_to_destroy_lock; - greenlet::cleanup_queue_t thread_states_to_destroy; - - GreenletGlobals() : - event_switch("switch"), - event_throw("throw"), - PyExc_GreenletError("greenlet.error"), - PyExc_GreenletExit("greenlet.GreenletExit", PyExc_BaseException), - empty_tuple(Require(PyTuple_New(0))), - empty_dict(Require(PyDict_New())), - str_run("run"), - thread_states_to_destroy_lock(new Mutex()) - {} - - ~GreenletGlobals() - { - // This object is (currently) effectively immortal, and not - // just because of those placement new tricks; if we try to - // deallocate the static object we allocated, and overwrote, - // we would be doing so at C++ teardown time, which is after - // the final Python GIL is released, and we can't use the API - // then. - // (The members will still be destructed, but they also don't - // do any deallocation.) - } - - void queue_to_destroy(ThreadState* ts) const - { - // we're currently accessed through a static const object, - // implicitly marking our members as const, so code can't just - // call push_back (or pop_back) without casting away the - // const. - // - // Do that for callers. - greenlet::cleanup_queue_t& q = const_cast(this->thread_states_to_destroy); - q.push_back(ts); - } - - ThreadState* take_next_to_destroy() const - { - greenlet::cleanup_queue_t& q = const_cast(this->thread_states_to_destroy); - ThreadState* result = q.back(); - q.pop_back(); - return result; - } -}; - -}; // namespace greenlet - -static const greenlet::GreenletGlobals* mod_globs; - -#endif // T_GREENLET_GLOBALS diff --git a/.env/Lib/site-packages/greenlet/TMainGreenlet.cpp b/.env/Lib/site-packages/greenlet/TMainGreenlet.cpp deleted file mode 100644 index ee01481..0000000 --- a/.env/Lib/site-packages/greenlet/TMainGreenlet.cpp +++ /dev/null @@ -1,160 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of greenlet::MainGreenlet. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#ifndef T_MAIN_GREENLET_CPP -#define T_MAIN_GREENLET_CPP - -#include "TGreenlet.hpp" - -#ifdef Py_GIL_DISABLED -#include -#endif - -// Incremented when we create a main greenlet, in a new thread, decremented -// when it is destroyed. -#ifdef Py_GIL_DISABLED -static std::atomic G_TOTAL_MAIN_GREENLETS(0); -#else -// Protected by the GIL. -static Py_ssize_t G_TOTAL_MAIN_GREENLETS; -#endif - -namespace greenlet { -greenlet::PythonAllocator MainGreenlet::allocator; - -void* MainGreenlet::operator new(size_t UNUSED(count)) -{ - return allocator.allocate(1); -} - - -void MainGreenlet::operator delete(void* ptr) -{ - return allocator.deallocate(static_cast(ptr), - 1); -} - - -MainGreenlet::MainGreenlet(PyGreenlet* p, ThreadState* state) - : Greenlet(p, StackState::make_main()), - _self(p), - _thread_state(state) -{ - G_TOTAL_MAIN_GREENLETS++; -} - -MainGreenlet::~MainGreenlet() -{ - G_TOTAL_MAIN_GREENLETS--; - this->tp_clear(); -} - -ThreadState* -MainGreenlet::thread_state() const noexcept -{ - return this->_thread_state; -} - -void -MainGreenlet::thread_state(ThreadState* t) noexcept -{ - assert(!t); - this->_thread_state = t; -} - - -const BorrowedMainGreenlet -MainGreenlet::main_greenlet() const -{ - return this->_self; -} - -BorrowedMainGreenlet -MainGreenlet::find_main_greenlet_in_lineage() const -{ - return BorrowedMainGreenlet(this->_self); -} - -bool -MainGreenlet::was_running_in_dead_thread() const noexcept -{ - return !this->_thread_state; -} - -OwnedObject -MainGreenlet::g_switch() -{ - try { - this->check_switch_allowed(); - } - catch (const PyErrOccurred&) { - this->release_args(); - throw; - } - - switchstack_result_t err = this->g_switchstack(); - if (err.status < 0) { - // XXX: This code path is untested, but it is shared - // with the UserGreenlet path that is tested. - return this->on_switchstack_or_initialstub_failure( - this, - err, - true, // target was me - false // was initial stub - ); - } - - return err.the_new_current_greenlet->g_switch_finish(err); -} - -int -MainGreenlet::tp_traverse(visitproc visit, void* arg) -{ - if (this->_thread_state) { - // we've already traversed main, (self), don't do it again. - int result = this->_thread_state->tp_traverse(visit, arg, false); - if (result) { - return result; - } - } - return Greenlet::tp_traverse(visit, arg); -} - -const OwnedObject& -MainGreenlet::run() const -{ - throw AttributeError("Main greenlets do not have a run attribute."); -} - -void -MainGreenlet::run(const BorrowedObject UNUSED(nrun)) -{ - throw AttributeError("Main greenlets do not have a run attribute."); -} - -void -MainGreenlet::parent(const BorrowedObject raw_new_parent) -{ - if (!raw_new_parent) { - throw AttributeError("can't delete attribute"); - } - throw AttributeError("cannot set the parent of a main greenlet"); -} - -const OwnedGreenlet -MainGreenlet::parent() const -{ - return OwnedGreenlet(); // null becomes None -} - -}; // namespace greenlet - -#endif diff --git a/.env/Lib/site-packages/greenlet/TPythonState.cpp b/.env/Lib/site-packages/greenlet/TPythonState.cpp deleted file mode 100644 index 6375922..0000000 --- a/.env/Lib/site-packages/greenlet/TPythonState.cpp +++ /dev/null @@ -1,439 +0,0 @@ -#ifndef GREENLET_PYTHON_STATE_CPP -#define GREENLET_PYTHON_STATE_CPP - -#include -#include "TGreenlet.hpp" - -namespace greenlet { - -PythonState::PythonState() - : _top_frame() -#if GREENLET_USE_CFRAME - ,cframe(nullptr) - ,use_tracing(0) -#endif -#if GREENLET_PY314 - ,py_recursion_depth(0) - ,current_executor(nullptr) - ,stackpointer(nullptr) - #ifdef Py_GIL_DISABLED - ,c_stack_refs(nullptr) - #endif -#elif GREENLET_PY312 - ,py_recursion_depth(0) - ,c_recursion_depth(0) -#else - ,recursion_depth(0) -#endif -#if GREENLET_PY313 - ,delete_later(nullptr) -#else - ,trash_delete_nesting(0) -#endif -#if GREENLET_PY311 - ,current_frame(nullptr) - ,datastack_chunk(nullptr) - ,datastack_top(nullptr) - ,datastack_limit(nullptr) -#endif -{ -#if GREENLET_USE_CFRAME - /* - The PyThreadState->cframe pointer usually points to memory on - the stack, alloceted in a call into PyEval_EvalFrameDefault. - - Initially, before any evaluation begins, it points to the - initial PyThreadState object's ``root_cframe`` object, which is - statically allocated for the lifetime of the thread. - - A greenlet can last for longer than a call to - PyEval_EvalFrameDefault, so we can't set its ``cframe`` pointer - to be the current ``PyThreadState->cframe``; nor could we use - one from the greenlet parent for the same reason. Yet a further - no: we can't allocate one scoped to the greenlet and then - destroy it when the greenlet is deallocated, because inside the - interpreter the _PyCFrame objects form a linked list, and that too - can result in accessing memory beyond its dynamic lifetime (if - the greenlet doesn't actually finish before it dies, its entry - could still be in the list). - - Using the ``root_cframe`` is problematic, though, because its - members are never modified by the interpreter and are set to 0, - meaning that its ``use_tracing`` flag is never updated. We don't - want to modify that value in the ``root_cframe`` ourself: it - *shouldn't* matter much because we should probably never get - back to the point where that's the only cframe on the stack; - even if it did matter, the major consequence of an incorrect - value for ``use_tracing`` is that if its true the interpreter - does some extra work --- however, it's just good code hygiene. - - Our solution: before a greenlet runs, after its initial - creation, it uses the ``root_cframe`` just to have something to - put there. However, once the greenlet is actually switched to - for the first time, ``g_initialstub`` (which doesn't actually - "return" while the greenlet is running) stores a new _PyCFrame on - its local stack, and copies the appropriate values from the - currently running _PyCFrame; this is then made the _PyCFrame for the - newly-minted greenlet. ``g_initialstub`` then proceeds to call - ``glet.run()``, which results in ``PyEval_...`` adding the - _PyCFrame to the list. Switches continue as normal. Finally, when - the greenlet finishes, the call to ``glet.run()`` returns and - the _PyCFrame is taken out of the linked list and the stack value - is now unused and free to expire. - - XXX: I think we can do better. If we're deallocing in the same - thread, can't we traverse the list and unlink our frame? - Can we just keep a reference to the thread state in case we - dealloc in another thread? (Is that even possible if we're still - running and haven't returned from g_initialstub?) - */ - this->cframe = &PyThreadState_GET()->root_cframe; -#endif -} - - -inline void PythonState::may_switch_away() noexcept -{ -#if GREENLET_PY311 - // PyThreadState_GetFrame is probably going to have to allocate a - // new frame object. That may trigger garbage collection. Because - // we call this during the early phases of a switch (it doesn't - // matter to which greenlet, as this has a global effect), if a GC - // triggers a switch away, two things can happen, both bad: - // - We might not get switched back to, halting forward progress. - // this is pathological, but possible. - // - We might get switched back to with a different set of - // arguments or a throw instead of a switch. That would corrupt - // our state (specifically, PyErr_Occurred() and this->args() - // would no longer agree). - // - // Thus, when we call this API, we need to have GC disabled. - // This method serves as a bottleneck we call when maybe beginning - // a switch. In this way, it is always safe -- no risk of GC -- to - // use ``_GetFrame()`` whenever we need to, just as it was in - // <=3.10 (because subsequent calls will be cached and not - // allocate memory). - - GCDisabledGuard no_gc; - Py_XDECREF(PyThreadState_GetFrame(PyThreadState_GET())); -#endif -} - -void PythonState::operator<<(const PyThreadState *const tstate) noexcept -{ - this->_context.steal(tstate->context); -#if GREENLET_USE_CFRAME - /* - IMPORTANT: ``cframe`` is a pointer into the STACK. Thus, because - the call to ``slp_switch()`` changes the contents of the stack, - you cannot read from ``ts_current->cframe`` after that call and - necessarily get the same values you get from reading it here. - Anything you need to restore from now to then must be saved in a - global/threadlocal variable (because we can't use stack - variables here either). For things that need to persist across - the switch, use `will_switch_from`. - */ - this->cframe = tstate->cframe; - #if !GREENLET_PY312 - this->use_tracing = tstate->cframe->use_tracing; - #endif -#endif // GREENLET_USE_CFRAME -#if GREENLET_PY311 - #if GREENLET_PY314 - this->py_recursion_depth = tstate->py_recursion_limit - tstate->py_recursion_remaining; - this->current_executor = tstate->current_executor; - #ifdef Py_GIL_DISABLED - this->c_stack_refs = ((_PyThreadStateImpl*)tstate)->c_stack_refs; - #endif - #elif GREENLET_PY312 - this->py_recursion_depth = tstate->py_recursion_limit - tstate->py_recursion_remaining; - this->c_recursion_depth = Py_C_RECURSION_LIMIT - tstate->c_recursion_remaining; - #else // not 312 - this->recursion_depth = tstate->recursion_limit - tstate->recursion_remaining; - #endif // GREENLET_PY312 - #if GREENLET_PY313 - this->current_frame = tstate->current_frame; - #elif GREENLET_USE_CFRAME - this->current_frame = tstate->cframe->current_frame; - #endif - this->datastack_chunk = tstate->datastack_chunk; - this->datastack_top = tstate->datastack_top; - this->datastack_limit = tstate->datastack_limit; - - PyFrameObject *frame = PyThreadState_GetFrame((PyThreadState *)tstate); - Py_XDECREF(frame); // PyThreadState_GetFrame gives us a new - // reference. - this->_top_frame.steal(frame); - #if GREENLET_PY314 - if (this->top_frame()) { - this->stackpointer = this->_top_frame->f_frame->stackpointer; - } - else { - this->stackpointer = nullptr; - } - #endif - #if GREENLET_PY313 - this->delete_later = Py_XNewRef(tstate->delete_later); - #elif GREENLET_PY312 - this->trash_delete_nesting = tstate->trash.delete_nesting; - #else // not 312 - this->trash_delete_nesting = tstate->trash_delete_nesting; - #endif // GREENLET_PY312 -#else // Not 311 - this->recursion_depth = tstate->recursion_depth; - this->_top_frame.steal(tstate->frame); - this->trash_delete_nesting = tstate->trash_delete_nesting; -#endif // GREENLET_PY311 -} - -#if GREENLET_PY312 -void GREENLET_NOINLINE(PythonState::unexpose_frames)() -{ - if (!this->top_frame()) { - return; - } - - // See GreenletState::expose_frames() and the comment on frames_were_exposed - // for more information about this logic. - _PyInterpreterFrame *iframe = this->_top_frame->f_frame; - while (iframe != nullptr) { - _PyInterpreterFrame *prev_exposed = iframe->previous; - assert(iframe->frame_obj); - memcpy(&iframe->previous, &iframe->frame_obj->_f_frame_data[0], - sizeof(void *)); - iframe = prev_exposed; - } -} -#else -void PythonState::unexpose_frames() -{} -#endif - -void PythonState::operator>>(PyThreadState *const tstate) noexcept -{ - tstate->context = this->_context.relinquish_ownership(); - /* Incrementing this value invalidates the contextvars cache, - which would otherwise remain valid across switches */ - tstate->context_ver++; -#if GREENLET_USE_CFRAME - tstate->cframe = this->cframe; - /* - If we were tracing, we need to keep tracing. - There should never be the possibility of hitting the - root_cframe here. See note above about why we can't - just copy this from ``origin->cframe->use_tracing``. - */ - #if !GREENLET_PY312 - tstate->cframe->use_tracing = this->use_tracing; - #endif -#endif // GREENLET_USE_CFRAME -#if GREENLET_PY311 - #if GREENLET_PY314 - tstate->py_recursion_remaining = tstate->py_recursion_limit - this->py_recursion_depth; - tstate->current_executor = this->current_executor; - #ifdef Py_GIL_DISABLED - ((_PyThreadStateImpl*)tstate)->c_stack_refs = this->c_stack_refs; - #endif - this->unexpose_frames(); - #elif GREENLET_PY312 - tstate->py_recursion_remaining = tstate->py_recursion_limit - this->py_recursion_depth; - tstate->c_recursion_remaining = Py_C_RECURSION_LIMIT - this->c_recursion_depth; - this->unexpose_frames(); - #else // \/ 3.11 - tstate->recursion_remaining = tstate->recursion_limit - this->recursion_depth; - #endif // GREENLET_PY312 - #if GREENLET_PY313 - tstate->current_frame = this->current_frame; - #elif GREENLET_USE_CFRAME - tstate->cframe->current_frame = this->current_frame; - #endif - tstate->datastack_chunk = this->datastack_chunk; - tstate->datastack_top = this->datastack_top; - tstate->datastack_limit = this->datastack_limit; -#if GREENLET_PY314 && defined(Py_GIL_DISABLED) - if (this->top_frame()) { - this->_top_frame->f_frame->stackpointer = this->stackpointer; - } -#endif - this->_top_frame.relinquish_ownership(); - #if GREENLET_PY313 - Py_XDECREF(tstate->delete_later); - tstate->delete_later = this->delete_later; - Py_CLEAR(this->delete_later); - #elif GREENLET_PY312 - tstate->trash.delete_nesting = this->trash_delete_nesting; - #else // not 3.12 - tstate->trash_delete_nesting = this->trash_delete_nesting; - #endif // GREENLET_PY312 -#else // not 3.11 - tstate->frame = this->_top_frame.relinquish_ownership(); - tstate->recursion_depth = this->recursion_depth; - tstate->trash_delete_nesting = this->trash_delete_nesting; -#endif // GREENLET_PY311 -} - -inline void PythonState::will_switch_from(PyThreadState *const origin_tstate) noexcept -{ -#if GREENLET_USE_CFRAME && !GREENLET_PY312 - // The weird thing is, we don't actually save this for an - // effect on the current greenlet, it's saved for an - // effect on the target greenlet. That is, we want - // continuity of this setting across the greenlet switch. - this->use_tracing = origin_tstate->cframe->use_tracing; -#endif -} - -void PythonState::set_initial_state(const PyThreadState* const tstate) noexcept -{ - this->_top_frame = nullptr; -#if GREENLET_PY314 - this->py_recursion_depth = tstate->py_recursion_limit - tstate->py_recursion_remaining; - this->current_executor = tstate->current_executor; -#elif GREENLET_PY312 - this->py_recursion_depth = tstate->py_recursion_limit - tstate->py_recursion_remaining; - // XXX: TODO: Comment from a reviewer: - // Should this be ``Py_C_RECURSION_LIMIT - tstate->c_recursion_remaining``? - // But to me it looks more like that might not be the right - // initialization either? - this->c_recursion_depth = tstate->py_recursion_limit - tstate->py_recursion_remaining; -#elif GREENLET_PY311 - this->recursion_depth = tstate->recursion_limit - tstate->recursion_remaining; -#else - this->recursion_depth = tstate->recursion_depth; -#endif -} -// TODO: Better state management about when we own the top frame. -int PythonState::tp_traverse(visitproc visit, void* arg, bool own_top_frame) noexcept -{ - Py_VISIT(this->_context.borrow()); - if (own_top_frame) { - Py_VISIT(this->_top_frame.borrow()); - } -#if GREENLET_PY314 - // TODO: Should we be visiting the c_stack_refs objects? - // CPython uses a specific macro to do that which takes into - // account boxing and null values and then calls - // ``_PyGC_VisitStackRef``, but we don't have access to that, and - // we can't duplicate it ourself (because it compares - // ``visitproc`` to another function we can't access). - // The naive way of looping over c_stack_refs->ref and visiting - // those crashes the process (at least with GIL disabled). -#endif - return 0; -} - -void PythonState::tp_clear(bool own_top_frame) noexcept -{ - PythonStateContext::tp_clear(); - // If we get here owning a frame, - // we got dealloc'd without being finished. We may or may not be - // in the same thread. - if (own_top_frame) { - this->_top_frame.CLEAR(); - } -} - -#if GREENLET_USE_CFRAME -void PythonState::set_new_cframe(_PyCFrame& frame) noexcept -{ - frame = *PyThreadState_GET()->cframe; - /* Make the target greenlet refer to the stack value. */ - this->cframe = &frame; - /* - And restore the link to the previous frame so this one gets - unliked appropriately. - */ - this->cframe->previous = &PyThreadState_GET()->root_cframe; -} -#endif - -const PythonState::OwnedFrame& PythonState::top_frame() const noexcept -{ - return this->_top_frame; -} - -void PythonState::did_finish(PyThreadState* tstate) noexcept -{ -#if GREENLET_PY311 - // See https://github.com/gevent/gevent/issues/1924 and - // https://github.com/python-greenlet/greenlet/issues/328. In - // short, Python 3.11 allocates memory for frames as a sort of - // linked list that's kept as part of PyThreadState in the - // ``datastack_chunk`` member and friends. These are saved and - // restored as part of switching greenlets. - // - // When we initially switch to a greenlet, we set those to NULL. - // That causes the frame management code to treat this like a - // brand new thread and start a fresh list of chunks, beginning - // with a new "root" chunk. As we make calls in this greenlet, - // those chunks get added, and as calls return, they get popped. - // But the frame code (pystate.c) is careful to make sure that the - // root chunk never gets popped. - // - // Thus, when a greenlet exits for the last time, there will be at - // least a single root chunk that we must be responsible for - // deallocating. - // - // The complex part is that these chunks are allocated and freed - // using ``_PyObject_VirtualAlloc``/``Free``. Those aren't public - // functions, and they aren't exported for linking. It so happens - // that we know they are just thin wrappers around the Arena - // allocator, so we can use that directly to deallocate in a - // compatible way. - // - // CAUTION: Check this implementation detail on every major version. - // - // It might be nice to be able to do this in our destructor, but - // can we be sure that no one else is using that memory? Plus, as - // described below, our pointers may not even be valid anymore. As - // a special case, there is one time that we know we can do this, - // and that's from the destructor of the associated UserGreenlet - // (NOT main greenlet) - PyObjectArenaAllocator alloc; - _PyStackChunk* chunk = nullptr; - if (tstate) { - // We really did finish, we can never be switched to again. - chunk = tstate->datastack_chunk; - // Unfortunately, we can't do much sanity checking. Our - // this->datastack_chunk pointer is out of date (evaluation may - // have popped down through it already) so we can't verify that - // we deallocate it. I don't think we can even check datastack_top - // for the same reason. - - PyObject_GetArenaAllocator(&alloc); - tstate->datastack_chunk = nullptr; - tstate->datastack_limit = nullptr; - tstate->datastack_top = nullptr; - - } - else if (this->datastack_chunk) { - // The UserGreenlet (NOT the main greenlet!) is being deallocated. If we're - // still holding a stack chunk, it's garbage because we know - // we can never switch back to let cPython clean it up. - // Because the last time we got switched away from, and we - // haven't run since then, we know our chain is valid and can - // be dealloced. - chunk = this->datastack_chunk; - PyObject_GetArenaAllocator(&alloc); - } - - if (alloc.free && chunk) { - // In case the arena mechanism has been torn down already. - while (chunk) { - _PyStackChunk *prev = chunk->previous; - chunk->previous = nullptr; - alloc.free(alloc.ctx, chunk, chunk->size); - chunk = prev; - } - } - - this->datastack_chunk = nullptr; - this->datastack_limit = nullptr; - this->datastack_top = nullptr; -#endif -} - - -}; // namespace greenlet - -#endif // GREENLET_PYTHON_STATE_CPP diff --git a/.env/Lib/site-packages/greenlet/TStackState.cpp b/.env/Lib/site-packages/greenlet/TStackState.cpp deleted file mode 100644 index 9743ab5..0000000 --- a/.env/Lib/site-packages/greenlet/TStackState.cpp +++ /dev/null @@ -1,265 +0,0 @@ -#ifndef GREENLET_STACK_STATE_CPP -#define GREENLET_STACK_STATE_CPP - -#include "TGreenlet.hpp" - -namespace greenlet { - -#ifdef GREENLET_USE_STDIO -#include -using std::cerr; -using std::endl; - -std::ostream& operator<<(std::ostream& os, const StackState& s) -{ - os << "StackState(stack_start=" << (void*)s._stack_start - << ", stack_stop=" << (void*)s.stack_stop - << ", stack_copy=" << (void*)s.stack_copy - << ", stack_saved=" << s._stack_saved - << ", stack_prev=" << s.stack_prev - << ", addr=" << &s - << ")"; - return os; -} -#endif - -StackState::StackState(void* mark, StackState& current) - : _stack_start(nullptr), - stack_stop((char*)mark), - stack_copy(nullptr), - _stack_saved(0), - /* Skip a dying greenlet */ - stack_prev(current._stack_start - ? ¤t - : current.stack_prev) -{ -} - -StackState::StackState() - : _stack_start(nullptr), - stack_stop(nullptr), - stack_copy(nullptr), - _stack_saved(0), - stack_prev(nullptr) -{ -} - -StackState::StackState(const StackState& other) -// can't use a delegating constructor because of -// MSVC for Python 2.7 - : _stack_start(nullptr), - stack_stop(nullptr), - stack_copy(nullptr), - _stack_saved(0), - stack_prev(nullptr) -{ - this->operator=(other); -} - -StackState& StackState::operator=(const StackState& other) -{ - if (&other == this) { - return *this; - } - if (other._stack_saved) { - throw std::runtime_error("Refusing to steal memory."); - } - - //If we have memory allocated, dispose of it - this->free_stack_copy(); - - this->_stack_start = other._stack_start; - this->stack_stop = other.stack_stop; - this->stack_copy = other.stack_copy; - this->_stack_saved = other._stack_saved; - this->stack_prev = other.stack_prev; - return *this; -} - -inline void StackState::free_stack_copy() noexcept -{ - PyMem_Free(this->stack_copy); - this->stack_copy = nullptr; - this->_stack_saved = 0; -} - -inline void StackState::copy_heap_to_stack(const StackState& current) noexcept -{ - - /* Restore the heap copy back into the C stack */ - if (this->_stack_saved != 0) { - memcpy(this->_stack_start, this->stack_copy, this->_stack_saved); - this->free_stack_copy(); - } - StackState* owner = const_cast(¤t); - if (!owner->_stack_start) { - owner = owner->stack_prev; /* greenlet is dying, skip it */ - } - while (owner && owner->stack_stop <= this->stack_stop) { - // cerr << "\tOwner: " << owner << endl; - owner = owner->stack_prev; /* find greenlet with more stack */ - } - this->stack_prev = owner; - // cerr << "\tFinished with: " << *this << endl; -} - -inline int StackState::copy_stack_to_heap_up_to(const char* const stop) noexcept -{ - /* Save more of g's stack into the heap -- at least up to 'stop' - g->stack_stop |________| - | | - | __ stop . . . . . - | | ==> . . - |________| _______ - | | | | - | | | | - g->stack_start | | |_______| g->stack_copy - */ - intptr_t sz1 = this->_stack_saved; - intptr_t sz2 = stop - this->_stack_start; - assert(this->_stack_start); - if (sz2 > sz1) { - char* c = (char*)PyMem_Realloc(this->stack_copy, sz2); - if (!c) { - PyErr_NoMemory(); - return -1; - } - memcpy(c + sz1, this->_stack_start + sz1, sz2 - sz1); - this->stack_copy = c; - this->_stack_saved = sz2; - } - return 0; -} - -inline int StackState::copy_stack_to_heap(char* const stackref, - const StackState& current) noexcept -{ - /* must free all the C stack up to target_stop */ - const char* const target_stop = this->stack_stop; - - StackState* owner = const_cast(¤t); - assert(owner->_stack_saved == 0); // everything is present on the stack - if (!owner->_stack_start) { - owner = owner->stack_prev; /* not saved if dying */ - } - else { - owner->_stack_start = stackref; - } - - while (owner->stack_stop < target_stop) { - /* ts_current is entierely within the area to free */ - if (owner->copy_stack_to_heap_up_to(owner->stack_stop)) { - return -1; /* XXX */ - } - owner = owner->stack_prev; - } - if (owner != this) { - if (owner->copy_stack_to_heap_up_to(target_stop)) { - return -1; /* XXX */ - } - } - return 0; -} - -inline bool StackState::started() const noexcept -{ - return this->stack_stop != nullptr; -} - -inline bool StackState::main() const noexcept -{ - return this->stack_stop == (char*)-1; -} - -inline bool StackState::active() const noexcept -{ - return this->_stack_start != nullptr; -} - -inline void StackState::set_active() noexcept -{ - assert(this->_stack_start == nullptr); - this->_stack_start = (char*)1; -} - -inline void StackState::set_inactive() noexcept -{ - this->_stack_start = nullptr; - // XXX: What if we still have memory out there? - // That case is actually triggered by - // test_issue251_issue252_explicit_reference_not_collectable (greenlet.tests.test_leaks.TestLeaks) - // and - // test_issue251_issue252_need_to_collect_in_background - // (greenlet.tests.test_leaks.TestLeaks) - // - // Those objects never get deallocated, so the destructor never - // runs. - // It *seems* safe to clean up the memory here? - if (this->_stack_saved) { - this->free_stack_copy(); - } -} - -inline intptr_t StackState::stack_saved() const noexcept -{ - return this->_stack_saved; -} - -inline char* StackState::stack_start() const noexcept -{ - return this->_stack_start; -} - - -inline StackState StackState::make_main() noexcept -{ - StackState s; - s._stack_start = (char*)1; - s.stack_stop = (char*)-1; - return s; -} - -StackState::~StackState() -{ - if (this->_stack_saved != 0) { - this->free_stack_copy(); - } -} - -void StackState::copy_from_stack(void* vdest, const void* vsrc, size_t n) const -{ - char* dest = static_cast(vdest); - const char* src = static_cast(vsrc); - if (src + n <= this->_stack_start - || src >= this->_stack_start + this->_stack_saved - || this->_stack_saved == 0) { - // Nothing we're copying was spilled from the stack - memcpy(dest, src, n); - return; - } - - if (src < this->_stack_start) { - // Copy the part before the saved stack. - // We know src + n > _stack_start due to the test above. - const size_t nbefore = this->_stack_start - src; - memcpy(dest, src, nbefore); - dest += nbefore; - src += nbefore; - n -= nbefore; - } - // We know src >= _stack_start after the before-copy, and - // src < _stack_start + _stack_saved due to the first if condition - size_t nspilled = std::min(n, this->_stack_start + this->_stack_saved - src); - memcpy(dest, this->stack_copy + (src - this->_stack_start), nspilled); - dest += nspilled; - src += nspilled; - n -= nspilled; - if (n > 0) { - // Copy the part after the saved stack - memcpy(dest, src, n); - } -} - -}; // namespace greenlet - -#endif // GREENLET_STACK_STATE_CPP diff --git a/.env/Lib/site-packages/greenlet/TThreadState.hpp b/.env/Lib/site-packages/greenlet/TThreadState.hpp deleted file mode 100644 index b3451a0..0000000 --- a/.env/Lib/site-packages/greenlet/TThreadState.hpp +++ /dev/null @@ -1,523 +0,0 @@ -#ifndef GREENLET_THREAD_STATE_HPP -#define GREENLET_THREAD_STATE_HPP - -#include -#include -#include - -#include "greenlet_internal.hpp" -#include "greenlet_refs.hpp" -#include "greenlet_thread_support.hpp" - -using greenlet::refs::BorrowedObject; -using greenlet::refs::BorrowedGreenlet; -using greenlet::refs::BorrowedMainGreenlet; -using greenlet::refs::OwnedMainGreenlet; -using greenlet::refs::OwnedObject; -using greenlet::refs::OwnedGreenlet; -using greenlet::refs::OwnedList; -using greenlet::refs::PyErrFetchParam; -using greenlet::refs::PyArgParseParam; -using greenlet::refs::ImmortalString; -using greenlet::refs::CreatedModule; -using greenlet::refs::PyErrPieces; -using greenlet::refs::NewReference; - -namespace greenlet { -/** - * Thread-local state of greenlets. - * - * Each native thread will get exactly one of these objects, - * automatically accessed through the best available thread-local - * mechanism the compiler supports (``thread_local`` for C++11 - * compilers or ``__thread``/``declspec(thread)`` for older GCC/clang - * or MSVC, respectively.) - * - * Previously, we kept thread-local state mostly in a bunch of - * ``static volatile`` variables in the main greenlet file.. This had - * the problem of requiring extra checks, loops, and great care - * accessing these variables if we potentially invoked any Python code - * that could release the GIL, because the state could change out from - * under us. Making the variables thread-local solves this problem. - * - * When we detected that a greenlet API accessing the current greenlet - * was invoked from a different thread than the greenlet belonged to, - * we stored a reference to the greenlet in the Python thread - * dictionary for the thread the greenlet belonged to. This could lead - * to memory leaks if the thread then exited (because of a reference - * cycle, as greenlets referred to the thread dictionary, and deleting - * non-current greenlets leaked their frame plus perhaps arguments on - * the C stack). If a thread exited while still having running - * greenlet objects (perhaps that had just switched back to the main - * greenlet), and did not invoke one of the greenlet APIs *in that - * thread, immediately before it exited, without some other thread - * then being invoked*, such a leak was guaranteed. - * - * This can be partly solved by using compiler thread-local variables - * instead of the Python thread dictionary, thus avoiding a cycle. - * - * To fully solve this problem, we need a reliable way to know that a - * thread is done and we should clean up the main greenlet. On POSIX, - * we can use the destructor function of ``pthread_key_create``, but - * there's nothing similar on Windows; a C++11 thread local object - * reliably invokes its destructor when the thread it belongs to exits - * (non-C++11 compilers offer ``__thread`` or ``declspec(thread)`` to - * create thread-local variables, but they can't hold C++ objects that - * invoke destructors; the C++11 version is the most portable solution - * I found). When the thread exits, we can drop references and - * otherwise manipulate greenlets and frames that we know can no - * longer be switched to. - * - * There are two small wrinkles. The first is that when the thread - * exits, it is too late to actually invoke Python APIs: the Python - * thread state is gone, and the GIL is released. To solve *this* - * problem, our destructor uses ``Py_AddPendingCall`` to transfer the - * destruction work to the main thread. - * - * The second is that once the thread exits, the thread local object - * is invalid and we can't even access a pointer to it, so we can't - * pass it to ``Py_AddPendingCall``. This is handled by actually using - * a second object that's thread local (ThreadStateCreator) and having - * it dynamically allocate this object so it can live until the - * pending call runs. - */ - - - -class ThreadState { -private: - // As of commit 08ad1dd7012b101db953f492e0021fb08634afad - // this class needed 56 bytes in o Py_DEBUG build - // on 64-bit macOS 11. - // Adding the vector takes us up to 80 bytes () - - /* Strong reference to the main greenlet */ - OwnedMainGreenlet main_greenlet; - - /* Strong reference to the current greenlet. */ - OwnedGreenlet current_greenlet; - - /* Strong reference to the trace function, if any. */ - OwnedObject tracefunc; - - typedef std::vector > deleteme_t; - /* A vector of raw PyGreenlet pointers representing things that need - deleted when this thread is running. The vector owns the - references, but you need to manually INCREF/DECREF as you use - them. We don't use a vector because we - make copy of this vector, and that would become O(n) as all the - refcounts are incremented in the copy. - */ - deleteme_t deleteme; - -#ifdef GREENLET_NEEDS_EXCEPTION_STATE_SAVED - void* exception_state; -#endif - -#ifdef Py_GIL_DISABLED - static std::atomic _clocks_used_doing_gc; -#else - static std::clock_t _clocks_used_doing_gc; -#endif - static ImmortalString get_referrers_name; - static PythonAllocator allocator; - - G_NO_COPIES_OF_CLS(ThreadState); - - - // Allocates a main greenlet for the thread state. If this fails, - // exits the process. Called only during constructing a ThreadState. - MainGreenlet* alloc_main() - { - PyGreenlet* gmain; - - /* create the main greenlet for this thread */ - gmain = reinterpret_cast(PyType_GenericAlloc(&PyGreenlet_Type, 0)); - if (gmain == NULL) { - throw PyFatalError("alloc_main failed to alloc"); //exits the process - } - - MainGreenlet* const main = new MainGreenlet(gmain, this); - - assert(Py_REFCNT(gmain) == 1); - assert(gmain->pimpl == main); - return main; - } - - -public: - static void* operator new(size_t UNUSED(count)) - { - return ThreadState::allocator.allocate(1); - } - - static void operator delete(void* ptr) - { - return ThreadState::allocator.deallocate(static_cast(ptr), - 1); - } - - static void init() - { - ThreadState::get_referrers_name = "get_referrers"; - ThreadState::set_clocks_used_doing_gc(0); - } - - ThreadState() - { - -#ifdef GREENLET_NEEDS_EXCEPTION_STATE_SAVED - this->exception_state = slp_get_exception_state(); -#endif - - // XXX: Potentially dangerous, exposing a not fully - // constructed object. - MainGreenlet* const main = this->alloc_main(); - this->main_greenlet = OwnedMainGreenlet::consuming( - main->self() - ); - assert(this->main_greenlet); - this->current_greenlet = main->self(); - // The main greenlet starts with 1 refs: The returned one. We - // then copied it to the current greenlet. - assert(this->main_greenlet.REFCNT() == 2); - } - - inline void restore_exception_state() - { -#ifdef GREENLET_NEEDS_EXCEPTION_STATE_SAVED - // It's probably important this be inlined and only call C - // functions to avoid adding an SEH frame. - slp_set_exception_state(this->exception_state); -#endif - } - - inline bool has_main_greenlet() const noexcept - { - return bool(this->main_greenlet); - } - - // Called from the ThreadStateCreator when we're in non-standard - // threading mode. In that case, there is an object in the Python - // thread state dictionary that points to us. The main greenlet - // also traverses into us, in which case it's crucial not to - // traverse back into the main greenlet. - int tp_traverse(visitproc visit, void* arg, bool traverse_main=true) - { - if (traverse_main) { - Py_VISIT(main_greenlet.borrow_o()); - } - if (traverse_main || current_greenlet != main_greenlet) { - Py_VISIT(current_greenlet.borrow_o()); - } - Py_VISIT(tracefunc.borrow()); - return 0; - } - - inline BorrowedMainGreenlet borrow_main_greenlet() const noexcept - { - assert(this->main_greenlet); - assert(this->main_greenlet.REFCNT() >= 2); - return this->main_greenlet; - }; - - inline OwnedMainGreenlet get_main_greenlet() const noexcept - { - return this->main_greenlet; - } - - /** - * In addition to returning a new reference to the currunt - * greenlet, this performs any maintenance needed. - */ - inline OwnedGreenlet get_current() - { - /* green_dealloc() cannot delete greenlets from other threads, so - it stores them in the thread dict; delete them now. */ - this->clear_deleteme_list(); - //assert(this->current_greenlet->main_greenlet == this->main_greenlet); - //assert(this->main_greenlet->main_greenlet == this->main_greenlet); - return this->current_greenlet; - } - - /** - * As for non-const get_current(); - */ - inline BorrowedGreenlet borrow_current() - { - this->clear_deleteme_list(); - return this->current_greenlet; - } - - /** - * Does no maintenance. - */ - inline OwnedGreenlet get_current() const - { - return this->current_greenlet; - } - - template - inline bool is_current(const refs::PyObjectPointer& obj) const - { - return this->current_greenlet.borrow_o() == obj.borrow_o(); - } - - inline void set_current(const OwnedGreenlet& target) - { - this->current_greenlet = target; - } - -private: - /** - * Deref and remove the greenlets from the deleteme list. Must be - * holding the GIL. - * - * If *murder* is true, then we must be called from a different - * thread than the one that these greenlets were running in. - * In that case, if the greenlet was actually running, we destroy - * the frame reference and otherwise make it appear dead before - * proceeding; otherwise, we would try (and fail) to raise an - * exception in it and wind up right back in this list. - */ - inline void clear_deleteme_list(const bool murder=false) - { - if (!this->deleteme.empty()) { - // It's possible we could add items to this list while - // running Python code if there's a thread switch, so we - // need to defensively copy it before that can happen. - deleteme_t copy = this->deleteme; - this->deleteme.clear(); // in case things come back on the list - for(deleteme_t::iterator it = copy.begin(), end = copy.end(); - it != end; - ++it ) { - PyGreenlet* to_del = *it; - if (murder) { - // Force each greenlet to appear dead; we can't raise an - // exception into it anymore anyway. - to_del->pimpl->murder_in_place(); - } - - // The only reference to these greenlets should be in - // this list, decreffing them should let them be - // deleted again, triggering calls to green_dealloc() - // in the correct thread (if we're not murdering). - // This may run arbitrary Python code and switch - // threads or greenlets! - Py_DECREF(to_del); - if (PyErr_Occurred()) { - PyErr_WriteUnraisable(nullptr); - PyErr_Clear(); - } - } - } - } - -public: - - /** - * Returns a new reference, or a false object. - */ - inline OwnedObject get_tracefunc() const - { - return tracefunc; - }; - - - inline void set_tracefunc(BorrowedObject tracefunc) - { - assert(tracefunc); - if (tracefunc == BorrowedObject(Py_None)) { - this->tracefunc.CLEAR(); - } - else { - this->tracefunc = tracefunc; - } - } - - /** - * Given a reference to a greenlet that some other thread - * attempted to delete (has a refcount of 0) store it for later - * deletion when the thread this state belongs to is current. - */ - inline void delete_when_thread_running(PyGreenlet* to_del) - { - Py_INCREF(to_del); - this->deleteme.push_back(to_del); - } - - /** - * Set to std::clock_t(-1) to disable. - */ - inline static std::clock_t clocks_used_doing_gc() - { -#ifdef Py_GIL_DISABLED - return ThreadState::_clocks_used_doing_gc.load(std::memory_order_relaxed); -#else - return ThreadState::_clocks_used_doing_gc; -#endif - } - - inline static void set_clocks_used_doing_gc(std::clock_t value) - { -#ifdef Py_GIL_DISABLED - ThreadState::_clocks_used_doing_gc.store(value, std::memory_order_relaxed); -#else - ThreadState::_clocks_used_doing_gc = value; -#endif - } - - inline static void add_clocks_used_doing_gc(std::clock_t value) - { -#ifdef Py_GIL_DISABLED - ThreadState::_clocks_used_doing_gc.fetch_add(value, std::memory_order_relaxed); -#else - ThreadState::_clocks_used_doing_gc += value; -#endif - } - - ~ThreadState() - { - if (!PyInterpreterState_Head()) { - // We shouldn't get here (our callers protect us) - // but if we do, all we can do is bail early. - return; - } - - // We should not have an "origin" greenlet; that only exists - // for the temporary time during a switch, which should not - // be in progress as the thread dies. - //assert(!this->switching_state.origin); - - this->tracefunc.CLEAR(); - - // Forcibly GC as much as we can. - this->clear_deleteme_list(true); - - // The pending call did this. - assert(this->main_greenlet->thread_state() == nullptr); - - // If the main greenlet is the current greenlet, - // then we "fell off the end" and the thread died. - // It's possible that there is some other greenlet that - // switched to us, leaving a reference to the main greenlet - // on the stack, somewhere uncollectible. Try to detect that. - if (this->current_greenlet == this->main_greenlet && this->current_greenlet) { - assert(this->current_greenlet->is_currently_running_in_some_thread()); - // Drop one reference we hold. - this->current_greenlet.CLEAR(); - assert(!this->current_greenlet); - // Only our reference to the main greenlet should be left, - // But hold onto the pointer in case we need to do extra cleanup. - PyGreenlet* old_main_greenlet = this->main_greenlet.borrow(); - Py_ssize_t cnt = this->main_greenlet.REFCNT(); - this->main_greenlet.CLEAR(); - if (ThreadState::clocks_used_doing_gc() != std::clock_t(-1) - && cnt == 2 && Py_REFCNT(old_main_greenlet) == 1) { - // Highly likely that the reference is somewhere on - // the stack, not reachable by GC. Verify. - // XXX: This is O(n) in the total number of objects. - // TODO: Add a way to disable this at runtime, and - // another way to report on it. - std::clock_t begin = std::clock(); - NewReference gc(PyImport_ImportModule("gc")); - if (gc) { - OwnedObject get_referrers = gc.PyRequireAttr(ThreadState::get_referrers_name); - OwnedList refs(get_referrers.PyCall(old_main_greenlet)); - if (refs && refs.empty()) { - assert(refs.REFCNT() == 1); - // We found nothing! So we left a dangling - // reference: Probably the last thing some - // other greenlet did was call - // 'getcurrent().parent.switch()' to switch - // back to us. Clean it up. This will be the - // case on CPython 3.7 and newer, as they use - // an internal calling conversion that avoids - // creating method objects and storing them on - // the stack. - Py_DECREF(old_main_greenlet); - } - else if (refs - && refs.size() == 1 - && PyCFunction_Check(refs.at(0)) - && Py_REFCNT(refs.at(0)) == 2) { - assert(refs.REFCNT() == 1); - // Ok, we found a C method that refers to the - // main greenlet, and its only referenced - // twice, once in the list we just created, - // once from...somewhere else. If we can't - // find where else, then this is a leak. - // This happens in older versions of CPython - // that create a bound method object somewhere - // on the stack that we'll never get back to. - if (PyCFunction_GetFunction(refs.at(0).borrow()) == (PyCFunction)green_switch) { - BorrowedObject function_w = refs.at(0); - refs.clear(); // destroy the reference - // from the list. - // back to one reference. Can *it* be - // found? - assert(function_w.REFCNT() == 1); - refs = get_referrers.PyCall(function_w); - if (refs && refs.empty()) { - // Nope, it can't be found so it won't - // ever be GC'd. Drop it. - Py_CLEAR(function_w); - } - } - } - std::clock_t end = std::clock(); - ThreadState::add_clocks_used_doing_gc(end - begin); - } - } - } - - // We need to make sure this greenlet appears to be dead, - // because otherwise deallocing it would fail to raise an - // exception in it (the thread is dead) and put it back in our - // deleteme list. - if (this->current_greenlet) { - this->current_greenlet->murder_in_place(); - this->current_greenlet.CLEAR(); - } - - if (this->main_greenlet) { - // Couldn't have been the main greenlet that was running - // when the thread exited (because we already cleared this - // pointer if it was). This shouldn't be possible? - - // If the main greenlet was current when the thread died (it - // should be, right?) then we cleared its self pointer above - // when we cleared the current greenlet's main greenlet pointer. - // assert(this->main_greenlet->main_greenlet == this->main_greenlet - // || !this->main_greenlet->main_greenlet); - // // self reference, probably gone - // this->main_greenlet->main_greenlet.CLEAR(); - - // This will actually go away when the ivar is destructed. - this->main_greenlet.CLEAR(); - } - - if (PyErr_Occurred()) { - PyErr_WriteUnraisable(NULL); - PyErr_Clear(); - } - - } - -}; - -ImmortalString ThreadState::get_referrers_name(nullptr); -PythonAllocator ThreadState::allocator; -#ifdef Py_GIL_DISABLED -std::atomic ThreadState::_clocks_used_doing_gc(0); -#else -std::clock_t ThreadState::_clocks_used_doing_gc(0); -#endif - - - - - -}; // namespace greenlet - -#endif diff --git a/.env/Lib/site-packages/greenlet/TThreadStateCreator.hpp b/.env/Lib/site-packages/greenlet/TThreadStateCreator.hpp deleted file mode 100644 index ebd33a3..0000000 --- a/.env/Lib/site-packages/greenlet/TThreadStateCreator.hpp +++ /dev/null @@ -1,102 +0,0 @@ -#ifndef GREENLET_THREAD_STATE_CREATOR_HPP -#define GREENLET_THREAD_STATE_CREATOR_HPP - -#include -#include - -#include "greenlet_internal.hpp" -#include "greenlet_refs.hpp" -#include "greenlet_thread_support.hpp" - -#include "TThreadState.hpp" - -namespace greenlet { - - -typedef void (*ThreadStateDestructor)(ThreadState* const); - -// Only one of these, auto created per thread as a thread_local. -// Constructing the state constructs the MainGreenlet. -template -class ThreadStateCreator -{ -private: - // Initialized to 1, and, if still 1, created on access. - // Set to 0 on destruction. - ThreadState* _state; - G_NO_COPIES_OF_CLS(ThreadStateCreator); - - inline bool has_initialized_state() const noexcept - { - return this->_state != (ThreadState*)1; - } - - inline bool has_state() const noexcept - { - return this->has_initialized_state() && this->_state != nullptr; - } - -public: - - ThreadStateCreator() : - _state((ThreadState*)1) - { - } - - ~ThreadStateCreator() - { - if (this->has_state()) { - Destructor(this->_state); - } - - this->_state = nullptr; - } - - inline ThreadState& state() - { - // The main greenlet will own this pointer when it is created, - // which will be right after this. The plan is to give every - // greenlet a pointer to the main greenlet for the thread it - // runs in; if we are doing something cross-thread, we need to - // access the pointer from the main greenlet. Deleting the - // thread, and hence the thread-local storage, will delete the - // state pointer in the main greenlet. - if (!this->has_initialized_state()) { - // XXX: Assuming allocation never fails - this->_state = new ThreadState; - // For non-standard threading, we need to store an object - // in the Python thread state dictionary so that it can be - // DECREF'd when the thread ends (ideally; the dict could - // last longer) and clean this object up. - } - if (!this->_state) { - throw std::runtime_error("Accessing state after destruction."); - } - return *this->_state; - } - - operator ThreadState&() - { - return this->state(); - } - - operator ThreadState*() - { - return &this->state(); - } - - inline int tp_traverse(visitproc visit, void* arg) - { - if (this->has_state()) { - return this->_state->tp_traverse(visit, arg); - } - return 0; - } - -}; - - - -}; // namespace greenlet - -#endif diff --git a/.env/Lib/site-packages/greenlet/TThreadStateDestroy.cpp b/.env/Lib/site-packages/greenlet/TThreadStateDestroy.cpp deleted file mode 100644 index ae0b9ae..0000000 --- a/.env/Lib/site-packages/greenlet/TThreadStateDestroy.cpp +++ /dev/null @@ -1,223 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of the ThreadState destructors. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#ifndef T_THREADSTATE_DESTROY -#define T_THREADSTATE_DESTROY - -#include "TGreenlet.hpp" - -#include "greenlet_thread_support.hpp" -#include "greenlet_compiler_compat.hpp" -#include "TGreenletGlobals.cpp" -#include "TThreadState.hpp" -#include "TThreadStateCreator.hpp" - -namespace greenlet { - -extern "C" { - -struct ThreadState_DestroyNoGIL -{ - /** - This function uses the same lock that the PendingCallback does - */ - static void - MarkGreenletDeadAndQueueCleanup(ThreadState* const state) - { -#if GREENLET_BROKEN_THREAD_LOCAL_CLEANUP_JUST_LEAK - // One rare platform. - return; -#endif - // We are *NOT* holding the GIL. Our thread is in the middle - // of its death throes and the Python thread state is already - // gone so we can't use most Python APIs. One that is safe is - // ``Py_AddPendingCall``, unless the interpreter itself has - // been torn down. There is a limited number of calls that can - // be queued: 32 (NPENDINGCALLS) in CPython 3.10, so we - // coalesce these calls using our own queue. - - if (!MarkGreenletDeadIfNeeded(state)) { - // No state, or no greenlet - return; - } - - // XXX: Because we don't have the GIL, this is a race condition. - if (!PyInterpreterState_Head()) { - // We have to leak the thread state, if the - // interpreter has shut down when we're getting - // deallocated, we can't run the cleanup code that - // deleting it would imply. - return; - } - - AddToCleanupQueue(state); - - } - -private: - - // If the state has an allocated main greenlet: - // - mark the greenlet as dead by disassociating it from the state; - // - return 1 - // Otherwise, return 0. - static bool - MarkGreenletDeadIfNeeded(ThreadState* const state) - { - if (!state) { - return false; - } - LockGuard cleanup_lock(*mod_globs->thread_states_to_destroy_lock); - if (state->has_main_greenlet()) { - // mark the thread as dead ASAP. - // this is racy! If we try to throw or switch to a - // greenlet from this thread from some other thread before - // we clear the state pointer, it won't realize the state - // is dead which can crash the process. - PyGreenlet* p(state->borrow_main_greenlet().borrow()); - assert(p->pimpl->thread_state() == state || p->pimpl->thread_state() == nullptr); - dynamic_cast(p->pimpl)->thread_state(nullptr); - return true; - } - return false; - } - - static void - AddToCleanupQueue(ThreadState* const state) - { - assert(state && state->has_main_greenlet()); - - // NOTE: Because we're not holding the GIL here, some other - // Python thread could run and call ``os.fork()``, which would - // be bad if that happened while we are holding the cleanup - // lock (it wouldn't function in the child process). - // Make a best effort to try to keep the duration we hold the - // lock short. - // TODO: On platforms that support it, use ``pthread_atfork`` to - // drop this lock. - LockGuard cleanup_lock(*mod_globs->thread_states_to_destroy_lock); - - mod_globs->queue_to_destroy(state); - if (mod_globs->thread_states_to_destroy.size() == 1) { - // We added the first item to the queue. We need to schedule - // the cleanup. - - // A size greater than 1 means that we have already added the pending call, - // and in fact, it may be executing now. - // If it is executing, our lock makes sure that it will see the item we just added - // to the queue on its next iteration (after we release the lock) - // - // A size of 1 means there is no pending call, OR the pending call is - // currently executing, has dropped the lock, and is deleting the last item - // from the queue; its next iteration will go ahead and delete the item we just added. - // And the pending call we schedule here will have no work to do. - int result = AddPendingCall( - PendingCallback_DestroyQueue, - nullptr); - if (result < 0) { - // Hmm, what can we do here? - fprintf(stderr, - "greenlet: WARNING: failed in call to Py_AddPendingCall; " - "expect a memory leak.\n"); - } - } - } - - static int - PendingCallback_DestroyQueue(void* UNUSED(arg)) - { - // We're may or may not be holding the GIL here (depending on - // Py_GIL_DISABLED), so calls to ``os.fork()`` may or may not - // be possible. - while (1) { - ThreadState* to_destroy; - { - LockGuard cleanup_lock(*mod_globs->thread_states_to_destroy_lock); - if (mod_globs->thread_states_to_destroy.empty()) { - break; - } - to_destroy = mod_globs->take_next_to_destroy(); - } - assert(to_destroy); - assert(to_destroy->has_main_greenlet()); - // Drop the lock while we do the actual deletion. - // This allows other calls to MarkGreenletDeadAndQueueCleanup - // to enter and add to our queue. - DestroyOne(to_destroy); - } - return 0; - } - - static void - DestroyOne(const ThreadState* const state) - { - // May or may not be holding the GIL (depending on Py_GIL_DISABLED). - // Passed a non-shared pointer to the actual thread state. - // state -> main greenlet - assert(state->has_main_greenlet()); - PyGreenlet* main(state->borrow_main_greenlet()); - // When we need to do cross-thread operations, we check this. - // A NULL value means the thread died some time ago. - // We do this here, rather than in a Python dealloc function - // for the greenlet, in case there's still a reference out - // there. - dynamic_cast(main->pimpl)->thread_state(nullptr); - - delete state; // Deleting this runs the destructor, DECREFs the main greenlet. - } - - - static int AddPendingCall(int (*func)(void*), void* arg) - { - // If the interpreter is in the middle of finalizing, we can't add a - // pending call. Trying to do so will end up in a SIGSEGV, as - // Py_AddPendingCall will not be able to get the interpreter and will - // try to dereference a NULL pointer. It's possible this can still - // segfault if we happen to get context switched, and maybe we should - // just always implement our own AddPendingCall, but I'd like to see if - // this works first -#if GREENLET_PY313 - if (Py_IsFinalizing()) { -#else - if (_Py_IsFinalizing()) { -#endif -#ifdef GREENLET_DEBUG - // No need to log in the general case. Yes, we'll leak, - // but we're shutting down so it should be ok. - fprintf(stderr, - "greenlet: WARNING: Interpreter is finalizing. Ignoring " - "call to Py_AddPendingCall; \n"); -#endif - return 0; - } - return Py_AddPendingCall(func, arg); - } - - - - - -}; -}; - -}; // namespace greenlet - -// The intent when GET_THREAD_STATE() is needed multiple times in a -// function is to take a reference to its return value in a local -// variable, to avoid the thread-local indirection. On some platforms -// (macOS), accessing a thread-local involves a function call (plus an -// initial function call in each function that uses a thread local); -// in contrast, static volatile variables are at some pre-computed -// offset. -typedef greenlet::ThreadStateCreator ThreadStateCreator; -static thread_local ThreadStateCreator g_thread_state_global; -#define GET_THREAD_STATE() g_thread_state_global - -#endif //T_THREADSTATE_DESTROY diff --git a/.env/Lib/site-packages/greenlet/TUserGreenlet.cpp b/.env/Lib/site-packages/greenlet/TUserGreenlet.cpp deleted file mode 100644 index 73a8133..0000000 --- a/.env/Lib/site-packages/greenlet/TUserGreenlet.cpp +++ /dev/null @@ -1,662 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/** - * Implementation of greenlet::UserGreenlet. - * - * Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#ifndef T_USER_GREENLET_CPP -#define T_USER_GREENLET_CPP - -#include "greenlet_internal.hpp" -#include "TGreenlet.hpp" - -#include "TThreadStateDestroy.cpp" - - -namespace greenlet { -using greenlet::refs::BorrowedMainGreenlet; -greenlet::PythonAllocator UserGreenlet::allocator; - -void* UserGreenlet::operator new(size_t UNUSED(count)) -{ - return allocator.allocate(1); -} - - -void UserGreenlet::operator delete(void* ptr) -{ - return allocator.deallocate(static_cast(ptr), - 1); -} - - -UserGreenlet::UserGreenlet(PyGreenlet* p, BorrowedGreenlet the_parent) - : Greenlet(p), _parent(the_parent) -{ -} - -UserGreenlet::~UserGreenlet() -{ - // Python 3.11: If we don't clear out the raw frame datastack - // when deleting an unfinished greenlet, - // TestLeaks.test_untracked_memory_doesnt_increase_unfinished_thread_dealloc_in_main fails. - this->python_state.did_finish(nullptr); - this->tp_clear(); -} - - -const BorrowedMainGreenlet -UserGreenlet::main_greenlet() const -{ - return this->_main_greenlet; -} - - -BorrowedMainGreenlet -UserGreenlet::find_main_greenlet_in_lineage() const -{ - if (this->started()) { - assert(this->_main_greenlet); - return BorrowedMainGreenlet(this->_main_greenlet); - } - - if (!this->_parent) { - /* garbage collected greenlet in chain */ - // XXX: WHAT? - return BorrowedMainGreenlet(nullptr); - } - - return this->_parent->find_main_greenlet_in_lineage(); -} - - -/** - * CAUTION: This will allocate memory and may trigger garbage - * collection and arbitrary Python code. - */ -OwnedObject -UserGreenlet::throw_GreenletExit_during_dealloc(const ThreadState& current_thread_state) -{ - /* The dying greenlet cannot be a parent of ts_current - because the 'parent' field chain would hold a - reference */ - UserGreenlet::ParentIsCurrentGuard with_current_parent(this, current_thread_state); - - // We don't care about the return value, only whether an - // exception happened. Whether or not an exception happens, - // we need to restore the parent in case the greenlet gets - // resurrected. - return Greenlet::throw_GreenletExit_during_dealloc(current_thread_state); -} - -ThreadState* -UserGreenlet::thread_state() const noexcept -{ - // TODO: maybe make this throw, if the thread state isn't there? - // if (!this->main_greenlet) { - // throw std::runtime_error("No thread state"); // TODO: Better exception - // } - if (!this->_main_greenlet) { - return nullptr; - } - return this->_main_greenlet->thread_state(); -} - - -bool -UserGreenlet::was_running_in_dead_thread() const noexcept -{ - return this->_main_greenlet && !this->thread_state(); -} - -OwnedObject -UserGreenlet::g_switch() -{ - assert(this->args() || PyErr_Occurred()); - - try { - this->check_switch_allowed(); - } - catch (const PyErrOccurred&) { - this->release_args(); - throw; - } - - // Switching greenlets used to attempt to clean out ones that need - // deleted *if* we detected a thread switch. Should it still do - // that? - // An issue is that if we delete a greenlet from another thread, - // it gets queued to this thread, and ``kill_greenlet()`` switches - // back into the greenlet - - /* find the real target by ignoring dead greenlets, - and if necessary starting a greenlet. */ - switchstack_result_t err; - Greenlet* target = this; - // TODO: probably cleaner to handle the case where we do - // switch to ourself separately from the other cases. - // This can probably even further be simplified if we keep - // track of the switching_state we're going for and just call - // into g_switch() if it's not ourself. The main problem with that - // is that we would be using more stack space. - bool target_was_me = true; - bool was_initial_stub = false; - while (target) { - if (target->active()) { - if (!target_was_me) { - target->args() <<= this->args(); - assert(!this->args()); - } - err = target->g_switchstack(); - break; - } - if (!target->started()) { - // We never encounter a main greenlet that's not started. - assert(!target->main()); - UserGreenlet* real_target = static_cast(target); - assert(real_target); - void* dummymarker; - was_initial_stub = true; - if (!target_was_me) { - target->args() <<= this->args(); - assert(!this->args()); - } - try { - // This can only throw back to us while we're - // still in this greenlet. Once the new greenlet - // is bootstrapped, it has its own exception state. - err = real_target->g_initialstub(&dummymarker); - } - catch (const PyErrOccurred&) { - this->release_args(); - throw; - } - catch (const GreenletStartedWhileInPython&) { - // The greenlet was started sometime before this - // greenlet actually switched to it, i.e., - // "concurrent" calls to switch() or throw(). - // We need to retry the switch. - // Note that the current greenlet has been reset - // to this one (or we wouldn't be running!) - continue; - } - break; - } - - target = target->parent(); - target_was_me = false; - } - // The ``this`` pointer and all other stack or register based - // variables are invalid now, at least where things succeed - // above. - // But this one, probably not so much? It's not clear if it's - // safe to throw an exception at this point. - - if (err.status < 0) { - // If we get here, either g_initialstub() - // failed, or g_switchstack() failed. Either one of those - // cases SHOULD leave us in the original greenlet with a valid - // stack. - return this->on_switchstack_or_initialstub_failure(target, err, target_was_me, was_initial_stub); - } - - // err.the_new_current_greenlet would be the same as ``target``, - // if target wasn't probably corrupt. - return err.the_new_current_greenlet->g_switch_finish(err); -} - - - -Greenlet::switchstack_result_t -UserGreenlet::g_initialstub(void* mark) -{ - OwnedObject run; - - // We need to grab a reference to the current switch arguments - // in case we're entered concurrently during the call to - // GetAttr() and have to try again. - // We'll restore them when we return in that case. - // Scope them tightly to avoid ref leaks. - { - SwitchingArgs args(this->args()); - - /* save exception in case getattr clears it */ - PyErrPieces saved; - - /* - self.run is the object to call in the new greenlet. - This could run arbitrary python code and switch greenlets! - */ - run = this->self().PyRequireAttr(mod_globs->str_run); - /* restore saved exception */ - saved.PyErrRestore(); - - - /* recheck that it's safe to switch in case greenlet reparented anywhere above */ - this->check_switch_allowed(); - - /* by the time we got here another start could happen elsewhere, - * that means it should now be a regular switch. - * This can happen if the Python code is a subclass that implements - * __getattribute__ or __getattr__, or makes ``run`` a descriptor; - * all of those can run arbitrary code that switches back into - * this greenlet. - */ - if (this->stack_state.started()) { - // the successful switch cleared these out, we need to - // restore our version. They will be copied on up to the - // next target. - assert(!this->args()); - this->args() <<= args; - throw GreenletStartedWhileInPython(); - } - } - - // Sweet, if we got here, we have the go-ahead and will switch - // greenlets. - // Nothing we do from here on out should allow for a thread or - // greenlet switch: No arbitrary calls to Python, including - // decref'ing - -#if GREENLET_USE_CFRAME - /* OK, we need it, we're about to switch greenlets, save the state. */ - /* - See green_new(). This is a stack-allocated variable used - while *self* is in PyObject_Call(). - We want to defer copying the state info until we're sure - we need it and are in a stable place to do so. - */ - _PyCFrame trace_info; - - this->python_state.set_new_cframe(trace_info); -#endif - /* start the greenlet */ - ThreadState& thread_state = GET_THREAD_STATE().state(); - this->stack_state = StackState(mark, - thread_state.borrow_current()->stack_state); - this->python_state.set_initial_state(PyThreadState_GET()); - this->exception_state.clear(); - this->_main_greenlet = thread_state.get_main_greenlet(); - - /* perform the initial switch */ - switchstack_result_t err = this->g_switchstack(); - /* returns twice! - The 1st time with ``err == 1``: we are in the new greenlet. - This one owns a greenlet that used to be current. - The 2nd time with ``err <= 0``: back in the caller's - greenlet; this happens if the child finishes or switches - explicitly to us. Either way, the ``err`` variable is - created twice at the same memory location, but possibly - having different ``origin`` values. Note that it's not - constructed for the second time until the switch actually happens. - */ - if (err.status == 1) { - // In the new greenlet. - - // This never returns! Calling inner_bootstrap steals - // the contents of our run object within this stack frame, so - // it is not valid to do anything with it. - try { - this->inner_bootstrap(err.origin_greenlet.relinquish_ownership(), - run.relinquish_ownership()); - } - // Getting a C++ exception here isn't good. It's probably a - // bug in the underlying greenlet, meaning it's probably a - // C++ extension. We're going to abort anyway, but try to - // display some nice information *if* possible. Some obscure - // platforms don't properly support this (old 32-bit Arm, see see - // https://github.com/python-greenlet/greenlet/issues/385); that's not - // great, but should usually be OK because, as mentioned above, we're - // terminating anyway. - // - // The catching is tested by - // ``test_cpp.CPPTests.test_unhandled_exception_in_greenlet_aborts``. - // - // PyErrOccurred can theoretically be thrown by - // inner_bootstrap() -> g_switch_finish(), but that should - // never make it back to here. It is a std::exception and - // would be caught if it is. - catch (const std::exception& e) { - std::string base = "greenlet: Unhandled C++ exception: "; - base += e.what(); - Py_FatalError(base.c_str()); - } - catch (...) { - // Some compilers/runtimes use exceptions internally. - // It appears that GCC on Linux with libstdc++ throws an - // exception internally at process shutdown time to unwind - // stacks and clean up resources. Depending on exactly - // where we are when the process exits, that could result - // in an unknown exception getting here. If we - // Py_FatalError() or abort() here, we interfere with - // orderly process shutdown. Throwing the exception on up - // is the right thing to do. - // - // gevent's ``examples/dns_mass_resolve.py`` demonstrates this. -#ifndef NDEBUG - fprintf(stderr, - "greenlet: inner_bootstrap threw unknown exception; " - "is the process terminating?\n"); -#endif - throw; - } - Py_FatalError("greenlet: inner_bootstrap returned with no exception.\n"); - } - - - // In contrast, notice that we're keeping the origin greenlet - // around as an owned reference; we need it to call the trace - // function for the switch back into the parent. It was only - // captured at the time the switch actually happened, though, - // so we haven't been keeping an extra reference around this - // whole time. - - /* back in the parent */ - if (err.status < 0) { - /* start failed badly, restore greenlet state */ - this->stack_state = StackState(); - this->_main_greenlet.CLEAR(); - // CAUTION: This may run arbitrary Python code. - run.CLEAR(); // inner_bootstrap didn't run, we own the reference. - } - - // In the success case, the spawned code (inner_bootstrap) will - // take care of decrefing this, so we relinquish ownership so as - // to not double-decref. - - run.relinquish_ownership(); - - return err; -} - - -void -UserGreenlet::inner_bootstrap(PyGreenlet* origin_greenlet, PyObject* run) -{ - // The arguments here would be another great place for move. - // As it is, we take them as a reference so that when we clear - // them we clear what's on the stack above us. Do that NOW, and - // without using a C++ RAII object, - // so there's no way that exiting the parent frame can clear it, - // or we clear it unexpectedly. This arises in the context of the - // interpreter shutting down. See https://github.com/python-greenlet/greenlet/issues/325 - //PyObject* run = _run.relinquish_ownership(); - - /* in the new greenlet */ - assert(this->thread_state()->borrow_current() == BorrowedGreenlet(this->_self)); - // C++ exceptions cannot propagate to the parent greenlet from - // here. (TODO: Do we need a catch(...) clause, perhaps on the - // function itself? ALl we could do is terminate the program.) - // NOTE: On 32-bit Windows, the call chain is extremely - // important here in ways that are subtle, having to do with - // the depth of the SEH list. The call to restore it MUST NOT - // add a new SEH handler to the list, or we'll restore it to - // the wrong thing. - this->thread_state()->restore_exception_state(); - /* stack variables from above are no good and also will not unwind! */ - // EXCEPT: That can't be true, we access run, among others, here. - - this->stack_state.set_active(); /* running */ - - // We're about to possibly run Python code again, which - // could switch back/away to/from us, so we need to grab the - // arguments locally. - SwitchingArgs args; - args <<= this->args(); - assert(!this->args()); - - // XXX: We could clear this much earlier, right? - // Or would that introduce the possibility of running Python - // code when we don't want to? - // CAUTION: This may run arbitrary Python code. - this->_run_callable.CLEAR(); - - - // The first switch we need to manually call the trace - // function here instead of in g_switch_finish, because we - // never return there. - if (OwnedObject tracefunc = this->thread_state()->get_tracefunc()) { - OwnedGreenlet trace_origin; - trace_origin = origin_greenlet; - try { - g_calltrace(tracefunc, - args ? mod_globs->event_switch : mod_globs->event_throw, - trace_origin, - this->_self); - } - catch (const PyErrOccurred&) { - /* Turn trace errors into switch throws */ - args.CLEAR(); - } - } - - // We no longer need the origin, it was only here for - // tracing. - // We may never actually exit this stack frame so we need - // to explicitly clear it. - // This could run Python code and switch. - Py_CLEAR(origin_greenlet); - - OwnedObject result; - if (!args) { - /* pending exception */ - result = NULL; - } - else { - /* call g.run(*args, **kwargs) */ - // This could result in further switches - try { - //result = run.PyCall(args.args(), args.kwargs()); - // CAUTION: Just invoking this, before the function even - // runs, may cause memory allocations, which may trigger - // GC, which may run arbitrary Python code. - result = OwnedObject::consuming(PyObject_Call(run, args.args().borrow(), args.kwargs().borrow())); - } - catch (...) { - // Unhandled C++ exception! - - // If we declare ourselves as noexcept, if we don't catch - // this here, most platforms will just abort() the - // process. But on 64-bit Windows with older versions of - // the C runtime, this can actually corrupt memory and - // just return. We see this when compiling with the - // Windows 7.0 SDK targeting Windows Server 2008, but not - // when using the Appveyor Visual Studio 2019 image. So - // this currently only affects Python 2.7 on Windows 64. - // That is, the tests pass and the runtime aborts - // everywhere else. - // - // However, if we catch it and try to continue with a - // Python error, then all Windows 64 bit platforms corrupt - // memory. So all we can do is manually abort, hopefully - // with a good error message. (Note that the above was - // tested WITHOUT the `/EHr` switch being used at compile - // time, so MSVC may have "optimized" out important - // checking. Using that switch, we may be in a better - // place in terms of memory corruption.) But sometimes it - // can't be caught here at all, which is confusing but not - // terribly surprising; so again, the G_NOEXCEPT_WIN32 - // plus "/EHr". - // - // Hopefully the basic C stdlib is still functional enough - // for us to at least print an error. - // - // It gets more complicated than that, though, on some - // platforms, specifically at least Linux/gcc/libstdc++. They use - // an exception to unwind the stack when a background - // thread exits. (See comments about noexcept.) So this - // may not actually represent anything untoward. On those - // platforms we allow throws of this to propagate, or - // attempt to anyway. -# if defined(WIN32) || defined(_WIN32) - Py_FatalError( - "greenlet: Unhandled C++ exception from a greenlet run function. " - "Because memory is likely corrupted, terminating process."); - std::abort(); -#else - throw; -#endif - } - } - // These lines may run arbitrary code - args.CLEAR(); - Py_CLEAR(run); - - if (!result - && mod_globs->PyExc_GreenletExit.PyExceptionMatches() - && (this->args())) { - // This can happen, for example, if our only reference - // goes away after we switch back to the parent. - // See test_dealloc_switch_args_not_lost - PyErrPieces clear_error; - result <<= this->args(); - result = single_result(result); - } - this->release_args(); - this->python_state.did_finish(PyThreadState_GET()); - - result = g_handle_exit(result); - assert(this->thread_state()->borrow_current() == this->_self); - - /* jump back to parent */ - this->stack_state.set_inactive(); /* dead */ - - - // TODO: Can we decref some things here? Release our main greenlet - // and maybe parent? - for (Greenlet* parent = this->_parent; - parent; - parent = parent->parent()) { - // We need to somewhere consume a reference to - // the result; in most cases we'll never have control - // back in this stack frame again. Calling - // green_switch actually adds another reference! - // This would probably be clearer with a specific API - // to hand results to the parent. - parent->args() <<= result; - assert(!result); - // The parent greenlet now owns the result; in the - // typical case we'll never get back here to assign to - // result and thus release the reference. - try { - result = parent->g_switch(); - } - catch (const PyErrOccurred&) { - // Ignore, keep passing the error on up. - } - - /* Return here means switch to parent failed, - * in which case we throw *current* exception - * to the next parent in chain. - */ - assert(!result); - } - /* We ran out of parents, cannot continue */ - PyErr_WriteUnraisable(this->self().borrow_o()); - Py_FatalError("greenlet: ran out of parent greenlets while propagating exception; " - "cannot continue"); - std::abort(); -} - -void -UserGreenlet::run(const BorrowedObject nrun) -{ - if (this->started()) { - throw AttributeError( - "run cannot be set " - "after the start of the greenlet"); - } - this->_run_callable = nrun; -} - -const OwnedGreenlet -UserGreenlet::parent() const -{ - return this->_parent; -} - -void -UserGreenlet::parent(const BorrowedObject raw_new_parent) -{ - if (!raw_new_parent) { - throw AttributeError("can't delete attribute"); - } - - BorrowedMainGreenlet main_greenlet_of_new_parent; - BorrowedGreenlet new_parent(raw_new_parent.borrow()); // could - // throw - // TypeError! - for (BorrowedGreenlet p = new_parent; p; p = p->parent()) { - if (p == this->self()) { - throw ValueError("cyclic parent chain"); - } - main_greenlet_of_new_parent = p->main_greenlet(); - } - - if (!main_greenlet_of_new_parent) { - throw ValueError("parent must not be garbage collected"); - } - - if (this->started() - && this->_main_greenlet != main_greenlet_of_new_parent) { - throw ValueError("parent cannot be on a different thread"); - } - - this->_parent = new_parent; -} - -void -UserGreenlet::murder_in_place() -{ - this->_main_greenlet.CLEAR(); - Greenlet::murder_in_place(); -} - -bool -UserGreenlet::belongs_to_thread(const ThreadState* thread_state) const -{ - return Greenlet::belongs_to_thread(thread_state) && this->_main_greenlet == thread_state->borrow_main_greenlet(); -} - - -int -UserGreenlet::tp_traverse(visitproc visit, void* arg) -{ - Py_VISIT(this->_parent.borrow_o()); - Py_VISIT(this->_main_greenlet.borrow_o()); - Py_VISIT(this->_run_callable.borrow_o()); - - return Greenlet::tp_traverse(visit, arg); -} - -int -UserGreenlet::tp_clear() -{ - Greenlet::tp_clear(); - this->_parent.CLEAR(); - this->_main_greenlet.CLEAR(); - this->_run_callable.CLEAR(); - return 0; -} - -UserGreenlet::ParentIsCurrentGuard::ParentIsCurrentGuard(UserGreenlet* p, - const ThreadState& thread_state) - : oldparent(p->_parent), - greenlet(p) -{ - p->_parent = thread_state.get_current(); -} - -UserGreenlet::ParentIsCurrentGuard::~ParentIsCurrentGuard() -{ - this->greenlet->_parent = oldparent; - oldparent.CLEAR(); -} - -}; //namespace greenlet -#endif diff --git a/.env/Lib/site-packages/greenlet/__init__.py b/.env/Lib/site-packages/greenlet/__init__.py deleted file mode 100644 index ed1cc17..0000000 --- a/.env/Lib/site-packages/greenlet/__init__.py +++ /dev/null @@ -1,71 +0,0 @@ -# -*- coding: utf-8 -*- -""" -The root of the greenlet package. -""" -from __future__ import absolute_import -from __future__ import division -from __future__ import print_function - -__all__ = [ - '__version__', - '_C_API', - - 'GreenletExit', - 'error', - - 'getcurrent', - 'greenlet', - - 'gettrace', - 'settrace', -] - -# pylint:disable=no-name-in-module - -### -# Metadata -### -__version__ = '3.3.1' -from ._greenlet import _C_API # pylint:disable=no-name-in-module - -### -# Exceptions -### -from ._greenlet import GreenletExit -from ._greenlet import error - -### -# greenlets -### -from ._greenlet import getcurrent -from ._greenlet import greenlet - -### -# tracing -### -try: - from ._greenlet import gettrace - from ._greenlet import settrace -except ImportError: - # Tracing wasn't supported. - # XXX: The option to disable it was removed in 1.0, - # so this branch should be dead code. - pass - -### -# Constants -# These constants aren't documented and aren't recommended. -# In 1.0, USE_GC and USE_TRACING are always true, and USE_CONTEXT_VARS -# is the same as ``sys.version_info[:2] >= 3.7`` -### -from ._greenlet import GREENLET_USE_CONTEXT_VARS # pylint:disable=unused-import -from ._greenlet import GREENLET_USE_GC # pylint:disable=unused-import -from ._greenlet import GREENLET_USE_TRACING # pylint:disable=unused-import - -# Controlling the use of the gc module. Provisional API for this greenlet -# implementation in 2.0. -from ._greenlet import CLOCKS_PER_SEC # pylint:disable=unused-import -from ._greenlet import enable_optional_cleanup # pylint:disable=unused-import -from ._greenlet import get_clocks_used_doing_optional_cleanup # pylint:disable=unused-import - -# Other APIS in the _greenlet module are for test support. diff --git a/.env/Lib/site-packages/greenlet/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/greenlet/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 87a602c0872aed47bc9c17c697263f4e0495ece2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1067 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%0AQ-Ts7K0I5c9bNiL&fqC`)~3(x5x4R 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 diff --git a/.env/Lib/site-packages/greenlet/greenlet.cpp b/.env/Lib/site-packages/greenlet/greenlet.cpp deleted file mode 100644 index 7722bd0..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet.cpp +++ /dev/null @@ -1,323 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -/* Format with: - * clang-format -i --style=file src/greenlet/greenlet.c - * - * - * Fix missing braces with: - * clang-tidy src/greenlet/greenlet.c -fix -checks="readability-braces-around-statements" -*/ -#include -#include -#include -#include - - -#define PY_SSIZE_T_CLEAN -#include -#include "structmember.h" // PyMemberDef - -#include "greenlet_internal.hpp" -// Code after this point can assume access to things declared in stdint.h, -// including the fixed-width types. This goes for the platform-specific switch functions -// as well. -#include "greenlet_refs.hpp" -#include "greenlet_slp_switch.hpp" - -#include "greenlet_thread_support.hpp" -#include "TGreenlet.hpp" - -#include "TGreenletGlobals.cpp" - -#include "TGreenlet.cpp" -#include "TMainGreenlet.cpp" -#include "TUserGreenlet.cpp" -#include "TBrokenGreenlet.cpp" -#include "TExceptionState.cpp" -#include "TPythonState.cpp" -#include "TStackState.cpp" - -#include "TThreadState.hpp" -#include "TThreadStateCreator.hpp" -#include "TThreadStateDestroy.cpp" - -#include "PyGreenlet.cpp" -#include "PyGreenletUnswitchable.cpp" -#include "CObjects.cpp" - -using greenlet::LockGuard; -using greenlet::LockInitError; -using greenlet::PyErrOccurred; -using greenlet::Require; - -using greenlet::g_handle_exit; -using greenlet::single_result; - -using greenlet::Greenlet; -using greenlet::UserGreenlet; -using greenlet::MainGreenlet; -using greenlet::BrokenGreenlet; -using greenlet::ThreadState; -using greenlet::PythonState; - - - -// ******* Implementation of things from included files -template -greenlet::refs::_BorrowedGreenlet& greenlet::refs::_BorrowedGreenlet::operator=(const greenlet::refs::BorrowedObject& other) -{ - this->_set_raw_pointer(static_cast(other)); - return *this; -} - -template -inline greenlet::refs::_BorrowedGreenlet::operator Greenlet*() const noexcept -{ - if (!this->p) { - return nullptr; - } - return reinterpret_cast(this->p)->pimpl; -} - -template -greenlet::refs::_BorrowedGreenlet::_BorrowedGreenlet(const BorrowedObject& p) - : BorrowedReference(nullptr) -{ - - this->_set_raw_pointer(p.borrow()); -} - -template -inline greenlet::refs::_OwnedGreenlet::operator Greenlet*() const noexcept -{ - if (!this->p) { - return nullptr; - } - return reinterpret_cast(this->p)->pimpl; -} - - - -#ifdef __clang__ -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wmissing-field-initializers" -# pragma clang diagnostic ignored "-Wwritable-strings" -#elif defined(__GNUC__) -# pragma GCC diagnostic push -// warning: ISO C++ forbids converting a string constant to ‘char*’ -// (The python APIs aren't const correct and accept writable char*) -# pragma GCC diagnostic ignored "-Wwrite-strings" -#endif - - -/*********************************************************** - -A PyGreenlet is a range of C stack addresses that must be -saved and restored in such a way that the full range of the -stack contains valid data when we switch to it. - -Stack layout for a greenlet: - - | ^^^ | - | older data | - | | - stack_stop . |_______________| - . | | - . | greenlet data | - . | in stack | - . * |_______________| . . _____________ stack_copy + stack_saved - . | | | | - . | data | |greenlet data| - . | unrelated | | saved | - . | to | | in heap | - stack_start . | this | . . |_____________| stack_copy - | greenlet | - | | - | newer data | - | vvv | - - -Note that a greenlet's stack data is typically partly at its correct -place in the stack, and partly saved away in the heap, but always in -the above configuration: two blocks, the more recent one in the heap -and the older one still in the stack (either block may be empty). - -Greenlets are chained: each points to the previous greenlet, which is -the one that owns the data currently in the C stack above my -stack_stop. The currently running greenlet is the first element of -this chain. The main (initial) greenlet is the last one. Greenlets -whose stack is entirely in the heap can be skipped from the chain. - -The chain is not related to execution order, but only to the order -in which bits of C stack happen to belong to greenlets at a particular -point in time. - -The main greenlet doesn't have a stack_stop: it is responsible for the -complete rest of the C stack, and we don't know where it begins. We -use (char*) -1, the largest possible address. - -States: - stack_stop == NULL && stack_start == NULL: did not start yet - stack_stop != NULL && stack_start == NULL: already finished - stack_stop != NULL && stack_start != NULL: active - -The running greenlet's stack_start is undefined but not NULL. - - ***********************************************************/ - - - - -/***********************************************************/ - -/* Some functions must not be inlined: - * slp_restore_state, when inlined into slp_switch might cause - it to restore stack over its own local variables - * slp_save_state, when inlined would add its own local - variables to the saved stack, wasting space - * slp_switch, cannot be inlined for obvious reasons - * g_initialstub, when inlined would receive a pointer into its - own stack frame, leading to incomplete stack save/restore - -g_initialstub is a member function and declared virtual so that the -compiler always calls it through a vtable. - -slp_save_state and slp_restore_state are also member functions. They -are called from trampoline functions that themselves are declared as -not eligible for inlining. -*/ - -extern "C" { -static int GREENLET_NOINLINE(slp_save_state_trampoline)(char* stackref) -{ - return switching_thread_state->slp_save_state(stackref); -} -static void GREENLET_NOINLINE(slp_restore_state_trampoline)() -{ - switching_thread_state->slp_restore_state(); -} -} - - -/***********************************************************/ - - -#include "PyModule.cpp" - - - -static PyObject* -greenlet_internal_mod_init() noexcept -{ - static void* _PyGreenlet_API[PyGreenlet_API_pointers]; - - try { - CreatedModule m(greenlet_module_def); - - Require(PyType_Ready(&PyGreenlet_Type)); - Require(PyType_Ready(&PyGreenletUnswitchable_Type)); - - mod_globs = new greenlet::GreenletGlobals; - ThreadState::init(); - - m.PyAddObject("greenlet", PyGreenlet_Type); - m.PyAddObject("UnswitchableGreenlet", PyGreenletUnswitchable_Type); - m.PyAddObject("error", mod_globs->PyExc_GreenletError); - m.PyAddObject("GreenletExit", mod_globs->PyExc_GreenletExit); - - m.PyAddObject("GREENLET_USE_GC", 1); - m.PyAddObject("GREENLET_USE_TRACING", 1); - m.PyAddObject("GREENLET_USE_CONTEXT_VARS", 1L); - m.PyAddObject("GREENLET_USE_STANDARD_THREADING", 1L); - - OwnedObject clocks_per_sec = OwnedObject::consuming(PyLong_FromSsize_t(CLOCKS_PER_SEC)); - m.PyAddObject("CLOCKS_PER_SEC", clocks_per_sec); - - /* also publish module-level data as attributes of the greentype. */ - // XXX: This is weird, and enables a strange pattern of - // confusing the class greenlet with the module greenlet; with - // the exception of (possibly) ``getcurrent()``, this - // shouldn't be encouraged so don't add new items here. - for (const char* const* p = copy_on_greentype; *p; p++) { - OwnedObject o = m.PyRequireAttr(*p); - PyDict_SetItemString(PyGreenlet_Type.tp_dict, *p, o.borrow()); - } - - /* - * Expose C API - */ - - /* types */ - _PyGreenlet_API[PyGreenlet_Type_NUM] = (void*)&PyGreenlet_Type; - - /* exceptions */ - _PyGreenlet_API[PyExc_GreenletError_NUM] = (void*)mod_globs->PyExc_GreenletError; - _PyGreenlet_API[PyExc_GreenletExit_NUM] = (void*)mod_globs->PyExc_GreenletExit; - - /* methods */ - _PyGreenlet_API[PyGreenlet_New_NUM] = (void*)PyGreenlet_New; - _PyGreenlet_API[PyGreenlet_GetCurrent_NUM] = (void*)PyGreenlet_GetCurrent; - _PyGreenlet_API[PyGreenlet_Throw_NUM] = (void*)PyGreenlet_Throw; - _PyGreenlet_API[PyGreenlet_Switch_NUM] = (void*)PyGreenlet_Switch; - _PyGreenlet_API[PyGreenlet_SetParent_NUM] = (void*)PyGreenlet_SetParent; - - /* Previously macros, but now need to be functions externally. */ - _PyGreenlet_API[PyGreenlet_MAIN_NUM] = (void*)Extern_PyGreenlet_MAIN; - _PyGreenlet_API[PyGreenlet_STARTED_NUM] = (void*)Extern_PyGreenlet_STARTED; - _PyGreenlet_API[PyGreenlet_ACTIVE_NUM] = (void*)Extern_PyGreenlet_ACTIVE; - _PyGreenlet_API[PyGreenlet_GET_PARENT_NUM] = (void*)Extern_PyGreenlet_GET_PARENT; - - /* XXX: Note that our module name is ``greenlet._greenlet``, but for - backwards compatibility with existing C code, we need the _C_API to - be directly in greenlet. - */ - const NewReference c_api_object(Require( - PyCapsule_New( - (void*)_PyGreenlet_API, - "greenlet._C_API", - NULL))); - m.PyAddObject("_C_API", c_api_object); - assert(c_api_object.REFCNT() == 2); - - // cerr << "Sizes:" - // << "\n\tGreenlet : " << sizeof(Greenlet) - // << "\n\tUserGreenlet : " << sizeof(UserGreenlet) - // << "\n\tMainGreenlet : " << sizeof(MainGreenlet) - // << "\n\tExceptionState : " << sizeof(greenlet::ExceptionState) - // << "\n\tPythonState : " << sizeof(greenlet::PythonState) - // << "\n\tStackState : " << sizeof(greenlet::StackState) - // << "\n\tSwitchingArgs : " << sizeof(greenlet::SwitchingArgs) - // << "\n\tOwnedObject : " << sizeof(greenlet::refs::OwnedObject) - // << "\n\tBorrowedObject : " << sizeof(greenlet::refs::BorrowedObject) - // << "\n\tPyGreenlet : " << sizeof(PyGreenlet) - // << endl; - -#ifdef Py_GIL_DISABLED - PyUnstable_Module_SetGIL(m.borrow(), Py_MOD_GIL_NOT_USED); -#endif - return m.borrow(); // But really it's the main reference. - } - catch (const LockInitError& e) { - PyErr_SetString(PyExc_MemoryError, e.what()); - return NULL; - } - catch (const PyErrOccurred&) { - return NULL; - } - -} - -extern "C" { - -PyMODINIT_FUNC -PyInit__greenlet(void) -{ - return greenlet_internal_mod_init(); -} - -}; // extern C - -#ifdef __clang__ -# pragma clang diagnostic pop -#elif defined(__GNUC__) -# pragma GCC diagnostic pop -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet.h b/.env/Lib/site-packages/greenlet/greenlet.h deleted file mode 100644 index d02a16e..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet.h +++ /dev/null @@ -1,164 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ - -/* Greenlet object interface */ - -#ifndef Py_GREENLETOBJECT_H -#define Py_GREENLETOBJECT_H - - -#include - -#ifdef __cplusplus -extern "C" { -#endif - -/* This is deprecated and undocumented. It does not change. */ -#define GREENLET_VERSION "1.0.0" - -#ifndef GREENLET_MODULE -#define implementation_ptr_t void* -#endif - -typedef struct _greenlet { - PyObject_HEAD - PyObject* weakreflist; - PyObject* dict; - implementation_ptr_t pimpl; -} PyGreenlet; - -#define PyGreenlet_Check(op) (op && PyObject_TypeCheck(op, &PyGreenlet_Type)) - - -/* C API functions */ - -/* Total number of symbols that are exported */ -#define PyGreenlet_API_pointers 12 - -#define PyGreenlet_Type_NUM 0 -#define PyExc_GreenletError_NUM 1 -#define PyExc_GreenletExit_NUM 2 - -#define PyGreenlet_New_NUM 3 -#define PyGreenlet_GetCurrent_NUM 4 -#define PyGreenlet_Throw_NUM 5 -#define PyGreenlet_Switch_NUM 6 -#define PyGreenlet_SetParent_NUM 7 - -#define PyGreenlet_MAIN_NUM 8 -#define PyGreenlet_STARTED_NUM 9 -#define PyGreenlet_ACTIVE_NUM 10 -#define PyGreenlet_GET_PARENT_NUM 11 - -#ifndef GREENLET_MODULE -/* This section is used by modules that uses the greenlet C API */ -static void** _PyGreenlet_API = NULL; - -# define PyGreenlet_Type \ - (*(PyTypeObject*)_PyGreenlet_API[PyGreenlet_Type_NUM]) - -# define PyExc_GreenletError \ - ((PyObject*)_PyGreenlet_API[PyExc_GreenletError_NUM]) - -# define PyExc_GreenletExit \ - ((PyObject*)_PyGreenlet_API[PyExc_GreenletExit_NUM]) - -/* - * PyGreenlet_New(PyObject *args) - * - * greenlet.greenlet(run, parent=None) - */ -# define PyGreenlet_New \ - (*(PyGreenlet * (*)(PyObject * run, PyGreenlet * parent)) \ - _PyGreenlet_API[PyGreenlet_New_NUM]) - -/* - * PyGreenlet_GetCurrent(void) - * - * greenlet.getcurrent() - */ -# define PyGreenlet_GetCurrent \ - (*(PyGreenlet * (*)(void)) _PyGreenlet_API[PyGreenlet_GetCurrent_NUM]) - -/* - * PyGreenlet_Throw( - * PyGreenlet *greenlet, - * PyObject *typ, - * PyObject *val, - * PyObject *tb) - * - * g.throw(...) - */ -# define PyGreenlet_Throw \ - (*(PyObject * (*)(PyGreenlet * self, \ - PyObject * typ, \ - PyObject * val, \ - PyObject * tb)) \ - _PyGreenlet_API[PyGreenlet_Throw_NUM]) - -/* - * PyGreenlet_Switch(PyGreenlet *greenlet, PyObject *args) - * - * g.switch(*args, **kwargs) - */ -# define PyGreenlet_Switch \ - (*(PyObject * \ - (*)(PyGreenlet * greenlet, PyObject * args, PyObject * kwargs)) \ - _PyGreenlet_API[PyGreenlet_Switch_NUM]) - -/* - * PyGreenlet_SetParent(PyObject *greenlet, PyObject *new_parent) - * - * g.parent = new_parent - */ -# define PyGreenlet_SetParent \ - (*(int (*)(PyGreenlet * greenlet, PyGreenlet * nparent)) \ - _PyGreenlet_API[PyGreenlet_SetParent_NUM]) - -/* - * PyGreenlet_GetParent(PyObject* greenlet) - * - * return greenlet.parent; - * - * This could return NULL even if there is no exception active. - * If it does not return NULL, you are responsible for decrementing the - * reference count. - */ -# define PyGreenlet_GetParent \ - (*(PyGreenlet* (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_GET_PARENT_NUM]) - -/* - * deprecated, undocumented alias. - */ -# define PyGreenlet_GET_PARENT PyGreenlet_GetParent - -# define PyGreenlet_MAIN \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_MAIN_NUM]) - -# define PyGreenlet_STARTED \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_STARTED_NUM]) - -# define PyGreenlet_ACTIVE \ - (*(int (*)(PyGreenlet*)) \ - _PyGreenlet_API[PyGreenlet_ACTIVE_NUM]) - - - - -/* Macro that imports greenlet and initializes C API */ -/* NOTE: This has actually moved to ``greenlet._greenlet._C_API``, but we - keep the older definition to be sure older code that might have a copy of - the header still works. */ -# define PyGreenlet_Import() \ - { \ - _PyGreenlet_API = (void**)PyCapsule_Import("greenlet._C_API", 0); \ - } - -#endif /* GREENLET_MODULE */ - -#ifdef __cplusplus -} -#endif -#endif /* !Py_GREENLETOBJECT_H */ diff --git a/.env/Lib/site-packages/greenlet/greenlet_allocator.hpp b/.env/Lib/site-packages/greenlet/greenlet_allocator.hpp deleted file mode 100644 index 1cd9223..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_allocator.hpp +++ /dev/null @@ -1,76 +0,0 @@ -#ifndef GREENLET_ALLOCATOR_HPP -#define GREENLET_ALLOCATOR_HPP - -#define PY_SSIZE_T_CLEAN -#include -#include -#include "greenlet_compiler_compat.hpp" -#include "greenlet_cpython_compat.hpp" - - -namespace greenlet -{ - // This allocator is stateless; all instances are identical. - // It can *ONLY* be used when we're sure we're holding the GIL - // (Python's allocators require the GIL). - template - struct PythonAllocator : public std::allocator { - - PythonAllocator(const PythonAllocator& UNUSED(other)) - : std::allocator() - { - } - - PythonAllocator(const std::allocator other) - : std::allocator(other) - {} - - template - PythonAllocator(const std::allocator& other) - : std::allocator(other) - { - } - - PythonAllocator() : std::allocator() {} - - T* allocate(size_t number_objects, const void* UNUSED(hint)=0) - { - void* p; - if (number_objects == 1) { -#ifdef Py_GIL_DISABLED - p = PyMem_Malloc(sizeof(T) * number_objects); -#else - p = PyObject_Malloc(sizeof(T)); -#endif - } - else { - p = PyMem_Malloc(sizeof(T) * number_objects); - } - return static_cast(p); - } - - void deallocate(T* t, size_t n) - { - void* p = t; - if (n == 1) { -#ifdef Py_GIL_DISABLED - PyMem_Free(p); -#else - PyObject_Free(p); -#endif - } - else { - PyMem_Free(p); - } - } - // This member is deprecated in C++17 and removed in C++20 - template< class U > - struct rebind { - typedef PythonAllocator other; - }; - - }; - -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet_compiler_compat.hpp b/.env/Lib/site-packages/greenlet/greenlet_compiler_compat.hpp deleted file mode 100644 index af24bd8..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_compiler_compat.hpp +++ /dev/null @@ -1,98 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -#ifndef GREENLET_COMPILER_COMPAT_HPP -#define GREENLET_COMPILER_COMPAT_HPP - -/** - * Definitions to aid with compatibility with different compilers. - * - * .. caution:: Use extreme care with noexcept. - * Some compilers and runtimes, specifically gcc/libgcc/libstdc++ on - * Linux, implement stack unwinding by throwing an uncatchable - * exception, one that specifically does not appear to be an active - * exception to the rest of the runtime. If this happens while we're in a noexcept function, - * we have violated our dynamic exception contract, and so the runtime - * will call std::terminate(), which kills the process with the - * unhelpful message "terminate called without an active exception". - * - * This has happened in this scenario: A background thread is running - * a greenlet that has made a native call and released the GIL. - * Meanwhile, the main thread finishes and starts shutting down the - * interpreter. When the background thread is scheduled again and - * attempts to obtain the GIL, it notices that the interpreter is - * exiting and calls ``pthread_exit()``. This in turn starts to unwind - * the stack by throwing that exception. But we had the ``PyCall`` - * functions annotated as noexcept, so the runtime terminated us. - * - * #2 0x00007fab26fec2b7 in std::terminate() () from /lib/x86_64-linux-gnu/libstdc++.so.6 - * #3 0x00007fab26febb3c in __gxx_personality_v0 () from /lib/x86_64-linux-gnu/libstdc++.so.6 - * #4 0x00007fab26f34de6 in ?? () from /lib/x86_64-linux-gnu/libgcc_s.so.1 - * #6 0x00007fab276a34c6 in __GI___pthread_unwind at ./nptl/unwind.c:130 - * #7 0x00007fab2769bd3a in __do_cancel () at ../sysdeps/nptl/pthreadP.h:280 - * #8 __GI___pthread_exit (value=value@entry=0x0) at ./nptl/pthread_exit.c:36 - * #9 0x000000000052e567 in PyThread_exit_thread () at ../Python/thread_pthread.h:370 - * #10 0x00000000004d60b5 in take_gil at ../Python/ceval_gil.h:224 - * #11 0x00000000004d65f9 in PyEval_RestoreThread at ../Python/ceval.c:467 - * #12 0x000000000060cce3 in setipaddr at ../Modules/socketmodule.c:1203 - * #13 0x00000000006101cd in socket_gethostbyname - */ - -#include - -# define G_NO_COPIES_OF_CLS(Cls) private: \ - Cls(const Cls& other) = delete; \ - Cls& operator=(const Cls& other) = delete - -# define G_NO_ASSIGNMENT_OF_CLS(Cls) private: \ - Cls& operator=(const Cls& other) = delete - -# define G_NO_COPY_CONSTRUCTOR_OF_CLS(Cls) private: \ - Cls(const Cls& other) = delete; - - -// CAUTION: MSVC is stupidly picky: -// -// "The compiler ignores, without warning, any __declspec keywords -// placed after * or & and in front of the variable identifier in a -// declaration." -// (https://docs.microsoft.com/en-us/cpp/cpp/declspec?view=msvc-160) -// -// So pointer return types must be handled differently (because of the -// trailing *), or you get inscrutable compiler warnings like "error -// C2059: syntax error: ''" -// -// In C++ 11, there is a standard syntax for attributes, and -// GCC defines an attribute to use with this: [[gnu:noinline]]. -// In the future, this is expected to become standard. - -#if defined(__GNUC__) || defined(__clang__) -/* We used to check for GCC 4+ or 3.4+, but those compilers are - laughably out of date. Just assume they support it. */ -# define GREENLET_NOINLINE(name) __attribute__((noinline)) name -# define GREENLET_NOINLINE_P(rtype, name) rtype __attribute__((noinline)) name -# define UNUSED(x) UNUSED_ ## x __attribute__((__unused__)) -#elif defined(_MSC_VER) -/* We used to check for && (_MSC_VER >= 1300) but that's also out of date. */ -# define GREENLET_NOINLINE(name) __declspec(noinline) name -# define GREENLET_NOINLINE_P(rtype, name) __declspec(noinline) rtype name -# define UNUSED(x) UNUSED_ ## x -#endif - -#if defined(_MSC_VER) -# define G_NOEXCEPT_WIN32 noexcept -#else -# define G_NOEXCEPT_WIN32 -#endif - -#if defined(__GNUC__) && defined(__POWERPC__) && defined(__APPLE__) -// 32-bit PPC/MacOSX. Only known to be tested on unreleased versions -// of macOS 10.6 using a macports build gcc 14. It appears that -// running C++ destructors of thread-local variables is broken. - -// See https://github.com/python-greenlet/greenlet/pull/419 -# define GREENLET_BROKEN_THREAD_LOCAL_CLEANUP_JUST_LEAK 1 -#else -# define GREENLET_BROKEN_THREAD_LOCAL_CLEANUP_JUST_LEAK 0 -#endif - - -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet_cpython_compat.hpp b/.env/Lib/site-packages/greenlet/greenlet_cpython_compat.hpp deleted file mode 100644 index f46d977..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_cpython_compat.hpp +++ /dev/null @@ -1,156 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -#ifndef GREENLET_CPYTHON_COMPAT_H -#define GREENLET_CPYTHON_COMPAT_H - -/** - * Helpers for compatibility with multiple versions of CPython. - */ - -#define PY_SSIZE_T_CLEAN -#include "Python.h" - - -#if PY_VERSION_HEX >= 0x30A00B1 -# define GREENLET_PY310 1 -#else -# define GREENLET_PY310 0 -#endif - -/* -Python 3.10 beta 1 changed tstate->use_tracing to a nested cframe member. -See https://github.com/python/cpython/pull/25276 -We have to save and restore this as well. - -Python 3.13 removed PyThreadState.cframe (GH-108035). -*/ -#if GREENLET_PY310 && PY_VERSION_HEX < 0x30D0000 -# define GREENLET_USE_CFRAME 1 -#else -# define GREENLET_USE_CFRAME 0 -#endif - - -#if PY_VERSION_HEX >= 0x30B00A4 -/* -Greenlet won't compile on anything older than Python 3.11 alpha 4 (see -https://bugs.python.org/issue46090). Summary of breaking internal changes: -- Python 3.11 alpha 1 changed how frame objects are represented internally. - - https://github.com/python/cpython/pull/30122 -- Python 3.11 alpha 3 changed how recursion limits are stored. - - https://github.com/python/cpython/pull/29524 -- Python 3.11 alpha 4 changed how exception state is stored. It also includes a - change to help greenlet save and restore the interpreter frame "data stack". - - https://github.com/python/cpython/pull/30122 - - https://github.com/python/cpython/pull/30234 -*/ -# define GREENLET_PY311 1 -#else -# define GREENLET_PY311 0 -#endif - - -#if PY_VERSION_HEX >= 0x30C0000 -# define GREENLET_PY312 1 -#else -# define GREENLET_PY312 0 -#endif - -#if PY_VERSION_HEX >= 0x30D0000 -# define GREENLET_PY313 1 -#else -# define GREENLET_PY313 0 -#endif - -#if PY_VERSION_HEX >= 0x30E0000 -# define GREENLET_PY314 1 -#else -# define GREENLET_PY314 0 -#endif - -#if PY_VERSION_HEX >= 0x30F0000 -# define GREENLET_PY315 1 -#else -# define GREENLET_PY315 0 -#endif - -#ifndef Py_SET_REFCNT -/* Py_REFCNT and Py_SIZE macros are converted to functions -https://bugs.python.org/issue39573 */ -# define Py_SET_REFCNT(obj, refcnt) Py_REFCNT(obj) = (refcnt) -#endif - -#ifdef _Py_DEC_REFTOTAL -# define GREENLET_Py_DEC_REFTOTAL _Py_DEC_REFTOTAL -#else -/* _Py_DEC_REFTOTAL macro has been removed from Python 3.9 by: - https://github.com/python/cpython/commit/49932fec62c616ec88da52642339d83ae719e924 - - The symbol we use to replace it was removed by at least 3.12. -*/ -# ifdef Py_REF_DEBUG -# if GREENLET_PY312 -# define GREENLET_Py_DEC_REFTOTAL -# else -# define GREENLET_Py_DEC_REFTOTAL _Py_RefTotal-- -# endif -# else -# define GREENLET_Py_DEC_REFTOTAL -# endif -#endif -// Define these flags like Cython does if we're on an old version. -#ifndef Py_TPFLAGS_CHECKTYPES - #define Py_TPFLAGS_CHECKTYPES 0 -#endif -#ifndef Py_TPFLAGS_HAVE_INDEX - #define Py_TPFLAGS_HAVE_INDEX 0 -#endif -#ifndef Py_TPFLAGS_HAVE_NEWBUFFER - #define Py_TPFLAGS_HAVE_NEWBUFFER 0 -#endif - -#ifndef Py_TPFLAGS_HAVE_VERSION_TAG - #define Py_TPFLAGS_HAVE_VERSION_TAG 0 -#endif - -#define G_TPFLAGS_DEFAULT Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_VERSION_TAG | Py_TPFLAGS_CHECKTYPES | Py_TPFLAGS_HAVE_NEWBUFFER | Py_TPFLAGS_HAVE_GC - - -#if PY_VERSION_HEX < 0x03090000 -// The official version only became available in 3.9 -# define PyObject_GC_IsTracked(o) _PyObject_GC_IS_TRACKED(o) -#endif - - -// bpo-43760 added PyThreadState_EnterTracing() to Python 3.11.0a2 -#if PY_VERSION_HEX < 0x030B00A2 && !defined(PYPY_VERSION) -static inline void PyThreadState_EnterTracing(PyThreadState *tstate) -{ - tstate->tracing++; -#if PY_VERSION_HEX >= 0x030A00A1 - tstate->cframe->use_tracing = 0; -#else - tstate->use_tracing = 0; -#endif -} -#endif - -// bpo-43760 added PyThreadState_LeaveTracing() to Python 3.11.0a2 -#if PY_VERSION_HEX < 0x030B00A2 && !defined(PYPY_VERSION) -static inline void PyThreadState_LeaveTracing(PyThreadState *tstate) -{ - tstate->tracing--; - int use_tracing = (tstate->c_tracefunc != NULL - || tstate->c_profilefunc != NULL); -#if PY_VERSION_HEX >= 0x030A00A1 - tstate->cframe->use_tracing = use_tracing; -#else - tstate->use_tracing = use_tracing; -#endif -} -#endif - -#if !defined(Py_C_RECURSION_LIMIT) && defined(C_RECURSION_LIMIT) -# define Py_C_RECURSION_LIMIT C_RECURSION_LIMIT -#endif - -#endif /* GREENLET_CPYTHON_COMPAT_H */ diff --git a/.env/Lib/site-packages/greenlet/greenlet_exceptions.hpp b/.env/Lib/site-packages/greenlet/greenlet_exceptions.hpp deleted file mode 100644 index 617f07c..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_exceptions.hpp +++ /dev/null @@ -1,171 +0,0 @@ -#ifndef GREENLET_EXCEPTIONS_HPP -#define GREENLET_EXCEPTIONS_HPP - -#define PY_SSIZE_T_CLEAN -#include -#include -#include - -#ifdef __clang__ -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wunused-function" -#endif - -namespace greenlet { - - class PyErrOccurred : public std::runtime_error - { - public: - - // CAUTION: In debug builds, may run arbitrary Python code. - static const PyErrOccurred - from_current() - { - assert(PyErr_Occurred()); -#ifndef NDEBUG - // This is not exception safe, and - // not necessarily safe in general (what if it switches?) - // But we only do this in debug mode, where we are in - // tight control of what exceptions are getting raised and - // can prevent those issues. - - // You can't call PyObject_Str with a pending exception. - PyObject* typ; - PyObject* val; - PyObject* tb; - - PyErr_Fetch(&typ, &val, &tb); - PyObject* typs = PyObject_Str(typ); - PyObject* vals = PyObject_Str(val ? val : typ); - const char* typ_msg = PyUnicode_AsUTF8(typs); - const char* val_msg = PyUnicode_AsUTF8(vals); - PyErr_Restore(typ, val, tb); - - std::string msg(typ_msg); - msg += ": "; - msg += val_msg; - PyErrOccurred ex(msg); - Py_XDECREF(typs); - Py_XDECREF(vals); - - return ex; -#else - return PyErrOccurred(); -#endif - } - - PyErrOccurred() : std::runtime_error("") - { - assert(PyErr_Occurred()); - } - - PyErrOccurred(const std::string& msg) : std::runtime_error(msg) - { - assert(PyErr_Occurred()); - } - - PyErrOccurred(PyObject* exc_kind, const char* const msg) - : std::runtime_error(msg) - { - PyErr_SetString(exc_kind, msg); - } - - PyErrOccurred(PyObject* exc_kind, const std::string msg) - : std::runtime_error(msg) - { - // This copies the c_str, so we don't have any lifetime - // issues to worry about. - PyErr_SetString(exc_kind, msg.c_str()); - } - - PyErrOccurred(PyObject* exc_kind, - const std::string msg, //This is the format - //string; that's not - //usually safe! - - PyObject* borrowed_obj_one, PyObject* borrowed_obj_two) - : std::runtime_error(msg) - { - - //This is designed specifically for the - //``check_switch_allowed`` function. - - // PyObject_Str and PyObject_Repr are safe to call with - // NULL pointers; they return the string "" in that - // case. - // This function always returns null. - PyErr_Format(exc_kind, - msg.c_str(), - borrowed_obj_one, borrowed_obj_two); - } - }; - - class TypeError : public PyErrOccurred - { - public: - TypeError(const char* const what) - : PyErrOccurred(PyExc_TypeError, what) - { - } - TypeError(const std::string what) - : PyErrOccurred(PyExc_TypeError, what) - { - } - }; - - class ValueError : public PyErrOccurred - { - public: - ValueError(const char* const what) - : PyErrOccurred(PyExc_ValueError, what) - { - } - }; - - class AttributeError : public PyErrOccurred - { - public: - AttributeError(const char* const what) - : PyErrOccurred(PyExc_AttributeError, what) - { - } - }; - - /** - * Calls `Py_FatalError` when constructed, so you can't actually - * throw this. It just makes static analysis easier. - */ - class PyFatalError : public std::runtime_error - { - public: - PyFatalError(const char* const msg) - : std::runtime_error(msg) - { - Py_FatalError(msg); - } - }; - - static inline PyObject* - Require(PyObject* p, const std::string& msg="") - { - if (!p) { - throw PyErrOccurred(msg); - } - return p; - }; - - static inline void - Require(const int retval) - { - if (retval < 0) { - throw PyErrOccurred(); - } - }; - - -}; -#ifdef __clang__ -# pragma clang diagnostic pop -#endif - -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet_internal.hpp b/.env/Lib/site-packages/greenlet/greenlet_internal.hpp deleted file mode 100644 index f2b15d5..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_internal.hpp +++ /dev/null @@ -1,107 +0,0 @@ -/* -*- indent-tabs-mode: nil; tab-width: 4; -*- */ -#ifndef GREENLET_INTERNAL_H -#define GREENLET_INTERNAL_H -#ifdef __clang__ -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wunused-function" -#endif - -/** - * Implementation helpers. - * - * C++ templates and inline functions should go here. - */ -#define PY_SSIZE_T_CLEAN -#include "greenlet_compiler_compat.hpp" -#include "greenlet_cpython_compat.hpp" -#include "greenlet_exceptions.hpp" -#include "TGreenlet.hpp" -#include "greenlet_allocator.hpp" - -#include -#include - -#define GREENLET_MODULE -struct _greenlet; -typedef struct _greenlet PyGreenlet; -namespace greenlet { - - class ThreadState; - // We can't use the PythonAllocator for this, because we push to it - // from the thread state destructor, which doesn't have the GIL, - // and Python's allocators can only be called with the GIL. - typedef std::vector cleanup_queue_t; - -}; - - -#define implementation_ptr_t greenlet::Greenlet* - - -#include "greenlet.h" - -void -greenlet::refs::MainGreenletExactChecker(void *p) -{ - if (!p) { - return; - } - // We control the class of the main greenlet exactly. - if (Py_TYPE(p) != &PyGreenlet_Type) { - std::string err("MainGreenlet: Expected exactly a greenlet, not a "); - err += Py_TYPE(p)->tp_name; - throw greenlet::TypeError(err); - } - - // Greenlets from dead threads no longer respond to main() with a - // true value; so in that case we need to perform an additional - // check. - Greenlet* g = static_cast(p)->pimpl; - if (g->main()) { - return; - } - if (!dynamic_cast(g)) { - std::string err("MainGreenlet: Expected exactly a main greenlet, not a "); - err += Py_TYPE(p)->tp_name; - throw greenlet::TypeError(err); - } -} - - - -template -inline greenlet::Greenlet* greenlet::refs::_OwnedGreenlet::operator->() const noexcept -{ - return reinterpret_cast(this->p)->pimpl; -} - -template -inline greenlet::Greenlet* greenlet::refs::_BorrowedGreenlet::operator->() const noexcept -{ - return reinterpret_cast(this->p)->pimpl; -} - -#include -#include - - -extern PyTypeObject PyGreenlet_Type; - - - -/** - * Forward declarations needed in multiple files. - */ -static PyObject* green_switch(PyGreenlet* self, PyObject* args, PyObject* kwargs); - - -#ifdef __clang__ -# pragma clang diagnostic pop -#endif - - -#endif - -// Local Variables: -// flycheck-clang-include-path: ("../../include" "/opt/local/Library/Frameworks/Python.framework/Versions/3.10/include/python3.10") -// End: diff --git a/.env/Lib/site-packages/greenlet/greenlet_msvc_compat.hpp b/.env/Lib/site-packages/greenlet/greenlet_msvc_compat.hpp deleted file mode 100644 index 9635a1b..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_msvc_compat.hpp +++ /dev/null @@ -1,100 +0,0 @@ -#ifndef GREENLET_MSVC_COMPAT_HPP -#define GREENLET_MSVC_COMPAT_HPP -/* - * Support for MSVC on Windows. - * - * Beginning with Python 3.14, some of the internal - * include files we need are not compatible with MSVC - * in C++ mode: - * - * internal\pycore_stackref.h(253): error C4576: a parenthesized type - * followed by an initializer list is a non-standard explicit type conversion syntax - * - * This file is included from ``internal/pycore_interpframe.h``, which - * we need for the ``_PyFrame_IsIncomplete`` API. - * - * Unfortunately, that API is a ``static inline`` function, as are a - * bunch of the functions it calls. The only solution seems to be to - * copy those definitions and the supporting inline functions here. - * - * Now, this makes us VERY fragile to changes in those functions. Because - * they're internal and static, the CPython devs might feel free to change - * them in even minor versions, meaning that we could runtime link and load, - * but still crash. We have that problem on all platforms though. It's just worse - * here because we have to keep copying the updated definitions. - */ -#include -#include "greenlet_cpython_compat.hpp" - -// This file is only included on 3.14+ - -extern "C" { - -// pycore_code.h ---------------- -#define _PyCode_CODE(CO) _Py_RVALUE((_Py_CODEUNIT *)(CO)->co_code_adaptive) - -#ifdef Py_GIL_DISABLED -static inline _PyCodeArray * -_PyCode_GetTLBCArray(PyCodeObject *co) -{ - return _Py_STATIC_CAST(_PyCodeArray *, - _Py_atomic_load_ptr_acquire(&co->co_tlbc)); -} -#endif -// End pycore_code.h ---------- - -// pycore_interpframe.h ---------- -#if !defined(Py_GIL_DISABLED) && defined(Py_STACKREF_DEBUG) - -#define Py_TAG_BITS 0 -#else -#define Py_TAG_BITS ((uintptr_t)1) -#define Py_TAG_DEFERRED (1) -#endif - - -static const _PyStackRef PyStackRef_NULL = { .bits = Py_TAG_DEFERRED}; -#define PyStackRef_IsNull(stackref) ((stackref).bits == PyStackRef_NULL.bits) - -static inline PyObject * -PyStackRef_AsPyObjectBorrow(_PyStackRef stackref) -{ - PyObject *cleared = ((PyObject *)((stackref).bits & (~Py_TAG_BITS))); - return cleared; -} - -static inline PyCodeObject *_PyFrame_GetCode(_PyInterpreterFrame *f) { - assert(!PyStackRef_IsNull(f->f_executable)); - PyObject *executable = PyStackRef_AsPyObjectBorrow(f->f_executable); - assert(PyCode_Check(executable)); - return (PyCodeObject *)executable; -} - - -static inline _Py_CODEUNIT * -_PyFrame_GetBytecode(_PyInterpreterFrame *f) -{ -#ifdef Py_GIL_DISABLED - PyCodeObject *co = _PyFrame_GetCode(f); - _PyCodeArray *tlbc = _PyCode_GetTLBCArray(co); - assert(f->tlbc_index >= 0 && f->tlbc_index < tlbc->size); - return (_Py_CODEUNIT *)tlbc->entries[f->tlbc_index]; -#else - return _PyCode_CODE(_PyFrame_GetCode(f)); -#endif -} - -static inline bool //_Py_NO_SANITIZE_THREAD -_PyFrame_IsIncomplete(_PyInterpreterFrame *frame) -{ - if (frame->owner >= FRAME_OWNED_BY_INTERPRETER) { - return true; - } - return frame->owner != FRAME_OWNED_BY_GENERATOR && - frame->instr_ptr < _PyFrame_GetBytecode(frame) + - _PyFrame_GetCode(frame)->_co_firsttraceable; -} -// pycore_interpframe.h ---------- - -} -#endif // GREENLET_MSVC_COMPAT_HPP diff --git a/.env/Lib/site-packages/greenlet/greenlet_refs.hpp b/.env/Lib/site-packages/greenlet/greenlet_refs.hpp deleted file mode 100644 index b7e5e3f..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_refs.hpp +++ /dev/null @@ -1,1118 +0,0 @@ -#ifndef GREENLET_REFS_HPP -#define GREENLET_REFS_HPP - -#define PY_SSIZE_T_CLEAN -#include - -#include - -//#include "greenlet_internal.hpp" -#include "greenlet_compiler_compat.hpp" -#include "greenlet_cpython_compat.hpp" -#include "greenlet_exceptions.hpp" - -struct _greenlet; -struct _PyMainGreenlet; - -typedef struct _greenlet PyGreenlet; -extern PyTypeObject PyGreenlet_Type; - - -#ifdef GREENLET_USE_STDIO -#include -using std::cerr; -using std::endl; -#endif - -namespace greenlet -{ - class Greenlet; - - namespace refs - { - // Type checkers throw a TypeError if the argument is not - // null, and isn't of the required Python type. - // (We can't use most of the defined type checkers - // like PyList_Check, etc, directly, because they are - // implemented as macros.) - typedef void (*TypeChecker)(void*); - - void - NoOpChecker(void*) - { - return; - } - - void - GreenletChecker(void *p) - { - if (!p) { - return; - } - - PyTypeObject* typ = Py_TYPE(p); - // fast, common path. (PyObject_TypeCheck is a macro or - // static inline function, and it also does a - // direct comparison of the type pointers, but its fast - // path only handles one type) - if (typ == &PyGreenlet_Type) { - return; - } - - if (!PyObject_TypeCheck(p, &PyGreenlet_Type)) { - std::string err("GreenletChecker: Expected any type of greenlet, not "); - err += Py_TYPE(p)->tp_name; - throw TypeError(err); - } - } - - void - MainGreenletExactChecker(void *p); - - template - class PyObjectPointer; - - template - class OwnedReference; - - - template - class BorrowedReference; - - typedef BorrowedReference BorrowedObject; - typedef OwnedReference OwnedObject; - - class ImmortalObject; - class ImmortalString; - - template - class _OwnedGreenlet; - - typedef _OwnedGreenlet OwnedGreenlet; - typedef _OwnedGreenlet OwnedMainGreenlet; - - template - class _BorrowedGreenlet; - - typedef _BorrowedGreenlet BorrowedGreenlet; - - void - ContextExactChecker(void *p) - { - if (!p) { - return; - } - if (!PyContext_CheckExact(p)) { - throw TypeError( - "greenlet context must be a contextvars.Context or None" - ); - } - } - - typedef OwnedReference OwnedContext; - } -} - -namespace greenlet { - - - namespace refs { - // A set of classes to make reference counting rules in python - // code explicit. - // - // Rules of use: - // (1) Functions returning a new reference that the caller of the - // function is expected to dispose of should return a - // ``OwnedObject`` object. This object automatically releases its - // reference when it goes out of scope. It works like a ``std::shared_ptr`` - // and can be copied or used as a function parameter (but don't do - // that). Note that constructing a ``OwnedObject`` from a - // PyObject* steals the reference. - // (2) Parameters to functions should be either a - // ``OwnedObject&``, or, more generally, a ``PyObjectPointer&``. - // If the function needs to create its own new reference, it can - // do so by copying to a local ``OwnedObject``. - // (3) Functions returning an existing pointer that is NOT - // incref'd, and which the caller MUST NOT decref, - // should return a ``BorrowedObject``. - - // XXX: The following two paragraphs do not hold for all platforms. - // Notably, 32-bit PPC Linux passes structs by reference, not by - // value, so this actually doesn't work. (Although that's the only - // platform that doesn't work on.) DO NOT ATTEMPT IT. The - // unfortunate consequence of that is that the slots which we - // *know* are already type safe will wind up calling the type - // checker function (when we had the slots accepting - // BorrowedGreenlet, this was bypassed), so this slows us down. - // TODO: Optimize this again. - - // For a class with a single pointer member, whose constructor - // does nothing but copy a pointer parameter into the member, and - // which can then be converted back to the pointer type, compilers - // generate code that's the same as just passing the pointer. - // That is, func(BorrowedObject x) called like ``PyObject* p = - // ...; f(p)`` has 0 overhead. Similarly, they "unpack" to the - // pointer type with 0 overhead. - // - // If there are no virtual functions, no complex inheritance (maybe?) and - // no destructor, these can be directly used as parameters in - // Python callbacks like tp_init: the layout is the same as a - // single pointer. Only subclasses with trivial constructors that - // do nothing but set the single pointer member are safe to use - // that way. - - - // This is the base class for things that can be done with a - // PyObject pointer. It assumes nothing about memory management. - // NOTE: Nothing is virtual, so subclasses shouldn't add new - // storage fields or try to override these methods. - template - class PyObjectPointer - { - public: - typedef T PyType; - protected: - T* p; - public: - PyObjectPointer(T* it=nullptr) : p(it) - { - TC(p); - } - - // We don't allow automatic casting to PyObject* at this - // level, because then we could be passed to Py_DECREF/INCREF, - // but we want nothing to do with memory management. If you - // know better, then you can use the get() method, like on a - // std::shared_ptr. Except we name it borrow() to clarify that - // if this is a reference-tracked object, the pointer you get - // back will go away when the object does. - // TODO: This should probably not exist here, but be moved - // down to relevant sub-types. - - T* borrow() const noexcept - { - return this->p; - } - - PyObject* borrow_o() const noexcept - { - return reinterpret_cast(this->p); - } - - T* operator->() const noexcept - { - return this->p; - } - - bool is_None() const noexcept - { - return this->p == Py_None; - } - - PyObject* acquire_or_None() const noexcept - { - PyObject* result = this->p ? reinterpret_cast(this->p) : Py_None; - Py_INCREF(result); - return result; - } - - explicit operator bool() const noexcept - { - return this->p != nullptr; - } - - bool operator!() const noexcept - { - return this->p == nullptr; - } - - Py_ssize_t REFCNT() const noexcept - { - return p ? Py_REFCNT(p) : -42; - } - - PyTypeObject* TYPE() const noexcept - { - return p ? Py_TYPE(p) : nullptr; - } - - inline OwnedObject PyStr() const noexcept; - inline const std::string as_str() const noexcept; - inline OwnedObject PyGetAttr(const ImmortalObject& name) const noexcept; - inline OwnedObject PyRequireAttr(const char* const name) const; - inline OwnedObject PyRequireAttr(const ImmortalString& name) const; - inline OwnedObject PyCall(const BorrowedObject& arg) const; - inline OwnedObject PyCall(PyGreenlet* arg) const ; - inline OwnedObject PyCall(PyObject* arg) const ; - // PyObject_Call(this, args, kwargs); - inline OwnedObject PyCall(const BorrowedObject args, - const BorrowedObject kwargs) const; - inline OwnedObject PyCall(const OwnedObject& args, - const OwnedObject& kwargs) const; - - protected: - void _set_raw_pointer(void* t) - { - TC(t); - p = reinterpret_cast(t); - } - void* _get_raw_pointer() const - { - return p; - } - }; - -#ifdef GREENLET_USE_STDIO - template - std::ostream& operator<<(std::ostream& os, const PyObjectPointer& s) - { - const std::type_info& t = typeid(s); - os << t.name() - << "(addr=" << s.borrow() - << ", refcnt=" << s.REFCNT() - << ", value=" << s.as_str() - << ")"; - - return os; - } -#endif - - template - inline bool operator==(const PyObjectPointer& lhs, const PyObject* const rhs) noexcept - { - return static_cast(lhs.borrow_o()) == static_cast(rhs); - } - - template - inline bool operator==(const PyObjectPointer& lhs, const PyObjectPointer& rhs) noexcept - { - return lhs.borrow_o() == rhs.borrow_o(); - } - - template - inline bool operator!=(const PyObjectPointer& lhs, - const PyObjectPointer& rhs) noexcept - { - return lhs.borrow_o() != rhs.borrow_o(); - } - - template - class OwnedReference : public PyObjectPointer - { - private: - friend class OwnedList; - - protected: - explicit OwnedReference(T* it) : PyObjectPointer(it) - { - } - - public: - - // Constructors - - static OwnedReference consuming(PyObject* p) - { - return OwnedReference(reinterpret_cast(p)); - } - - static OwnedReference owning(T* p) - { - OwnedReference result(p); - Py_XINCREF(result.p); - return result; - } - - OwnedReference() : PyObjectPointer(nullptr) - {} - - explicit OwnedReference(const PyObjectPointer<>& other) - : PyObjectPointer(nullptr) - { - T* op = other.borrow(); - TC(op); - this->p = other.borrow(); - Py_XINCREF(this->p); - } - - // It would be good to make use of the C++11 distinction - // between move and copy operations, e.g., constructing from a - // pointer should be a move operation. - // In the common case of ``OwnedObject x = Py_SomeFunction()``, - // the call to the copy constructor will be elided completely. - OwnedReference(const OwnedReference& other) - : PyObjectPointer(other.p) - { - Py_XINCREF(this->p); - } - - static OwnedReference None() - { - Py_INCREF(Py_None); - return OwnedReference(Py_None); - } - - // We can assign from exactly our type without any extra checking - OwnedReference& operator=(const OwnedReference& other) - { - Py_XINCREF(other.p); - const T* tmp = this->p; - this->p = other.p; - Py_XDECREF(tmp); - return *this; - } - - OwnedReference& operator=(const BorrowedReference other) - { - return this->operator=(other.borrow()); - } - - OwnedReference& operator=(T* const other) - { - TC(other); - Py_XINCREF(other); - T* tmp = this->p; - this->p = other; - Py_XDECREF(tmp); - return *this; - } - - // We can assign from an arbitrary reference type - // if it passes our check. - template - OwnedReference& operator=(const OwnedReference& other) - { - X* op = other.borrow(); - TC(op); - return this->operator=(reinterpret_cast(op)); - } - - inline void steal(T* other) - { - assert(this->p == nullptr); - TC(other); - this->p = other; - } - - T* relinquish_ownership() - { - T* result = this->p; - this->p = nullptr; - return result; - } - - T* acquire() const - { - // Return a new reference. - // TODO: This may go away when we have reference objects - // throughout the code. - Py_XINCREF(this->p); - return this->p; - } - - // Nothing else declares a destructor, we're the leaf, so we - // should be able to get away without virtual. - ~OwnedReference() - { - Py_CLEAR(this->p); - } - - void CLEAR() - { - Py_CLEAR(this->p); - assert(this->p == nullptr); - } - }; - - static inline - void operator<<=(PyObject*& target, OwnedObject& o) - { - target = o.relinquish_ownership(); - } - - - class NewReference : public OwnedObject - { - private: - G_NO_COPIES_OF_CLS(NewReference); - public: - // Consumes the reference. Only use this - // for API return values. - NewReference(PyObject* it) : OwnedObject(it) - { - } - }; - - class NewDictReference : public NewReference - { - private: - G_NO_COPIES_OF_CLS(NewDictReference); - public: - NewDictReference() : NewReference(PyDict_New()) - { - if (!this->p) { - throw PyErrOccurred(); - } - } - - void SetItem(const char* const key, PyObject* value) - { - Require(PyDict_SetItemString(this->p, key, value)); - } - - void SetItem(const PyObjectPointer<>& key, PyObject* value) - { - Require(PyDict_SetItem(this->p, key.borrow_o(), value)); - } - }; - - template - class _OwnedGreenlet: public OwnedReference - { - private: - protected: - _OwnedGreenlet(T* it) : OwnedReference(it) - {} - - public: - _OwnedGreenlet() : OwnedReference() - {} - - _OwnedGreenlet(const _OwnedGreenlet& other) : OwnedReference(other) - { - } - _OwnedGreenlet(OwnedMainGreenlet& other) : - OwnedReference(reinterpret_cast(other.acquire())) - { - } - _OwnedGreenlet(const BorrowedGreenlet& other); - // Steals a reference. - static _OwnedGreenlet consuming(PyGreenlet* it) - { - return _OwnedGreenlet(reinterpret_cast(it)); - } - - inline _OwnedGreenlet& operator=(const OwnedGreenlet& other) - { - return this->operator=(other.borrow()); - } - - inline _OwnedGreenlet& operator=(const BorrowedGreenlet& other); - - _OwnedGreenlet& operator=(const OwnedMainGreenlet& other) - { - PyGreenlet* owned = other.acquire(); - Py_XDECREF(this->p); - this->p = reinterpret_cast(owned); - return *this; - } - - _OwnedGreenlet& operator=(T* const other) - { - OwnedReference::operator=(other); - return *this; - } - - T* relinquish_ownership() - { - T* result = this->p; - this->p = nullptr; - return result; - } - - PyObject* relinquish_ownership_o() - { - return reinterpret_cast(relinquish_ownership()); - } - - inline Greenlet* operator->() const noexcept; - inline operator Greenlet*() const noexcept; - }; - - template - class BorrowedReference : public PyObjectPointer - { - public: - // Allow implicit creation from PyObject* pointers as we - // transition to using these classes. Also allow automatic - // conversion to PyObject* for passing to C API calls and even - // for Py_INCREF/DECREF, because we ourselves do no memory management. - BorrowedReference(T* it) : PyObjectPointer(it) - {} - - BorrowedReference(const PyObjectPointer& ref) : PyObjectPointer(ref.borrow()) - {} - - BorrowedReference() : PyObjectPointer(nullptr) - {} - - operator T*() const - { - return this->p; - } - }; - - typedef BorrowedReference BorrowedObject; - //typedef BorrowedReference BorrowedGreenlet; - - template - class _BorrowedGreenlet : public BorrowedReference - { - public: - _BorrowedGreenlet() : - BorrowedReference(nullptr) - {} - - _BorrowedGreenlet(T* it) : - BorrowedReference(it) - {} - - _BorrowedGreenlet(const BorrowedObject& it); - - _BorrowedGreenlet(const OwnedGreenlet& it) : - BorrowedReference(it.borrow()) - {} - - _BorrowedGreenlet& operator=(const BorrowedObject& other); - - // We get one of these for PyGreenlet, but one for PyObject - // is handy as well - operator PyObject*() const - { - return reinterpret_cast(this->p); - } - Greenlet* operator->() const noexcept; - operator Greenlet*() const noexcept; - }; - - typedef _BorrowedGreenlet BorrowedGreenlet; - - template - _OwnedGreenlet::_OwnedGreenlet(const BorrowedGreenlet& other) - : OwnedReference(reinterpret_cast(other.borrow())) - { - Py_XINCREF(this->p); - } - - - class BorrowedMainGreenlet - : public _BorrowedGreenlet - { - public: - BorrowedMainGreenlet(const OwnedMainGreenlet& it) : - _BorrowedGreenlet(it.borrow()) - {} - BorrowedMainGreenlet(PyGreenlet* it=nullptr) - : _BorrowedGreenlet(it) - {} - }; - - template - _OwnedGreenlet& _OwnedGreenlet::operator=(const BorrowedGreenlet& other) - { - return this->operator=(other.borrow()); - } - - - class ImmortalObject : public PyObjectPointer<> - { - private: - G_NO_ASSIGNMENT_OF_CLS(ImmortalObject); - public: - explicit ImmortalObject(PyObject* it) : PyObjectPointer<>(it) - { - } - - ImmortalObject(const ImmortalObject& other) - : PyObjectPointer<>(other.p) - { - - } - - /** - * Become the new owner of the object. Does not change the - * reference count. - */ - ImmortalObject& operator=(PyObject* it) - { - assert(this->p == nullptr); - this->p = it; - return *this; - } - - static ImmortalObject consuming(PyObject* it) - { - return ImmortalObject(it); - } - - inline operator PyObject*() const - { - return this->p; - } - }; - - class ImmortalString : public ImmortalObject - { - private: - G_NO_COPIES_OF_CLS(ImmortalString); - const char* str; - public: - ImmortalString(const char* const str) : - ImmortalObject(str ? Require(PyUnicode_InternFromString(str)) : nullptr) - { - this->str = str; - } - - inline ImmortalString& operator=(const char* const str) - { - if (!this->p) { - this->p = Require(PyUnicode_InternFromString(str)); - this->str = str; - } - else { - assert(this->str == str); - } - return *this; - } - - inline operator std::string() const - { - return this->str; - } - - }; - - class ImmortalEventName : public ImmortalString - { - private: - G_NO_COPIES_OF_CLS(ImmortalEventName); - public: - ImmortalEventName(const char* const str) : ImmortalString(str) - {} - }; - - class ImmortalException : public ImmortalObject - { - private: - G_NO_COPIES_OF_CLS(ImmortalException); - public: - ImmortalException(const char* const name, PyObject* base=nullptr) : - ImmortalObject(name - // Python 2.7 isn't const correct - ? Require(PyErr_NewException((char*)name, base, nullptr)) - : nullptr) - {} - - inline bool PyExceptionMatches() const - { - return PyErr_ExceptionMatches(this->p) > 0; - } - - }; - - template - inline OwnedObject PyObjectPointer::PyStr() const noexcept - { - if (!this->p) { - return OwnedObject(); - } - return OwnedObject::consuming(PyObject_Str(reinterpret_cast(this->p))); - } - - template - inline const std::string PyObjectPointer::as_str() const noexcept - { - // NOTE: This is not Python exception safe. - if (this->p) { - // The Python APIs return a cached char* value that's only valid - // as long as the original object stays around, and we're - // about to (probably) toss it. Hence the copy to std::string. - OwnedObject py_str = this->PyStr(); - if (!py_str) { - return "(nil)"; - } - return PyUnicode_AsUTF8(py_str.borrow()); - } - return "(nil)"; - } - - template - inline OwnedObject PyObjectPointer::PyGetAttr(const ImmortalObject& name) const noexcept - { - assert(this->p); - return OwnedObject::consuming(PyObject_GetAttr(reinterpret_cast(this->p), name)); - } - - template - inline OwnedObject PyObjectPointer::PyRequireAttr(const char* const name) const - { - assert(this->p); - return OwnedObject::consuming(Require(PyObject_GetAttrString(this->p, name), name)); - } - - template - inline OwnedObject PyObjectPointer::PyRequireAttr(const ImmortalString& name) const - { - assert(this->p); - return OwnedObject::consuming(Require( - PyObject_GetAttr( - reinterpret_cast(this->p), - name - ), - name - )); - } - - template - inline OwnedObject PyObjectPointer::PyCall(const BorrowedObject& arg) const - { - return this->PyCall(arg.borrow()); - } - - template - inline OwnedObject PyObjectPointer::PyCall(PyGreenlet* arg) const - { - return this->PyCall(reinterpret_cast(arg)); - } - - template - inline OwnedObject PyObjectPointer::PyCall(PyObject* arg) const - { - assert(this->p); - return OwnedObject::consuming(PyObject_CallFunctionObjArgs(this->p, arg, NULL)); - } - - template - inline OwnedObject PyObjectPointer::PyCall(const BorrowedObject args, - const BorrowedObject kwargs) const - { - assert(this->p); - return OwnedObject::consuming(PyObject_Call(this->p, args, kwargs)); - } - - template - inline OwnedObject PyObjectPointer::PyCall(const OwnedObject& args, - const OwnedObject& kwargs) const - { - assert(this->p); - return OwnedObject::consuming(PyObject_Call(this->p, args.borrow(), kwargs.borrow())); - } - - inline void - ListChecker(void * p) - { - if (!p) { - return; - } - if (!PyList_Check(p)) { - throw TypeError("Expected a list"); - } - } - - class OwnedList : public OwnedReference - { - private: - G_NO_ASSIGNMENT_OF_CLS(OwnedList); - public: - // TODO: Would like to use move. - explicit OwnedList(const OwnedObject& other) - : OwnedReference(other) - { - } - - OwnedList& operator=(const OwnedObject& other) - { - if (other && PyList_Check(other.p)) { - // Valid list. Own a new reference to it, discard the - // reference to what we did own. - PyObject* new_ptr = other.p; - Py_INCREF(new_ptr); - Py_XDECREF(this->p); - this->p = new_ptr; - } - else { - // Either the other object was NULL (an error) or it - // wasn't a list. Either way, we're now invalidated. - Py_XDECREF(this->p); - this->p = nullptr; - } - return *this; - } - - inline bool empty() const - { - return PyList_GET_SIZE(p) == 0; - } - - inline Py_ssize_t size() const - { - return PyList_GET_SIZE(p); - } - - inline BorrowedObject at(const Py_ssize_t index) const - { - return PyList_GET_ITEM(p, index); - } - - inline void clear() - { - PyList_SetSlice(p, 0, PyList_GET_SIZE(p), NULL); - } - }; - - // Use this to represent the module object used at module init - // time. - // This could either be a borrowed (Py2) or new (Py3) reference; - // either way, we don't want to do any memory management - // on it here, Python itself will handle that. - // XXX: Actually, that's not quite right. On Python 3, if an - // exception occurs before we return to the interpreter, this will - // leak; but all previous versions also had that problem. - class CreatedModule : public PyObjectPointer<> - { - private: - G_NO_COPIES_OF_CLS(CreatedModule); - public: - CreatedModule(PyModuleDef& mod_def) : PyObjectPointer<>( - Require(PyModule_Create(&mod_def))) - { - } - - // PyAddObject(): Add a reference to the object to the module. - // On return, the reference count of the object is unchanged. - // - // The docs warn that PyModule_AddObject only steals the - // reference on success, so if it fails after we've incref'd - // or allocated, we're responsible for the decref. - void PyAddObject(const char* name, const long new_bool) - { - OwnedObject p = OwnedObject::consuming(Require(PyBool_FromLong(new_bool))); - this->PyAddObject(name, p); - } - - void PyAddObject(const char* name, const OwnedObject& new_object) - { - // The caller already owns a reference they will decref - // when their variable goes out of scope, we still need to - // incref/decref. - this->PyAddObject(name, new_object.borrow()); - } - - void PyAddObject(const char* name, const ImmortalObject& new_object) - { - this->PyAddObject(name, new_object.borrow()); - } - - void PyAddObject(const char* name, PyTypeObject& type) - { - this->PyAddObject(name, reinterpret_cast(&type)); - } - - void PyAddObject(const char* name, PyObject* new_object) - { - Py_INCREF(new_object); - try { - Require(PyModule_AddObject(this->p, name, new_object)); - } - catch (const PyErrOccurred&) { - Py_DECREF(p); - throw; - } - } - }; - - class PyErrFetchParam : public PyObjectPointer<> - { - // Not an owned object, because we can't be initialized with - // one, and we only sometimes acquire ownership. - private: - G_NO_COPIES_OF_CLS(PyErrFetchParam); - public: - // To allow declaring these and passing them to - // PyErr_Fetch we implement the empty constructor, - // and the address operator. - PyErrFetchParam() : PyObjectPointer<>(nullptr) - { - } - - PyObject** operator&() - { - return &this->p; - } - - // This allows us to pass one directly without the &, - // BUT it has higher precedence than the bool operator - // if it's not explicit. - operator PyObject**() - { - return &this->p; - } - - // We don't want to be able to pass these to Py_DECREF and - // such so we don't have the implicit PyObject* conversion. - - inline PyObject* relinquish_ownership() - { - PyObject* result = this->p; - this->p = nullptr; - return result; - } - - ~PyErrFetchParam() - { - Py_XDECREF(p); - } - }; - - class OwnedErrPiece : public OwnedObject - { - private: - - public: - // Unlike OwnedObject, this increments the refcount. - OwnedErrPiece(PyObject* p=nullptr) : OwnedObject(p) - { - this->acquire(); - } - - PyObject** operator&() - { - return &this->p; - } - - inline operator PyObject*() const - { - return this->p; - } - - operator PyTypeObject*() const - { - return reinterpret_cast(this->p); - } - }; - - class PyErrPieces - { - private: - OwnedErrPiece type; - OwnedErrPiece instance; - OwnedErrPiece traceback; - bool restored; - public: - // Takes new references; if we're destroyed before - // restoring the error, we drop the references. - PyErrPieces(PyObject* t, PyObject* v, PyObject* tb) : - type(t), - instance(v), - traceback(tb), - restored(0) - { - this->normalize(); - } - - PyErrPieces() : - restored(0) - { - // PyErr_Fetch transfers ownership to us, so - // we don't actually need to INCREF; but we *do* - // need to DECREF if we're not restored. - PyErrFetchParam t, v, tb; - PyErr_Fetch(&t, &v, &tb); - type.steal(t.relinquish_ownership()); - instance.steal(v.relinquish_ownership()); - traceback.steal(tb.relinquish_ownership()); - } - - void PyErrRestore() - { - // can only do this once - assert(!this->restored); - this->restored = true; - PyErr_Restore( - this->type.relinquish_ownership(), - this->instance.relinquish_ownership(), - this->traceback.relinquish_ownership()); - assert(!this->type && !this->instance && !this->traceback); - } - - private: - void normalize() - { - // First, check the traceback argument, replacing None, - // with NULL - if (traceback.is_None()) { - traceback = nullptr; - } - - if (traceback && !PyTraceBack_Check(traceback.borrow())) { - throw PyErrOccurred(PyExc_TypeError, - "throw() third argument must be a traceback object"); - } - - if (PyExceptionClass_Check(type)) { - // If we just had a type, we'll now have a type and - // instance. - // The type's refcount will have gone up by one - // because of the instance and the instance will have - // a refcount of one. Either way, we owned, and still - // do own, exactly one reference. - PyErr_NormalizeException(&type, &instance, &traceback); - - } - else if (PyExceptionInstance_Check(type)) { - /* Raising an instance --- usually that means an - object that is a subclass of BaseException, but on - Python 2, that can also mean an arbitrary old-style - object. The value should be a dummy. */ - if (instance && !instance.is_None()) { - throw PyErrOccurred( - PyExc_TypeError, - "instance exception may not have a separate value"); - } - /* Normalize to raise , */ - this->instance = this->type; - this->type = PyExceptionInstance_Class(instance.borrow()); - - /* - It would be tempting to do this: - - Py_ssize_t type_count = Py_REFCNT(Py_TYPE(instance.borrow())); - this->type = PyExceptionInstance_Class(instance.borrow()); - assert(this->type.REFCNT() == type_count + 1); - - But that doesn't work on Python 2 in the case of - old-style instances: The result of Py_TYPE is going to - be the global shared that all - old-style classes have, while the return of Instance_Class() - will be the Python-level class object. The two are unrelated. - */ - } - else { - /* Not something you can raise. throw() fails. */ - PyErr_Format(PyExc_TypeError, - "exceptions must be classes, or instances, not %s", - Py_TYPE(type.borrow())->tp_name); - throw PyErrOccurred(); - } - } - }; - - // PyArg_Parse's O argument returns a borrowed reference. - class PyArgParseParam : public BorrowedObject - { - private: - G_NO_COPIES_OF_CLS(PyArgParseParam); - public: - explicit PyArgParseParam(PyObject* p=nullptr) : BorrowedObject(p) - { - } - - inline PyObject** operator&() - { - return &this->p; - } - }; - -};}; - -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet_slp_switch.hpp b/.env/Lib/site-packages/greenlet/greenlet_slp_switch.hpp deleted file mode 100644 index bdffcca..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_slp_switch.hpp +++ /dev/null @@ -1,103 +0,0 @@ -#ifndef GREENLET_SLP_SWITCH_HPP -#define GREENLET_SLP_SWITCH_HPP - -#include "greenlet_compiler_compat.hpp" -#include "greenlet_refs.hpp" - -/* - * the following macros are spliced into the OS/compiler - * specific code, in order to simplify maintenance. - */ -// We can save about 10% of the time it takes to switch greenlets if -// we thread the thread state through the slp_save_state() and the -// following slp_restore_state() calls from -// slp_switch()->g_switchstack() (which already needs to access it). -// -// However: -// -// that requires changing the prototypes and implementations of the -// switching functions. If we just change the prototype of -// slp_switch() to accept the argument and update the macros, without -// changing the implementation of slp_switch(), we get crashes on -// 64-bit Linux and 32-bit x86 (for reasons that aren't 100% clear); -// on the other hand, 64-bit macOS seems to be fine. Also, 64-bit -// windows is an issue because slp_switch is written fully in assembly -// and currently ignores its argument so some code would have to be -// adjusted there to pass the argument on to the -// ``slp_save_state_asm()`` function (but interestingly, because of -// the calling convention, the extra argument is just ignored and -// things function fine, albeit slower, if we just modify -// ``slp_save_state_asm`()` to fetch the pointer to pass to the -// macro.) -// -// Our compromise is to use a *glabal*, untracked, weak, pointer -// to the necessary thread state during the process of switching only. -// This is safe because we're protected by the GIL, and if we're -// running this code, the thread isn't exiting. This also nets us a -// 10-12% speed improvement. - -#if Py_GIL_DISABLED -thread_local greenlet::Greenlet* switching_thread_state = nullptr; -#else -static greenlet::Greenlet* volatile switching_thread_state = nullptr; -#endif - - -extern "C" { -static int GREENLET_NOINLINE(slp_save_state_trampoline)(char* stackref); -static void GREENLET_NOINLINE(slp_restore_state_trampoline)(); -} - - -#define SLP_SAVE_STATE(stackref, stsizediff) \ -do { \ - assert(switching_thread_state); \ - stackref += STACK_MAGIC; \ - if (slp_save_state_trampoline((char*)stackref)) \ - return -1; \ - if (!switching_thread_state->active()) \ - return 1; \ - stsizediff = switching_thread_state->stack_start() - (char*)stackref; \ -} while (0) - -#define SLP_RESTORE_STATE() slp_restore_state_trampoline() - -#define SLP_EVAL -extern "C" { -#define slp_switch GREENLET_NOINLINE(slp_switch) -#include "slp_platformselect.h" -} -#undef slp_switch - -#ifndef STACK_MAGIC -# error \ - "greenlet needs to be ported to this platform, or taught how to detect your compiler properly." -#endif /* !STACK_MAGIC */ - - - -#ifdef EXTERNAL_ASM -/* CCP addition: Make these functions, to be called from assembler. - * The token include file for the given platform should enable the - * EXTERNAL_ASM define so that this is included. - */ -extern "C" { -intptr_t -slp_save_state_asm(intptr_t* ref) -{ - intptr_t diff; - SLP_SAVE_STATE(ref, diff); - return diff; -} - -void -slp_restore_state_asm(void) -{ - SLP_RESTORE_STATE(); -} - -extern int slp_switch(void); -}; -#endif - -#endif diff --git a/.env/Lib/site-packages/greenlet/greenlet_thread_support.hpp b/.env/Lib/site-packages/greenlet/greenlet_thread_support.hpp deleted file mode 100644 index 3ded7d2..0000000 --- a/.env/Lib/site-packages/greenlet/greenlet_thread_support.hpp +++ /dev/null @@ -1,31 +0,0 @@ -#ifndef GREENLET_THREAD_SUPPORT_HPP -#define GREENLET_THREAD_SUPPORT_HPP - -/** - * Defines various utility functions to help greenlet integrate well - * with threads. This used to be needed when we supported Python - * 2.7 on Windows, which used a very old compiler. We wrote an - * alternative implementation using Python APIs and POSIX or Windows - * APIs, but that's no longer needed. So this file is a shadow of its - * former self --- but may be needed in the future. - */ - -#include -#include -#include - -#include "greenlet_compiler_compat.hpp" - -namespace greenlet { - typedef std::mutex Mutex; - typedef std::lock_guard LockGuard; - class LockInitError : public std::runtime_error - { - public: - LockInitError(const char* what) : std::runtime_error(what) - {}; - }; -}; - - -#endif /* GREENLET_THREAD_SUPPORT_HPP */ diff --git a/.env/Lib/site-packages/greenlet/platform/__init__.py b/.env/Lib/site-packages/greenlet/platform/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/greenlet/platform/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/greenlet/platform/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index fe76821592d83b88f0d9f567847c7e0f8f5d300d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 190 zcmX@j%ge<81iKHkWrFC(AOanHW&w&!XQ*V*Wb|9fP{ah}eFmxd73*vj6Iz^FR2&nO zpOUZOo0y!DnwOcK8dFdic_`t=4F<|$LkeT{^GF7%}*)KNwq6t1=_|4#Kj=SM`lJw J#v*1Q3jif5G=Bg9 diff --git a/.env/Lib/site-packages/greenlet/platform/setup_switch_x64_masm.cmd b/.env/Lib/site-packages/greenlet/platform/setup_switch_x64_masm.cmd deleted file mode 100644 index 038ced2..0000000 --- a/.env/Lib/site-packages/greenlet/platform/setup_switch_x64_masm.cmd +++ /dev/null @@ -1,2 +0,0 @@ -call "C:\Program Files (x86)\Microsoft Visual Studio 9.0\VC\vcvarsall.bat" amd64 -ml64 /nologo /c /Fo switch_x64_masm.obj switch_x64_masm.asm diff --git a/.env/Lib/site-packages/greenlet/platform/switch_aarch64_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_aarch64_gcc.h deleted file mode 100644 index 058617c..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_aarch64_gcc.h +++ /dev/null @@ -1,124 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 07-Sep-16 Add clang support using x register naming. Fredrik Fornwall - * 13-Apr-13 Add support for strange GCC caller-save decisions - * 08-Apr-13 File creation. Michael Matz - * - * NOTES - * - * Simply save all callee saved registers - * - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 -#define REGS_TO_SAVE "x19", "x20", "x21", "x22", "x23", "x24", "x25", "x26", \ - "x27", "x28", "x30" /* aka lr */, \ - "v8", "v9", "v10", "v11", \ - "v12", "v13", "v14", "v15" - -/* - * Recall: - asm asm-qualifiers ( AssemblerTemplate - : OutputOperands - [ : InputOperands - [ : Clobbers ] ]) - - or (if asm-qualifiers contains 'goto') - - asm asm-qualifiers ( AssemblerTemplate - : OutputOperands - : InputOperands - : Clobbers - : GotoLabels) - - and OutputOperands are - - [ [asmSymbolicName] ] constraint (cvariablename) - - When a name is given, refer to it as ``%[the name]``. - When not given, ``%i`` where ``i`` is the zero-based index. - - constraints starting with ``=`` means only writing; ``+`` means - reading and writing. - - This is followed by ``r`` (must be register) or ``m`` (must be memory) - and these can be combined. - - The ``cvariablename`` is actually an lvalue expression. - - In AArch65, 31 general purpose registers. If named X0... they are - 64-bit. If named W0... they are the bottom 32 bits of the - corresponding 64 bit register. - - XZR and WZR are hardcoded to 0, and ignore writes. - - Arguments are in X0..X7. C++ uses X0 for ``this``. X0 holds simple return - values (?) - - Whenever a W register is written, the top half of the X register is zeroed. - */ - -static int -slp_switch(void) -{ - int err; - void *fp; - /* Windowz uses a 32-bit long on a 64-bit platform, unlike the rest of - the world, and in theory we can be compiled with GCC/llvm on 64-bit - windows. So we need a fixed-width type. - */ - int64_t *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("str x29, %0" : "=m"(fp) : : ); - __asm__ ("mov %0, sp" : "=r" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "add sp,sp,%0\n" - "add x29,x29,%0\n" - : - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - /* SLP_SAVE_STATE macro contains some return statements - (of -1 and 1). It falls through only when - the return value of slp_save_state() is zero, which - is placed in x0. - In that case we (slp_switch) also want to return zero - (also in x0 of course). - Now, some GCC versions (seen with 4.8) think it's a - good idea to save/restore x0 around the call to - slp_restore_state(), instead of simply zeroing it - at the return below. But slp_restore_state - writes random values to the stack slot used for this - save/restore (from when it once was saved above in - SLP_SAVE_STATE, when it was still uninitialized), so - "restoring" that precious zero actually makes us - return random values. There are some ways to make - GCC not use that zero value in the normal return path - (e.g. making err volatile, but that costs a little - stack space), and the simplest is to call a function - that returns an unknown value (which happens to be zero), - so the saved/restored value is unused. - - Thus, this line stores a 0 into the ``err`` variable - (which must be held in a register for this instruction, - of course). The ``w`` qualifier causes the instruction - to use W0 instead of X0, otherwise we get a warning - about a value size mismatch (because err is an int, - and aarch64 platforms are LP64: 32-bit int, 64 bit long - and pointer). - */ - __asm__ volatile ("mov %w0, #0" : "=r" (err)); - } - __asm__ volatile ("ldr x29, %0" : : "m" (fp) :); - __asm__ volatile ("" : : : REGS_TO_SAVE); - return err; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_alpha_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_alpha_unix.h deleted file mode 100644 index 7e07abf..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_alpha_unix.h +++ /dev/null @@ -1,30 +0,0 @@ -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "$9", "$10", "$11", "$12", "$13", "$14", "$15", \ - "$f2", "$f3", "$f4", "$f5", "$f6", "$f7", "$f8", "$f9" - -static int -slp_switch(void) -{ - int ret; - long *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("mov $30, %0" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "addq $30, %0, $30\n\t" - : /* no outputs */ - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("mov $31, %0" : "=r" (ret) : ); - return ret; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_amd64_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_amd64_unix.h deleted file mode 100644 index d470110..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_amd64_unix.h +++ /dev/null @@ -1,87 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 3-May-13 Ralf Schmitt - * Add support for strange GCC caller-save decisions - * (ported from switch_aarch64_gcc.h) - * 18-Aug-11 Alexey Borzenkov - * Correctly save rbp, csr and cw - * 01-Apr-04 Hye-Shik Chang - * Ported from i386 to amd64. - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for spark - * 31-Avr-02 Armin Rigo - * Added ebx, esi and edi register-saves. - * 01-Mar-02 Samual M. Rushing - * Ported from i386. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -/* #define STACK_MAGIC 3 */ -/* the above works fine with gcc 2.96, but 2.95.3 wants this */ -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "r12", "r13", "r14", "r15" - -static int -slp_switch(void) -{ - int err; - void* rbp; - void* rbx; - unsigned int csr; - unsigned short cw; - /* This used to be declared 'register', but that does nothing in - modern compilers and is explicitly forbidden in some new - standards. */ - long *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("fstcw %0" : "=m" (cw)); - __asm__ volatile ("stmxcsr %0" : "=m" (csr)); - __asm__ volatile ("movq %%rbp, %0" : "=m" (rbp)); - __asm__ volatile ("movq %%rbx, %0" : "=m" (rbx)); - __asm__ ("movq %%rsp, %0" : "=g" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "addq %0, %%rsp\n" - "addq %0, %%rbp\n" - : - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - __asm__ volatile ("xorq %%rax, %%rax" : "=a" (err)); - } - __asm__ volatile ("movq %0, %%rbx" : : "m" (rbx)); - __asm__ volatile ("movq %0, %%rbp" : : "m" (rbp)); - __asm__ volatile ("ldmxcsr %0" : : "m" (csr)); - __asm__ volatile ("fldcw %0" : : "m" (cw)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_arm32_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_arm32_gcc.h deleted file mode 100644 index 655003a..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_arm32_gcc.h +++ /dev/null @@ -1,79 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 14-Aug-06 File creation. Ported from Arm Thumb. Sylvain Baro - * 3-Sep-06 Commented out saving of r1-r3 (r4 already commented out) as I - * read that these do not need to be saved. Also added notes and - * errors related to the frame pointer. Richard Tew. - * - * NOTES - * - * It is not possible to detect if fp is used or not, so the supplied - * switch function needs to support it, so that you can remove it if - * it does not apply to you. - * - * POSSIBLE ERRORS - * - * "fp cannot be used in asm here" - * - * - Try commenting out "fp" in REGS_TO_SAVE. - * - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 -#define REG_SP "sp" -#define REG_SPSP "sp,sp" -#ifdef __thumb__ -#define REG_FP "r7" -#define REG_FPFP "r7,r7" -#define REGS_TO_SAVE_GENERAL "r4", "r5", "r6", "r8", "r9", "r10", "r11", "lr" -#else -#define REG_FP "fp" -#define REG_FPFP "fp,fp" -#define REGS_TO_SAVE_GENERAL "r4", "r5", "r6", "r7", "r8", "r9", "r10", "lr" -#endif -#if defined(__SOFTFP__) -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL -#elif defined(__VFP_FP__) -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL, "d8", "d9", "d10", "d11", \ - "d12", "d13", "d14", "d15" -#elif defined(__MAVERICK__) -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL, "mvf4", "mvf5", "mvf6", "mvf7", \ - "mvf8", "mvf9", "mvf10", "mvf11", \ - "mvf12", "mvf13", "mvf14", "mvf15" -#else -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL, "f4", "f5", "f6", "f7" -#endif - -static int -#ifdef __GNUC__ -__attribute__((optimize("no-omit-frame-pointer"))) -#endif -slp_switch(void) -{ - void *fp; - int *stackref, stsizediff; - int result; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("mov r0," REG_FP "\n\tstr r0,%0" : "=m" (fp) : : "r0"); - __asm__ ("mov %0," REG_SP : "=r" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "add " REG_SPSP ",%0\n" - "add " REG_FPFP ",%0\n" - : - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("ldr r0,%1\n\tmov " REG_FP ",r0\n\tmov %0, #0" : "=r" (result) : "m" (fp) : "r0"); - __asm__ volatile ("" : : : REGS_TO_SAVE); - return result; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_arm32_ios.h b/.env/Lib/site-packages/greenlet/platform/switch_arm32_ios.h deleted file mode 100644 index 9e640e1..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_arm32_ios.h +++ /dev/null @@ -1,67 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 31-May-15 iOS support. Ported from arm32. Proton - * - * NOTES - * - * It is not possible to detect if fp is used or not, so the supplied - * switch function needs to support it, so that you can remove it if - * it does not apply to you. - * - * POSSIBLE ERRORS - * - * "fp cannot be used in asm here" - * - * - Try commenting out "fp" in REGS_TO_SAVE. - * - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 0 -#define REG_SP "sp" -#define REG_SPSP "sp,sp" -#define REG_FP "r7" -#define REG_FPFP "r7,r7" -#define REGS_TO_SAVE_GENERAL "r4", "r5", "r6", "r8", "r10", "r11", "lr" -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL, "d8", "d9", "d10", "d11", \ - "d12", "d13", "d14", "d15" - -static int -#ifdef __GNUC__ -__attribute__((optimize("no-omit-frame-pointer"))) -#endif -slp_switch(void) -{ - void *fp; - int *stackref, stsizediff, result; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("str " REG_FP ",%0" : "=m" (fp)); - __asm__ ("mov %0," REG_SP : "=r" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "add " REG_SPSP ",%0\n" - "add " REG_FPFP ",%0\n" - : - : "r" (stsizediff) - : REGS_TO_SAVE /* Clobber registers, force compiler to - * recalculate address of void *fp from REG_SP or REG_FP */ - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ( - "ldr " REG_FP ", %1\n\t" - "mov %0, #0" - : "=r" (result) - : "m" (fp) - : REGS_TO_SAVE /* Force compiler to restore saved registers after this */ - ); - return result; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_arm64_masm.asm b/.env/Lib/site-packages/greenlet/platform/switch_arm64_masm.asm deleted file mode 100644 index 29f9c22..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_arm64_masm.asm +++ /dev/null @@ -1,53 +0,0 @@ - AREA switch_arm64_masm, CODE, READONLY; - GLOBAL slp_switch [FUNC] - EXTERN slp_save_state_asm - EXTERN slp_restore_state_asm - -slp_switch - ; push callee saved registers to stack - stp x19, x20, [sp, #-16]! - stp x21, x22, [sp, #-16]! - stp x23, x24, [sp, #-16]! - stp x25, x26, [sp, #-16]! - stp x27, x28, [sp, #-16]! - stp x29, x30, [sp, #-16]! - stp d8, d9, [sp, #-16]! - stp d10, d11, [sp, #-16]! - stp d12, d13, [sp, #-16]! - stp d14, d15, [sp, #-16]! - - ; call slp_save_state_asm with stack pointer - mov x0, sp - bl slp_save_state_asm - - ; early return for return value of 1 and -1 - cmp x0, #-1 - b.eq RETURN - cmp x0, #1 - b.eq RETURN - - ; increment stack and frame pointer - add sp, sp, x0 - add x29, x29, x0 - - bl slp_restore_state_asm - - ; store return value for successful completion of routine - mov x0, #0 - -RETURN - ; pop registers from stack - ldp d14, d15, [sp], #16 - ldp d12, d13, [sp], #16 - ldp d10, d11, [sp], #16 - ldp d8, d9, [sp], #16 - ldp x29, x30, [sp], #16 - ldp x27, x28, [sp], #16 - ldp x25, x26, [sp], #16 - ldp x23, x24, [sp], #16 - ldp x21, x22, [sp], #16 - ldp x19, x20, [sp], #16 - - ret - - END diff --git a/.env/Lib/site-packages/greenlet/platform/switch_arm64_masm.obj b/.env/Lib/site-packages/greenlet/platform/switch_arm64_masm.obj deleted file mode 100644 index f6f220e4310baaa9756110685ce7d6a2bdf90c37..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 746 zcma)4PiqrF6n~qoo~*PNZ{i+=wji4b#Xu2~wiJpGk)*AMF07NyB(9n1#+i+!)I;v| zBKQG3?t1eB$T(lYgGb4+lu{_QmQrebldQB_4?cMF-uu0IZ{DA2e8@rZ+e^~30488W z`B{KPh%yV{HEIpyeum^wI#7P*HfX)ux?9U&c!SFK-$o|OFtKn{Q|a-#N>2inp0-tb zCRKXAtg2SolaoLv$Ll&ds_Epj?SH+8K{t?XSjKaFsEy%yh}=V70BaHj zEY5kWk_zcJo{$SSUL~=K(zW|tnhm$rA z<%dZ$q?>RX*18r{!azhaYR1lVb;g;mR-6h!#F>|p@;aje%0a|CZrE7s+SXuT>MS=Y ziQPiMY-5DD&5+S7^H03fy7qt7ir}L34kK|h68s-MU>{lXOqlr?!Y=`~WwviNenFS_ zZalVSHh-0FU4lj#X8u5`ODn6@#{f?dHE&%XdP~_I3;$RS9-(z*>>ydkm*f@oWlUn~ Qn+^;lsEi}=H#!Q3U&UU-WdHyG diff --git a/.env/Lib/site-packages/greenlet/platform/switch_arm64_msvc.h b/.env/Lib/site-packages/greenlet/platform/switch_arm64_msvc.h deleted file mode 100644 index 7ab7f45..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_arm64_msvc.h +++ /dev/null @@ -1,17 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 21-Oct-21 Niyas Sait - * First version to enable win/arm64 support. - */ - -#define STACK_REFPLUS 1 -#define STACK_MAGIC 0 - -/* Use the generic support for an external assembly language slp_switch function. */ -#define EXTERNAL_ASM - -#ifdef SLP_EVAL -/* This always uses the external masm assembly file. */ -#endif \ No newline at end of file diff --git a/.env/Lib/site-packages/greenlet/platform/switch_csky_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_csky_gcc.h deleted file mode 100644 index ac469d3..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_csky_gcc.h +++ /dev/null @@ -1,48 +0,0 @@ -#ifdef SLP_EVAL -#define STACK_MAGIC 0 -#define REG_FP "r8" -#ifdef __CSKYABIV2__ -#define REGS_TO_SAVE_GENERAL "r4", "r5", "r6", "r7", "r9", "r10", "r11", "r15",\ - "r16", "r17", "r18", "r19", "r20", "r21", "r22",\ - "r23", "r24", "r25" - -#if defined (__CSKY_HARD_FLOAT__) || (__CSKY_VDSP__) -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL, "vr8", "vr9", "vr10", "vr11", "vr12",\ - "vr13", "vr14", "vr15" -#else -#define REGS_TO_SAVE REGS_TO_SAVE_GENERAL -#endif -#else -#define REGS_TO_SAVE "r9", "r10", "r11", "r12", "r13", "r15" -#endif - - -static int -#ifdef __GNUC__ -__attribute__((optimize("no-omit-frame-pointer"))) -#endif -slp_switch(void) -{ - int *stackref, stsizediff; - int result; - - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ ("mov %0, sp" : "=r" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "addu sp,%0\n" - "addu "REG_FP",%0\n" - : - : "r" (stsizediff) - ); - - SLP_RESTORE_STATE(); - } - __asm__ volatile ("movi %0, 0" : "=r" (result)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - - return result; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_loongarch64_linux.h b/.env/Lib/site-packages/greenlet/platform/switch_loongarch64_linux.h deleted file mode 100644 index 9eaf34e..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_loongarch64_linux.h +++ /dev/null @@ -1,31 +0,0 @@ -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "s0", "s1", "s2", "s3", "s4", "s5", \ - "s6", "s7", "s8", "fp", \ - "f24", "f25", "f26", "f27", "f28", "f29", "f30", "f31" - -static int -slp_switch(void) -{ - int ret; - long *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("move %0, $sp" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "add.d $sp, $sp, %0\n\t" - : /* no outputs */ - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("move %0, $zero" : "=r" (ret) : ); - return ret; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_m68k_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_m68k_gcc.h deleted file mode 100644 index da761c2..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_m68k_gcc.h +++ /dev/null @@ -1,38 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 2014-01-06 Andreas Schwab - * File created. - */ - -#ifdef SLP_EVAL - -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "%d2", "%d3", "%d4", "%d5", "%d6", "%d7", \ - "%a2", "%a3", "%a4" - -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - void *fp, *a5; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("move.l %%fp, %0" : "=m"(fp)); - __asm__ volatile ("move.l %%a5, %0" : "=m"(a5)); - __asm__ ("move.l %%sp, %0" : "=r"(stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ("add.l %0, %%sp; add.l %0, %%fp" : : "r"(stsizediff)); - SLP_RESTORE_STATE(); - __asm__ volatile ("clr.l %0" : "=g" (err)); - } - __asm__ volatile ("move.l %0, %%a5" : : "m"(a5)); - __asm__ volatile ("move.l %0, %%fp" : : "m"(fp)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - return err; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_mips_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_mips_unix.h deleted file mode 100644 index 50e70c4..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_mips_unix.h +++ /dev/null @@ -1,65 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 20-Sep-14 Matt Madison - * Re-code the saving of the gp register for MIPS64. - * 05-Jan-08 Thiemo Seufer - * Ported from ppc. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "$16", "$17", "$18", "$19", "$20", "$21", "$22", \ - "$23", "$30" -__attribute__((nomips16)) -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; -#ifdef __mips64 - uint64_t gpsave; -#endif - __asm__ __volatile__ ("" : : : REGS_TO_SAVE); -#ifdef __mips64 - __asm__ __volatile__ ("sd $28,%0" : "=m" (gpsave) : : ); -#endif - __asm__ ("move %0, $29" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ __volatile__ ( -#ifdef __mips64 - "daddu $29, $29, %0\n" -#else - "addu $29, $29, %0\n" -#endif - : /* no outputs */ - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } -#ifdef __mips64 - __asm__ __volatile__ ("ld $28,%0" : : "m" (gpsave) : ); -#endif - __asm__ __volatile__ ("" : : : REGS_TO_SAVE); - __asm__ __volatile__ ("move %0, $0" : "=r" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc64_aix.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc64_aix.h deleted file mode 100644 index e7e0b87..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc64_aix.h +++ /dev/null @@ -1,103 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 16-Oct-20 Jesse Gorzinski - * Copied from Linux PPC64 implementation - * 04-Sep-18 Alexey Borzenkov - * Workaround a gcc bug using manual save/restore of r30 - * 21-Mar-18 Tulio Magno Quites Machado Filho - * Added r30 to the list of saved registers in order to fully comply with - * both ppc64 ELFv1 ABI and the ppc64le ELFv2 ABI, that classify this - * register as a nonvolatile register used for local variables. - * 21-Mar-18 Laszlo Boszormenyi - * Save r2 (TOC pointer) manually. - * 10-Dec-13 Ulrich Weigand - * Support ELFv2 ABI. Save float/vector registers. - * 09-Mar-12 Michael Ellerman - * 64-bit implementation, copied from 32-bit. - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 04-Oct-02 Gustavo Niemeyer - * Ported from MacOS version. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - * 31-Jul-12 Trevor Bowen - * Changed memory constraints to register only. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 6 - -#if defined(__ALTIVEC__) -#define ALTIVEC_REGS \ - "v20", "v21", "v22", "v23", "v24", "v25", "v26", "v27", \ - "v28", "v29", "v30", "v31", -#else -#define ALTIVEC_REGS -#endif - -#define REGS_TO_SAVE "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "r31", \ - "fr14", "fr15", "fr16", "fr17", "fr18", "fr19", "fr20", "fr21", \ - "fr22", "fr23", "fr24", "fr25", "fr26", "fr27", "fr28", "fr29", \ - "fr30", "fr31", \ - ALTIVEC_REGS \ - "cr2", "cr3", "cr4" - -static int -slp_switch(void) -{ - int err; - long *stackref, stsizediff; - void * toc; - void * r30; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("std 2, %0" : "=m" (toc)); - __asm__ volatile ("std 30, %0" : "=m" (r30)); - __asm__ ("mr %0, 1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "mr 11, %0\n" - "add 1, 1, 11\n" - : /* no outputs */ - : "r" (stsizediff) - : "11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("ld 30, %0" : : "m" (r30)); - __asm__ volatile ("ld 2, %0" : : "m" (toc)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc64_linux.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc64_linux.h deleted file mode 100644 index 3c324d0..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc64_linux.h +++ /dev/null @@ -1,105 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 04-Sep-18 Alexey Borzenkov - * Workaround a gcc bug using manual save/restore of r30 - * 21-Mar-18 Tulio Magno Quites Machado Filho - * Added r30 to the list of saved registers in order to fully comply with - * both ppc64 ELFv1 ABI and the ppc64le ELFv2 ABI, that classify this - * register as a nonvolatile register used for local variables. - * 21-Mar-18 Laszlo Boszormenyi - * Save r2 (TOC pointer) manually. - * 10-Dec-13 Ulrich Weigand - * Support ELFv2 ABI. Save float/vector registers. - * 09-Mar-12 Michael Ellerman - * 64-bit implementation, copied from 32-bit. - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 04-Oct-02 Gustavo Niemeyer - * Ported from MacOS version. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - * 31-Jul-12 Trevor Bowen - * Changed memory constraints to register only. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#if _CALL_ELF == 2 -#define STACK_MAGIC 4 -#else -#define STACK_MAGIC 6 -#endif - -#if defined(__ALTIVEC__) -#define ALTIVEC_REGS \ - "v20", "v21", "v22", "v23", "v24", "v25", "v26", "v27", \ - "v28", "v29", "v30", "v31", -#else -#define ALTIVEC_REGS -#endif - -#define REGS_TO_SAVE "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "r31", \ - "fr14", "fr15", "fr16", "fr17", "fr18", "fr19", "fr20", "fr21", \ - "fr22", "fr23", "fr24", "fr25", "fr26", "fr27", "fr28", "fr29", \ - "fr30", "fr31", \ - ALTIVEC_REGS \ - "cr2", "cr3", "cr4" - -static int -slp_switch(void) -{ - int err; - long *stackref, stsizediff; - void * toc; - void * r30; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("std 2, %0" : "=m" (toc)); - __asm__ volatile ("std 30, %0" : "=m" (r30)); - __asm__ ("mr %0, 1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "mr 11, %0\n" - "add 1, 1, 11\n" - : /* no outputs */ - : "r" (stsizediff) - : "11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("ld 30, %0" : : "m" (r30)); - __asm__ volatile ("ld 2, %0" : : "m" (toc)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc_aix.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc_aix.h deleted file mode 100644 index 6d93c13..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc_aix.h +++ /dev/null @@ -1,87 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 07-Mar-11 Floris Bruynooghe - * Do not add stsizediff to general purpose - * register (GPR) 30 as this is a non-volatile and - * unused by the PowerOpen Environment, therefore - * this was modifying a user register instead of the - * frame pointer (which does not seem to exist). - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 04-Oct-02 Gustavo Niemeyer - * Ported from MacOS version. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 3 - -/* !!!!WARNING!!!! need to add "r31" in the next line if this header file - * is meant to be compiled non-dynamically! - */ -#define REGS_TO_SAVE "r13", "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "cr2", "cr3", "cr4" -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ ("mr %0, 1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "mr 11, %0\n" - "add 1, 1, 11\n" - : /* no outputs */ - : "r" (stsizediff) - : "11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc_linux.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc_linux.h deleted file mode 100644 index e83ad70..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc_linux.h +++ /dev/null @@ -1,84 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 04-Oct-02 Gustavo Niemeyer - * Ported from MacOS version. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - * 31-Jul-12 Trevor Bowen - * Changed memory constraints to register only. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 3 - -/* !!!!WARNING!!!! need to add "r31" in the next line if this header file - * is meant to be compiled non-dynamically! - */ -#define REGS_TO_SAVE "r13", "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "cr2", "cr3", "cr4" -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ ("mr %0, 1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "mr 11, %0\n" - "add 1, 1, 11\n" - "add 30, 30, 11\n" - : /* no outputs */ - : "r" (stsizediff) - : "11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc_macosx.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc_macosx.h deleted file mode 100644 index bd414c6..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc_macosx.h +++ /dev/null @@ -1,82 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 3 - -/* !!!!WARNING!!!! need to add "r31" in the next line if this header file - * is meant to be compiled non-dynamically! - */ -#define REGS_TO_SAVE "r13", "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "cr2", "cr3", "cr4" - -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ ("; asm block 2\n\tmr %0, r1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "; asm block 3\n" - "\tmr r11, %0\n" - "\tadd r1, r1, r11\n" - "\tadd r30, r30, r11\n" - : /* no outputs */ - : "r" (stsizediff) - : "r11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_ppc_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_ppc_unix.h deleted file mode 100644 index bb18808..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_ppc_unix.h +++ /dev/null @@ -1,82 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'r31' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 14-Jan-04 Bob Ippolito - * added cr2-cr4 to the registers to be saved. - * Open questions: Should we save FP registers? - * What about vector registers? - * Differences between darwin and unix? - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 04-Oct-02 Gustavo Niemeyer - * Ported from MacOS version. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 29-Jun-02 Christian Tismer - * Added register 13-29, 31 saves. The same way as - * Armin Rigo did for the x86_unix version. - * This seems to be now fully functional! - * 04-Mar-02 Hye-Shik Chang - * Ported from i386. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 3 - -/* !!!!WARNING!!!! need to add "r31" in the next line if this header file - * is meant to be compiled non-dynamically! - */ -#define REGS_TO_SAVE "r13", "r14", "r15", "r16", "r17", "r18", "r19", "r20", \ - "r21", "r22", "r23", "r24", "r25", "r26", "r27", "r28", "r29", \ - "cr2", "cr3", "cr4" -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ ("mr %0, 1" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "mr 11, %0\n" - "add 1, 1, 11\n" - "add 30, 30, 11\n" - : /* no outputs */ - : "r" (stsizediff) - : "11" - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("li %0, 0" : "=r" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_riscv_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_riscv_unix.h deleted file mode 100644 index 8761122..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_riscv_unix.h +++ /dev/null @@ -1,41 +0,0 @@ -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "s1", "s2", "s3", "s4", "s5", \ - "s6", "s7", "s8", "s9", "s10", "s11", "fs0", "fs1", \ - "fs2", "fs3", "fs4", "fs5", "fs6", "fs7", "fs8", "fs9", \ - "fs10", "fs11" - -static int -slp_switch(void) -{ - int ret; - long fp; - long *stackref, stsizediff; - - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("mv %0, fp" : "=r" (fp) : ); - __asm__ volatile ("mv %0, sp" : "=r" (stackref) : ); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "add sp, sp, %0\n\t" - "add fp, fp, %0\n\t" - : /* no outputs */ - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); -#if __riscv_xlen == 32 - __asm__ volatile ("lw fp, %0" : : "m" (fp)); -#else - __asm__ volatile ("ld fp, %0" : : "m" (fp)); -#endif - __asm__ volatile ("mv %0, zero" : "=r" (ret) : ); - return ret; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_s390_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_s390_unix.h deleted file mode 100644 index 9199367..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_s390_unix.h +++ /dev/null @@ -1,87 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 25-Jan-12 Alexey Borzenkov - * Fixed Linux/S390 port to work correctly with - * different optimization options both on 31-bit - * and 64-bit. Thanks to Stefan Raabe for lots - * of testing. - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 06-Oct-02 Gustavo Niemeyer - * Ported to Linux/S390. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#ifdef __s390x__ -#define STACK_MAGIC 20 /* 20 * 8 = 160 bytes of function call area */ -#else -#define STACK_MAGIC 24 /* 24 * 4 = 96 bytes of function call area */ -#endif - -/* Technically, r11-r13 also need saving, but function prolog starts - with stm(g) and since there are so many saved registers already - it won't be optimized, resulting in all r6-r15 being saved */ -#define REGS_TO_SAVE "r6", "r7", "r8", "r9", "r10", "r14", \ - "f0", "f1", "f2", "f3", "f4", "f5", "f6", "f7", \ - "f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15" - -static int -slp_switch(void) -{ - int ret; - long *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); -#ifdef __s390x__ - __asm__ volatile ("lgr %0, 15" : "=r" (stackref) : ); -#else - __asm__ volatile ("lr %0, 15" : "=r" (stackref) : ); -#endif - { - SLP_SAVE_STATE(stackref, stsizediff); -/* N.B. - r11 may be used as the frame pointer, and in that case it cannot be - clobbered and needs offsetting just like the stack pointer (but in cases - where frame pointer isn't used we might clobber it accidentally). What's - scary is that r11 is 2nd (and even 1st when GOT is used) callee saved - register that gcc would chose for surviving function calls. However, - since r6-r10 are clobbered above, their cost for reuse is reduced, so - gcc IRA will chose them over r11 (not seeing r11 is implicitly saved), - making it relatively safe to offset in all cases. :) */ - __asm__ volatile ( -#ifdef __s390x__ - "agr 15, %0\n\t" - "agr 11, %0" -#else - "ar 15, %0\n\t" - "ar 11, %0" -#endif - : /* no outputs */ - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("lhi %0, 0" : "=r" (ret) : ); - return ret; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_sh_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_sh_gcc.h deleted file mode 100644 index 5ecc3b3..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_sh_gcc.h +++ /dev/null @@ -1,36 +0,0 @@ -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL -#define STACK_MAGIC 0 -#define REGS_TO_SAVE "r8", "r9", "r10", "r11", "r13", \ - "fr12", "fr13", "fr14", "fr15" - -// r12 Global context pointer, GP -// r14 Frame pointer, FP -// r15 Stack pointer, SP - -static int -slp_switch(void) -{ - int err; - void* fp; - int *stackref, stsizediff; - __asm__ volatile("" : : : REGS_TO_SAVE); - __asm__ volatile("mov.l r14, %0" : "=m"(fp) : :); - __asm__("mov r15, %0" : "=r"(stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile( - "add %0, r15\n" - "add %0, r14\n" - : /* no outputs */ - : "r"(stsizediff)); - SLP_RESTORE_STATE(); - __asm__ volatile("mov r0, %0" : "=r"(err) : :); - } - __asm__ volatile("mov.l %0, r14" : : "m"(fp) :); - __asm__ volatile("" : : : REGS_TO_SAVE); - return err; -} - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_sparc_sun_gcc.h b/.env/Lib/site-packages/greenlet/platform/switch_sparc_sun_gcc.h deleted file mode 100644 index 96990c3..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_sparc_sun_gcc.h +++ /dev/null @@ -1,92 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 16-May-15 Alexey Borzenkov - * Move stack spilling code inside save/restore functions - * 30-Aug-13 Floris Bruynooghe - Clean the register windows again before returning. - This does not clobber the PIC register as it leaves - the current window intact and is required for multi- - threaded code to work correctly. - * 08-Mar-11 Floris Bruynooghe - * No need to set return value register explicitly - * before the stack and framepointer are adjusted - * as none of the other registers are influenced by - * this. Also don't needlessly clean the windows - * ('ta %0" :: "i" (ST_CLEAN_WINDOWS)') as that - * clobbers the gcc PIC register (%l7). - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * added support for SunOS sparc with gcc - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - - -#define STACK_MAGIC 0 - - -#if defined(__sparcv9) -#define SLP_FLUSHW __asm__ volatile ("flushw") -#else -#define SLP_FLUSHW __asm__ volatile ("ta 3") /* ST_FLUSH_WINDOWS */ -#endif - -/* On sparc we need to spill register windows inside save/restore functions */ -#define SLP_BEFORE_SAVE_STATE() SLP_FLUSHW -#define SLP_BEFORE_RESTORE_STATE() SLP_FLUSHW - - -static int -slp_switch(void) -{ - int err; - int *stackref, stsizediff; - - /* Put current stack pointer into stackref. - * Register spilling is done in save/restore. - */ - __asm__ volatile ("mov %%sp, %0" : "=r" (stackref)); - - { - /* Thou shalt put SLP_SAVE_STATE into a local block */ - /* Copy the current stack onto the heap */ - SLP_SAVE_STATE(stackref, stsizediff); - - /* Increment stack and frame pointer by stsizediff */ - __asm__ volatile ( - "add %0, %%sp, %%sp\n\t" - "add %0, %%fp, %%fp" - : : "r" (stsizediff)); - - /* Copy new stack from it's save store on the heap */ - SLP_RESTORE_STATE(); - - __asm__ volatile ("mov %1, %0" : "=r" (err) : "i" (0)); - return err; - } -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x32_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_x32_unix.h deleted file mode 100644 index 893369c..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_x32_unix.h +++ /dev/null @@ -1,63 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 17-Aug-12 Fantix King - * Ported from amd64. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 0 - -#define REGS_TO_SAVE "r12", "r13", "r14", "r15" - - -static int -slp_switch(void) -{ - void* ebp; - void* ebx; - unsigned int csr; - unsigned short cw; - int err; - int *stackref, stsizediff; - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("fstcw %0" : "=m" (cw)); - __asm__ volatile ("stmxcsr %0" : "=m" (csr)); - __asm__ volatile ("movl %%ebp, %0" : "=m" (ebp)); - __asm__ volatile ("movl %%ebx, %0" : "=m" (ebx)); - __asm__ ("movl %%esp, %0" : "=g" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "addl %0, %%esp\n" - "addl %0, %%ebp\n" - : - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - } - __asm__ volatile ("movl %0, %%ebx" : : "m" (ebx)); - __asm__ volatile ("movl %0, %%ebp" : : "m" (ebp)); - __asm__ volatile ("ldmxcsr %0" : : "m" (csr)); - __asm__ volatile ("fldcw %0" : : "m" (cw)); - __asm__ volatile ("" : : : REGS_TO_SAVE); - __asm__ volatile ("xorl %%eax, %%eax" : "=a" (err)); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x64_masm.asm b/.env/Lib/site-packages/greenlet/platform/switch_x64_masm.asm deleted file mode 100644 index f5c72a2..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_x64_masm.asm +++ /dev/null @@ -1,111 +0,0 @@ -; -; stack switching code for MASM on x641 -; Kristjan Valur Jonsson, sept 2005 -; - - -;prototypes for our calls -slp_save_state_asm PROTO -slp_restore_state_asm PROTO - - -pushxmm MACRO reg - sub rsp, 16 - .allocstack 16 - movaps [rsp], reg ; faster than movups, but we must be aligned - ; .savexmm128 reg, offset (don't know what offset is, no documentation) -ENDM -popxmm MACRO reg - movaps reg, [rsp] ; faster than movups, but we must be aligned - add rsp, 16 -ENDM - -pushreg MACRO reg - push reg - .pushreg reg -ENDM -popreg MACRO reg - pop reg -ENDM - - -.code -slp_switch PROC FRAME - ;realign stack to 16 bytes after return address push, makes the following faster - sub rsp,8 - .allocstack 8 - - pushxmm xmm15 - pushxmm xmm14 - pushxmm xmm13 - pushxmm xmm12 - pushxmm xmm11 - pushxmm xmm10 - pushxmm xmm9 - pushxmm xmm8 - pushxmm xmm7 - pushxmm xmm6 - - pushreg r15 - pushreg r14 - pushreg r13 - pushreg r12 - - pushreg rbp - pushreg rbx - pushreg rdi - pushreg rsi - - sub rsp, 10h ;allocate the singlefunction argument (must be multiple of 16) - .allocstack 10h -.endprolog - - lea rcx, [rsp+10h] ;load stack base that we are saving - call slp_save_state_asm ;pass stackpointer, return offset in eax - cmp rax, 1 - je EXIT1 - cmp rax, -1 - je EXIT2 - ;actual stack switch: - add rsp, rax - call slp_restore_state_asm - xor rax, rax ;return 0 - -EXIT: - - add rsp, 10h - popreg rsi - popreg rdi - popreg rbx - popreg rbp - - popreg r12 - popreg r13 - popreg r14 - popreg r15 - - popxmm xmm6 - popxmm xmm7 - popxmm xmm8 - popxmm xmm9 - popxmm xmm10 - popxmm xmm11 - popxmm xmm12 - popxmm xmm13 - popxmm xmm14 - popxmm xmm15 - - add rsp, 8 - ret - -EXIT1: - mov rax, 1 - jmp EXIT - -EXIT2: - sar rax, 1 - jmp EXIT - -slp_switch ENDP - -END \ No newline at end of file diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x64_masm.obj b/.env/Lib/site-packages/greenlet/platform/switch_x64_masm.obj deleted file mode 100644 index 64e3e6b898ec765d4e37075f7b1635ad24c9efa2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1078 zcmZ{j&ubG=5XWb`DJB@*%~BA=L%=;Gk}d_~52VO$4J4q2U~MY6&1RFl{E&?scGnt@ zn(9GNy!ihFEO@PV4?T&H9`x2*oO!!jlNJZwd!P4xlX&_;U$Bg3z>p zje>}2kp+MsxE|w5hOUr>YC~(=fz6fwPdZd5+Hlb^P3{;ZO@Yuv96GG&+Gx?QfclNd zhy2KN&~>fNnlHQRR;U1cMyQ?hlQ$~k<0KBbB<0uD2#PTjVo+na7Q;#m=@=3m;xJOa zs2V#)&Db`cY;WzTF9)11;SjkVQWE!?bPTC%x3h3^F2;aBns5!i%m4&-*h69;~AUpZR%rDpm!zuXY+kc zFCz-n*^4&c)5~}y3e?r-Evy*n*(lp9r%ti58Y#l5&)rDjx5EbRd}nC+_8znRzz&#& XZ_Fi+`GM=5Rl{n4%KxAK>jC@)Nz=zi diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x64_msvc.h b/.env/Lib/site-packages/greenlet/platform/switch_x64_msvc.h deleted file mode 100644 index 601ea56..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_x64_msvc.h +++ /dev/null @@ -1,60 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 26-Sep-02 Christian Tismer - * again as a result of virtualized stack access, - * the compiler used less registers. Needed to - * explicit mention registers in order to get them saved. - * Thanks to Jeff Senn for pointing this out and help. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 01-Mar-02 Christian Tismer - * Initial final version after lots of iterations for i386. - */ - -/* Avoid alloca redefined warning on mingw64 */ -#ifndef alloca -#define alloca _alloca -#endif - -#define STACK_REFPLUS 1 -#define STACK_MAGIC 0 - -/* Use the generic support for an external assembly language slp_switch function. */ -#define EXTERNAL_ASM - -#ifdef SLP_EVAL -/* This always uses the external masm assembly file. */ -#endif - -/* - * further self-processing support - */ - -/* we have IsBadReadPtr available, so we can peek at objects */ -/* -#define STACKLESS_SPY - -#ifdef IMPLEMENT_STACKLESSMODULE -#include "Windows.h" -#define CANNOT_READ_MEM(p, bytes) IsBadReadPtr(p, bytes) - -static int IS_ON_STACK(void*p) -{ - int stackref; - intptr_t stackbase = ((intptr_t)&stackref) & 0xfffff000; - return (intptr_t)p >= stackbase && (intptr_t)p < stackbase + 0x00100000; -} - -#endif -*/ \ No newline at end of file diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x86_msvc.h b/.env/Lib/site-packages/greenlet/platform/switch_x86_msvc.h deleted file mode 100644 index 0f3a59f..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_x86_msvc.h +++ /dev/null @@ -1,326 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 26-Sep-02 Christian Tismer - * again as a result of virtualized stack access, - * the compiler used less registers. Needed to - * explicit mention registers in order to get them saved. - * Thanks to Jeff Senn for pointing this out and help. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for sparc - * 01-Mar-02 Christian Tismer - * Initial final version after lots of iterations for i386. - */ - -#define alloca _alloca - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -#define STACK_MAGIC 0 - -/* Some magic to quell warnings and keep slp_switch() from crashing when built - with VC90. Disable global optimizations, and the warning: frame pointer - register 'ebp' modified by inline assembly code. - - We used to just disable global optimizations ("g") but upstream stackless - Python, as well as stackman, turn off all optimizations. - -References: -https://github.com/stackless-dev/stackman/blob/dbc72fe5207a2055e658c819fdeab9731dee78b9/stackman/platforms/switch_x86_msvc.h -https://github.com/stackless-dev/stackless/blob/main-slp/Stackless/platf/switch_x86_msvc.h -*/ -#define WIN32_LEAN_AND_MEAN -#include - -#pragma optimize("", off) /* so that autos are stored on the stack */ -#pragma warning(disable:4731) -#pragma warning(disable:4733) /* disable warning about modifying FS[0] */ - -/** - * Most modern compilers and environments handle C++ exceptions without any - * special help from us. MSVC on 32-bit windows is an exception. There, C++ - * exceptions are dealt with using Windows' Structured Exception Handling - * (SEH). - * - * SEH is implemented as a singly linked list of nodes. The - * head of this list is stored in the Thread Information Block, which itself - * is pointed to from the FS register. It's the first field in the structure, - * or offset 0, so we can access it using assembly FS:[0], or the compiler - * intrinsics and field offset information from the headers (as we do below). - * Somewhat unusually, the tail of the list doesn't have prev == NULL, it has - * prev == 0xFFFFFFFF. - * - * SEH was designed for C, and traditionally uses the MSVC compiler - * intrinsincs __try{}/__except{}. It is also utilized for C++ exceptions by - * MSVC; there, every throw of a C++ exception raises a SEH error with the - * ExceptionCode 0xE06D7363; the SEH handler list is then traversed to - * deal with the exception. - * - * If the SEH list is corrupt, then when a C++ exception is thrown the program - * will abruptly exit with exit code 1. This does not use std::terminate(), so - * std::set_terminate() is useless to debug this. - * - * The SEH list is closely tied to the call stack; entering a function that - * uses __try{} or most C++ functions will push a new handler onto the front - * of the list. Returning from the function will remove the handler. Saving - * and restoring the head node of the SEH list (FS:[0]) per-greenlet is NOT - * ENOUGH to make SEH or exceptions work. - * - * Stack switching breaks SEH because the call stack no longer necessarily - * matches the SEH list. For example, given greenlet A that switches to - * greenlet B, at the moment of entering greenlet B, we will have any SEH - * handlers from greenlet A on the SEH list; greenlet B can then add its own - * handlers to the SEH list. When greenlet B switches back to greenlet A, - * greenlet B's handlers would still be on the SEH stack, but when switch() - * returns control to greenlet A, we have replaced the contents of the stack - * in memory, so all the address that greenlet B added to the SEH list are now - * invalid: part of the call stack has been unwound, but the SEH list was out - * of sync with the call stack. The net effect is that exception handling - * stops working. - * - * Thus, when switching greenlets, we need to be sure that the SEH list - * matches the effective call stack, "cutting out" any handlers that were - * pushed by the greenlet that switched out and which are no longer valid. - * - * The easiest way to do this is to capture the SEH list at the time the main - * greenlet for a thread is created, and, when initially starting a greenlet, - * start a new SEH list for it, which contains nothing but the handler - * established for the new greenlet itself, with the tail being the handlers - * for the main greenlet. If we then save and restore the SEH per-greenlet, - * they won't interfere with each others SEH lists. (No greenlet can unwind - * the call stack past the handlers established by the main greenlet). - * - * By observation, a new thread starts with three SEH handlers on the list. By - * the time we get around to creating the main greenlet, though, there can be - * many more, established by transient calls that lead to the creation of the - * main greenlet. Therefore, 3 is a magic constant telling us when to perform - * the initial slice. - * - * All of this can be debugged using a vectored exception handler, which - * operates independently of the SEH handler list, and is called first. - * Walking the SEH list at key points can also be helpful. - * - * References: - * https://en.wikipedia.org/wiki/Win32_Thread_Information_Block - * https://devblogs.microsoft.com/oldnewthing/20100730-00/?p=13273 - * https://docs.microsoft.com/en-us/cpp/cpp/try-except-statement?view=msvc-160 - * https://docs.microsoft.com/en-us/cpp/cpp/structured-exception-handling-c-cpp?view=msvc-160 - * https://docs.microsoft.com/en-us/windows/win32/debug/structured-exception-handling - * https://docs.microsoft.com/en-us/windows/win32/debug/using-a-vectored-exception-handler - * https://bytepointer.com/resources/pietrek_crash_course_depths_of_win32_seh.htm - */ -#define GREENLET_NEEDS_EXCEPTION_STATE_SAVED - - -typedef struct _GExceptionRegistration { - struct _GExceptionRegistration* prev; - void* handler_f; -} GExceptionRegistration; - -static void -slp_set_exception_state(const void *const seh_state) -{ - // Because the stack from from which we do this is ALSO a handler, and - // that one we want to keep, we need to relink the current SEH handler - // frame to point to this one, cutting out the middle men, as it were. - // - // Entering a try block doesn't change the SEH frame, but entering a - // function containing a try block does. - GExceptionRegistration* current_seh_state = (GExceptionRegistration*)__readfsdword(FIELD_OFFSET(NT_TIB, ExceptionList)); - current_seh_state->prev = (GExceptionRegistration*)seh_state; -} - - -static GExceptionRegistration* -x86_slp_get_third_oldest_handler() -{ - GExceptionRegistration* a = NULL; /* Closest to the top */ - GExceptionRegistration* b = NULL; /* second */ - GExceptionRegistration* c = NULL; - GExceptionRegistration* seh_state = (GExceptionRegistration*)__readfsdword(FIELD_OFFSET(NT_TIB, ExceptionList)); - a = b = c = seh_state; - - while (seh_state && seh_state != (GExceptionRegistration*)0xFFFFFFFF) { - if ((void*)seh_state->prev < (void*)100) { - fprintf(stderr, "\tERROR: Broken SEH chain.\n"); - return NULL; - } - a = b; - b = c; - c = seh_state; - - seh_state = seh_state->prev; - } - return a ? a : (b ? b : c); -} - - -static void* -slp_get_exception_state() -{ - // XXX: There appear to be three SEH handlers on the stack already at the - // start of the thread. Is that a guarantee? Almost certainly not. Yet in - // all observed cases it has been three. This is consistent with - // faulthandler off or on, and optimizations off or on. It may not be - // consistent with other operating system versions, though: we only have - // CI on one or two versions (don't ask what there are). - // In theory we could capture the number of handlers on the chain when - // PyInit__greenlet is called: there are probably only the default - // handlers at that point (unless we're embedded and people have used - // __try/__except or a C++ handler)? - return x86_slp_get_third_oldest_handler(); -} - -static int -slp_switch(void) -{ - /* MASM syntax is typically reversed from other assemblers. - It is usually - */ - int *stackref, stsizediff; - /* store the structured exception state for this stack */ - DWORD seh_state = __readfsdword(FIELD_OFFSET(NT_TIB, ExceptionList)); - __asm mov stackref, esp; - /* modify EBX, ESI and EDI in order to get them preserved */ - __asm mov ebx, ebx; - __asm xchg esi, edi; - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm { - mov eax, stsizediff - add esp, eax - add ebp, eax - } - SLP_RESTORE_STATE(); - } - __writefsdword(FIELD_OFFSET(NT_TIB, ExceptionList), seh_state); - return 0; -} - -/* re-enable ebp warning and global optimizations. */ -#pragma optimize("", on) -#pragma warning(default:4731) -#pragma warning(default:4733) /* disable warning about modifying FS[0] */ - - -#endif - -/* - * further self-processing support - */ - -/* we have IsBadReadPtr available, so we can peek at objects */ -#define STACKLESS_SPY - -#ifdef GREENLET_DEBUG - -#define CANNOT_READ_MEM(p, bytes) IsBadReadPtr(p, bytes) - -static int IS_ON_STACK(void*p) -{ - int stackref; - int stackbase = ((int)&stackref) & 0xfffff000; - return (int)p >= stackbase && (int)p < stackbase + 0x00100000; -} - -static void -x86_slp_show_seh_chain() -{ - GExceptionRegistration* seh_state = (GExceptionRegistration*)__readfsdword(FIELD_OFFSET(NT_TIB, ExceptionList)); - fprintf(stderr, "====== SEH Chain ======\n"); - while (seh_state && seh_state != (GExceptionRegistration*)0xFFFFFFFF) { - fprintf(stderr, "\tSEH_chain addr: %p handler: %p prev: %p\n", - seh_state, - seh_state->handler_f, seh_state->prev); - if ((void*)seh_state->prev < (void*)100) { - fprintf(stderr, "\tERROR: Broken chain.\n"); - break; - } - seh_state = seh_state->prev; - } - fprintf(stderr, "====== End SEH Chain ======\n"); - fflush(NULL); - return; -} - -//addVectoredExceptionHandler constants: -//CALL_FIRST means call this exception handler first; -//CALL_LAST means call this exception handler last -#define CALL_FIRST 1 -#define CALL_LAST 0 - -LONG WINAPI -GreenletVectorHandler(PEXCEPTION_POINTERS ExceptionInfo) -{ - // We get one of these for every C++ exception, with code - // E06D7363 - // This is a special value that means "C++ exception from MSVC" - // https://devblogs.microsoft.com/oldnewthing/20100730-00/?p=13273 - // - // Install in the module init function with: - // AddVectoredExceptionHandler(CALL_FIRST, GreenletVectorHandler); - PEXCEPTION_RECORD ExceptionRecord = ExceptionInfo->ExceptionRecord; - - fprintf(stderr, - "GOT VECTORED EXCEPTION:\n" - "\tExceptionCode : %p\n" - "\tExceptionFlags : %p\n" - "\tExceptionAddr : %p\n" - "\tNumberparams : %ld\n", - ExceptionRecord->ExceptionCode, - ExceptionRecord->ExceptionFlags, - ExceptionRecord->ExceptionAddress, - ExceptionRecord->NumberParameters - ); - if (ExceptionRecord->ExceptionFlags & 1) { - fprintf(stderr, "\t\tEH_NONCONTINUABLE\n" ); - } - if (ExceptionRecord->ExceptionFlags & 2) { - fprintf(stderr, "\t\tEH_UNWINDING\n" ); - } - if (ExceptionRecord->ExceptionFlags & 4) { - fprintf(stderr, "\t\tEH_EXIT_UNWIND\n" ); - } - if (ExceptionRecord->ExceptionFlags & 8) { - fprintf(stderr, "\t\tEH_STACK_INVALID\n" ); - } - if (ExceptionRecord->ExceptionFlags & 0x10) { - fprintf(stderr, "\t\tEH_NESTED_CALL\n" ); - } - if (ExceptionRecord->ExceptionFlags & 0x20) { - fprintf(stderr, "\t\tEH_TARGET_UNWIND\n" ); - } - if (ExceptionRecord->ExceptionFlags & 0x40) { - fprintf(stderr, "\t\tEH_COLLIDED_UNWIND\n" ); - } - fprintf(stderr, "\n"); - fflush(NULL); - for(DWORD i = 0; i < ExceptionRecord->NumberParameters; i++) { - fprintf(stderr, "\t\t\tParam %ld: %lX\n", i, ExceptionRecord->ExceptionInformation[i]); - } - - if (ExceptionRecord->NumberParameters == 3) { - fprintf(stderr, "\tAbout to traverse SEH chain\n"); - // C++ Exception records have 3 params. - x86_slp_show_seh_chain(); - } - - return EXCEPTION_CONTINUE_SEARCH; -} - - - - -#endif diff --git a/.env/Lib/site-packages/greenlet/platform/switch_x86_unix.h b/.env/Lib/site-packages/greenlet/platform/switch_x86_unix.h deleted file mode 100644 index 493fa6b..0000000 --- a/.env/Lib/site-packages/greenlet/platform/switch_x86_unix.h +++ /dev/null @@ -1,105 +0,0 @@ -/* - * this is the internal transfer function. - * - * HISTORY - * 3-May-13 Ralf Schmitt - * Add support for strange GCC caller-save decisions - * (ported from switch_aarch64_gcc.h) - * 19-Aug-11 Alexey Borzenkov - * Correctly save ebp, ebx and cw - * 07-Sep-05 (py-dev mailing list discussion) - * removed 'ebx' from the register-saved. !!!! WARNING !!!! - * It means that this file can no longer be compiled statically! - * It is now only suitable as part of a dynamic library! - * 24-Nov-02 Christian Tismer - * needed to add another magic constant to insure - * that f in slp_eval_frame(PyFrameObject *f) - * STACK_REFPLUS will probably be 1 in most cases. - * gets included into the saved stack area. - * 17-Sep-02 Christian Tismer - * after virtualizing stack save/restore, the - * stack size shrunk a bit. Needed to introduce - * an adjustment STACK_MAGIC per platform. - * 15-Sep-02 Gerd Woetzel - * slightly changed framework for spark - * 31-Avr-02 Armin Rigo - * Added ebx, esi and edi register-saves. - * 01-Mar-02 Samual M. Rushing - * Ported from i386. - */ - -#define STACK_REFPLUS 1 - -#ifdef SLP_EVAL - -/* #define STACK_MAGIC 3 */ -/* the above works fine with gcc 2.96, but 2.95.3 wants this */ -#define STACK_MAGIC 0 - -#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5) -# define ATTR_NOCLONE __attribute__((noclone)) -#else -# define ATTR_NOCLONE -#endif - -static int -slp_switch(void) -{ - int err; -#ifdef _WIN32 - void *seh; -#endif - void *ebp, *ebx; - unsigned short cw; - int *stackref, stsizediff; - __asm__ volatile ("" : : : "esi", "edi"); - __asm__ volatile ("fstcw %0" : "=m" (cw)); - __asm__ volatile ("movl %%ebp, %0" : "=m" (ebp)); - __asm__ volatile ("movl %%ebx, %0" : "=m" (ebx)); -#ifdef _WIN32 - __asm__ volatile ( - "movl %%fs:0x0, %%eax\n" - "movl %%eax, %0\n" - : "=m" (seh) - : - : "eax"); -#endif - __asm__ ("movl %%esp, %0" : "=g" (stackref)); - { - SLP_SAVE_STATE(stackref, stsizediff); - __asm__ volatile ( - "addl %0, %%esp\n" - "addl %0, %%ebp\n" - : - : "r" (stsizediff) - ); - SLP_RESTORE_STATE(); - __asm__ volatile ("xorl %%eax, %%eax" : "=a" (err)); - } -#ifdef _WIN32 - __asm__ volatile ( - "movl %0, %%eax\n" - "movl %%eax, %%fs:0x0\n" - : - : "m" (seh) - : "eax"); -#endif - __asm__ volatile ("movl %0, %%ebx" : : "m" (ebx)); - __asm__ volatile ("movl %0, %%ebp" : : "m" (ebp)); - __asm__ volatile ("fldcw %0" : : "m" (cw)); - __asm__ volatile ("" : : : "esi", "edi"); - return err; -} - -#endif - -/* - * further self-processing support - */ - -/* - * if you want to add self-inspection tools, place them - * here. See the x86_msvc for the necessary defines. - * These features are highly experimental und not - * essential yet. - */ diff --git a/.env/Lib/site-packages/greenlet/slp_platformselect.h b/.env/Lib/site-packages/greenlet/slp_platformselect.h deleted file mode 100644 index d9b7d0a..0000000 --- a/.env/Lib/site-packages/greenlet/slp_platformselect.h +++ /dev/null @@ -1,77 +0,0 @@ -/* - * Platform Selection for Stackless Python - */ -#ifdef __cplusplus -extern "C" { -#endif - -#if defined(MS_WIN32) && !defined(MS_WIN64) && defined(_M_IX86) && defined(_MSC_VER) -# include "platform/switch_x86_msvc.h" /* MS Visual Studio on X86 */ -#elif defined(MS_WIN64) && defined(_M_X64) && defined(_MSC_VER) || defined(__MINGW64__) -# include "platform/switch_x64_msvc.h" /* MS Visual Studio on X64 */ -#elif defined(MS_WIN64) && defined(_M_ARM64) -# include "platform/switch_arm64_msvc.h" /* MS Visual Studio on ARM64 */ -#elif defined(__GNUC__) && defined(__amd64__) && defined(__ILP32__) -# include "platform/switch_x32_unix.h" /* gcc on amd64 with x32 ABI */ -#elif defined(__GNUC__) && defined(__amd64__) -# include "platform/switch_amd64_unix.h" /* gcc on amd64 */ -#elif defined(__GNUC__) && defined(__i386__) -# include "platform/switch_x86_unix.h" /* gcc on X86 */ -#elif defined(__GNUC__) && defined(__powerpc64__) && (defined(__linux__) || defined(__FreeBSD__)) -# include "platform/switch_ppc64_linux.h" /* gcc on PowerPC 64-bit */ -#elif defined(__GNUC__) && defined(__PPC__) && (defined(__linux__) || defined(__FreeBSD__)) -# include "platform/switch_ppc_linux.h" /* gcc on PowerPC */ -#elif defined(__GNUC__) && defined(__POWERPC__) && defined(__APPLE__) -# include "platform/switch_ppc_macosx.h" /* Apple MacOS X on 32-bit PowerPC */ -#elif defined(__GNUC__) && defined(__powerpc64__) && defined(_AIX) -# include "platform/switch_ppc64_aix.h" /* gcc on AIX/PowerPC 64-bit */ -#elif defined(__GNUC__) && defined(_ARCH_PPC) && defined(_AIX) -# include "platform/switch_ppc_aix.h" /* gcc on AIX/PowerPC */ -#elif defined(__GNUC__) && defined(__powerpc__) && defined(__NetBSD__) -#include "platform/switch_ppc_unix.h" /* gcc on NetBSD/powerpc */ -#elif defined(__GNUC__) && defined(sparc) -# include "platform/switch_sparc_sun_gcc.h" /* SunOS sparc with gcc */ -#elif defined(__GNUC__) && defined(__sparc__) -# include "platform/switch_sparc_sun_gcc.h" /* NetBSD sparc with gcc */ -#elif defined(__SUNPRO_C) && defined(sparc) && defined(sun) -# include "platform/switch_sparc_sun_gcc.h" /* SunStudio on amd64 */ -#elif defined(__SUNPRO_C) && defined(__amd64__) && defined(sun) -# include "platform/switch_amd64_unix.h" /* SunStudio on amd64 */ -#elif defined(__SUNPRO_C) && defined(__i386__) && defined(sun) -# include "platform/switch_x86_unix.h" /* SunStudio on x86 */ -#elif defined(__GNUC__) && defined(__s390__) && defined(__linux__) -# include "platform/switch_s390_unix.h" /* Linux/S390 */ -#elif defined(__GNUC__) && defined(__s390x__) && defined(__linux__) -# include "platform/switch_s390_unix.h" /* Linux/S390 zSeries (64-bit) */ -#elif defined(__GNUC__) && defined(__arm__) -# ifdef __APPLE__ -# include -# endif -# if TARGET_OS_IPHONE -# include "platform/switch_arm32_ios.h" /* iPhone OS on arm32 */ -# else -# include "platform/switch_arm32_gcc.h" /* gcc using arm32 */ -# endif -#elif defined(__GNUC__) && defined(__mips__) && defined(__linux__) -# include "platform/switch_mips_unix.h" /* Linux/MIPS */ -#elif defined(__GNUC__) && defined(__aarch64__) -# include "platform/switch_aarch64_gcc.h" /* Aarch64 ABI */ -#elif defined(__GNUC__) && defined(__mc68000__) -# include "platform/switch_m68k_gcc.h" /* gcc on m68k */ -#elif defined(__GNUC__) && defined(__csky__) -#include "platform/switch_csky_gcc.h" /* gcc on csky */ -# elif defined(__GNUC__) && defined(__riscv) -# include "platform/switch_riscv_unix.h" /* gcc on RISC-V */ -#elif defined(__GNUC__) && defined(__alpha__) -# include "platform/switch_alpha_unix.h" /* gcc on DEC Alpha */ -#elif defined(MS_WIN32) && defined(__llvm__) && defined(__aarch64__) -# include "platform/switch_aarch64_gcc.h" /* LLVM Aarch64 ABI for Windows */ -#elif defined(__GNUC__) && defined(__loongarch64) && defined(__linux__) -# include "platform/switch_loongarch64_linux.h" /* LoongArch64 */ -#elif defined(__GNUC__) && defined(__sh__) -# include "platform/switch_sh_gcc.h" /* SuperH */ -#endif - -#ifdef __cplusplus -}; -#endif diff --git a/.env/Lib/site-packages/greenlet/tests/__init__.py b/.env/Lib/site-packages/greenlet/tests/__init__.py deleted file mode 100644 index 1861360..0000000 --- a/.env/Lib/site-packages/greenlet/tests/__init__.py +++ /dev/null @@ -1,248 +0,0 @@ -# -*- coding: utf-8 -*- -""" -Tests for greenlet. - -""" -import os -import sys -import sysconfig -import unittest - -from gc import collect -from gc import get_objects -from threading import active_count as active_thread_count -from time import sleep -from time import time - -import psutil - -from greenlet import greenlet as RawGreenlet -from greenlet import getcurrent - -from greenlet._greenlet import get_pending_cleanup_count -from greenlet._greenlet import get_total_main_greenlets - -from . import leakcheck - -PY312 = sys.version_info[:2] >= (3, 12) -PY313 = sys.version_info[:2] >= (3, 13) -# XXX: First tested on 3.14a7. Revisit all uses of this on later versions to ensure they -# are still valid. -PY314 = sys.version_info[:2] >= (3, 14) - -WIN = sys.platform.startswith("win") -RUNNING_ON_GITHUB_ACTIONS = os.environ.get('GITHUB_ACTIONS') -RUNNING_ON_TRAVIS = os.environ.get('TRAVIS') or RUNNING_ON_GITHUB_ACTIONS -RUNNING_ON_APPVEYOR = os.environ.get('APPVEYOR') -RUNNING_ON_CI = RUNNING_ON_TRAVIS or RUNNING_ON_APPVEYOR -RUNNING_ON_MANYLINUX = os.environ.get('GREENLET_MANYLINUX') - -# Is the current interpreter free-threaded?) Note that this -# isn't the same as whether the GIL is enabled, this is the build-time -# value. Certain CPython details, like the garbage collector, -# work very differently on potentially-free-threaded builds than -# standard builds. -RUNNING_ON_FREETHREAD_BUILD = bool(sysconfig.get_config_var("Py_GIL_DISABLED")) - -class TestCaseMetaClass(type): - # wrap each test method with - # a) leak checks - def __new__(cls, classname, bases, classDict): - # pylint and pep8 fight over what this should be called (mcs or cls). - # pylint gets it right, but we can't scope disable pep8, so we go with - # its convention. - # pylint: disable=bad-mcs-classmethod-argument - check_totalrefcount = True - - # Python 3: must copy, we mutate the classDict. Interestingly enough, - # it doesn't actually error out, but under 3.6 we wind up wrapping - # and re-wrapping the same items over and over and over. - for key, value in list(classDict.items()): - if key.startswith('test') and callable(value): - classDict.pop(key) - if check_totalrefcount: - value = leakcheck.wrap_refcount(value) - classDict[key] = value - return type.__new__(cls, classname, bases, classDict) - - -class TestCase(unittest.TestCase, metaclass=TestCaseMetaClass): - - cleanup_attempt_sleep_duration = 0.001 - cleanup_max_sleep_seconds = 1 - - def wait_for_pending_cleanups(self, - initial_active_threads=None, - initial_main_greenlets=None): - initial_active_threads = initial_active_threads or self.threads_before_test - initial_main_greenlets = initial_main_greenlets or self.main_greenlets_before_test - sleep_time = self.cleanup_attempt_sleep_duration - # NOTE: This is racy! A Python-level thread object may be dead - # and gone, but the C thread may not yet have fired its - # destructors and added to the queue. There's no particular - # way to know that's about to happen. We try to watch the - # Python threads to make sure they, at least, have gone away. - # Counting the main greenlets, which we can easily do deterministically, - # also helps. - - # Always sleep at least once to let other threads run - sleep(sleep_time) - quit_after = time() + self.cleanup_max_sleep_seconds - # TODO: We could add an API that calls us back when a particular main greenlet is deleted? - # It would have to drop the GIL - while ( - get_pending_cleanup_count() - or active_thread_count() > initial_active_threads - or (not self.expect_greenlet_leak - and get_total_main_greenlets() > initial_main_greenlets)): - sleep(sleep_time) - if time() > quit_after: - print("Time limit exceeded.") - print("Threads: Waiting for only", initial_active_threads, - "-->", active_thread_count()) - print("MGlets : Waiting for only", initial_main_greenlets, - "-->", get_total_main_greenlets()) - break - collect() - - def count_objects(self, kind=list, exact_kind=True): - # pylint:disable=unidiomatic-typecheck - # Collect the garbage. - for _ in range(3): - collect() - if exact_kind: - return sum( - 1 - for x in get_objects() - if type(x) is kind - ) - # instances - return sum( - 1 - for x in get_objects() - if isinstance(x, kind) - ) - - greenlets_before_test = 0 - threads_before_test = 0 - main_greenlets_before_test = 0 - expect_greenlet_leak = False - - def count_greenlets(self): - """ - Find all the greenlets and subclasses tracked by the GC. - """ - return self.count_objects(RawGreenlet, False) - - def setUp(self): - # Ensure the main greenlet exists, otherwise the first test - # gets a false positive leak - super().setUp() - getcurrent() - self.threads_before_test = active_thread_count() - self.main_greenlets_before_test = get_total_main_greenlets() - self.wait_for_pending_cleanups(self.threads_before_test, self.main_greenlets_before_test) - self.greenlets_before_test = self.count_greenlets() - - def tearDown(self): - if getattr(self, 'skipTearDown', False): - return - - self.wait_for_pending_cleanups(self.threads_before_test, self.main_greenlets_before_test) - super().tearDown() - - def get_expected_returncodes_for_aborted_process(self): - import signal - # The child should be aborted in an unusual way. On POSIX - # platforms, this is done with abort() and signal.SIGABRT, - # which is reflected in a negative return value; however, on - # Windows, even though we observe the child print "Fatal - # Python error: Aborted" and in older versions of the C - # runtime "This application has requested the Runtime to - # terminate it in an unusual way," it always has an exit code - # of 3. This is interesting because 3 is the error code for - # ERROR_PATH_NOT_FOUND; BUT: the C runtime abort() function - # also uses this code. - # - # If we link to the static C library on Windows, the error - # code changes to '0xc0000409' (hex(3221226505)), which - # apparently is STATUS_STACK_BUFFER_OVERRUN; but "What this - # means is that nowadays when you get a - # STATUS_STACK_BUFFER_OVERRUN, it doesn’t actually mean that - # there is a stack buffer overrun. It just means that the - # application decided to terminate itself with great haste." - # - # - # On windows, we've also seen '0xc0000005' (hex(3221225477)). - # That's "Access Violation" - # - # See - # https://devblogs.microsoft.com/oldnewthing/20110519-00/?p=10623 - # and - # https://docs.microsoft.com/en-us/previous-versions/k089yyh0(v=vs.140)?redirectedfrom=MSDN - # and - # https://devblogs.microsoft.com/oldnewthing/20190108-00/?p=100655 - expected_exit = ( - -signal.SIGABRT, - # But beginning on Python 3.11, the faulthandler - # that prints the C backtraces sometimes segfaults after - # reporting the exception but before printing the stack. - # This has only been seen on linux/gcc. - -signal.SIGSEGV, - ) if not WIN else ( - 3, - 0xc0000409, - 0xc0000005, - ) - return expected_exit - - def get_process_uss(self): - """ - Return the current process's USS in bytes. - - uss is available on Linux, macOS, Windows. Also known as - "Unique Set Size", this is the memory which is unique to a - process and which would be freed if the process was terminated - right now. - - If this is not supported by ``psutil``, this raises the - :exc:`unittest.SkipTest` exception. - """ - try: - return psutil.Process().memory_full_info().uss - except AttributeError as e: - raise unittest.SkipTest("uss not supported") from e - - def run_script(self, script_name, show_output=True): - import subprocess - script = os.path.join( - os.path.dirname(__file__), - script_name, - ) - - try: - return subprocess.check_output([sys.executable, script], - encoding='utf-8', - stderr=subprocess.STDOUT) - except subprocess.CalledProcessError as ex: - if show_output: - print('-----') - print('Failed to run script', script) - print('~~~~~') - print(ex.output) - print('------') - raise - - - def assertScriptRaises(self, script_name, exitcodes=None): - import subprocess - with self.assertRaises(subprocess.CalledProcessError) as exc: - output = self.run_script(script_name, show_output=False) - __traceback_info__ = output - # We're going to fail the assertion if we get here, at least - # preserve the output in the traceback. - - if exitcodes is None: - exitcodes = self.get_expected_returncodes_for_aborted_process() - self.assertIn(exc.exception.returncode, exitcodes) - return exc.exception diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index b07f4f109e235bd43e806406e56ce0a54d15abdc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9226 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 diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_clearing_run_switches.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_clearing_run_switches.cpython-312.pyc deleted file mode 100644 index 943694ef2a9a824608ca93b83bb25f7d18a6f4d1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2075 zcmb7F&1)M+6rWk`YBwuca_yw4<4}eYY^$UWbttrPyOO1t(vm@Xdh@>K&HKGK z`fX^)M6iDO^?UDK4WU2erng`pucHZ!&<1i*3Axx+d|U!rQWQjSJ>4_z|892Kl3GJ0 zO+nX?n_5Gzx~U|JQaYAtAWK7puC(uu&}OP@sk{0bDj9H+4&QVy@+RaXS4p8teKg?* zY0s`?QmuY=P?PYkLCvbti-Hlv&bp_l>6$-hdS_$w-?6)1S*rE!g;7f>4A}-GXWpoi zyOb>0%aquJH3Q<tk?Y(b_f)UoAme%`WbR2+L+*8IE)LFGBVw@8T>2+FFIQC_$x zr2)Ac6G+3SHtXkDq1BL4QlT|32)v+XLbh;;*oo0enIH)pjgX0E;E5IiYw}G?EQD~+ z(1pBW^J&xkQ`Woa3?k5l4jqkvbOTWY%>+QY7=TVe0aHPb?Sr@fB)MS=W19|JNPpgcd5KiUoy9he)?9m-?2qGF)Zh5*bLVphh~B zvFQsQrR_$825zJ-*DX|Pu0?nV-+u1u&Lx0)GnRMG9klsHU-cltXyuh$d znRVhstz=LOjfB)n9NIqtl>P@?fGd}{0ohrKgv6;gHvtK9VrFfF)#SiP`)1y}uiw0n z{V|o&fUR%8``kGe0e;3xrP8MST3LGiu$&-^8iG|azXPZuwus}KF7QZWvNfCd&E+CLOtH49e)$M^JLmH3;w zx$KjGG6w4v;#-)~yrwOgYnbZ3hjo*bs_WPb-i|apa~k`4(b8#@igVBkTMafsTLS$^e%L-#uy;X+L#rA(Si?DWm7@gM2Uv_=g0dzWV zq!>oZ$F2^~l(S7`x}i)zOTN}jUTP#S{V58nMpP!v38B&h63U%iY!t$K= zRXEI}nAGT@G6FUiI0)m!bSUf3CDzul6}L)PbD1!1u%cOp5sGw+5*{#2_YCtN3EQ~YvW)P52;&)+shMk?|YJlG-#3!L*L~1O^nN!<>^+bqD zgeRORobbFx89P=kvtg~ec$(bq$3Sb;ujeB74ioV+mqk?3CCx;Zr47@B<=pKT4iwWk$a8iugkYG5QaE@|^Af diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_initialstub_already_started.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_initialstub_already_started.cpython-312.pyc deleted file mode 100644 index d1d4184e493f031459f441e4603f505579080042..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3478 zcmcImO>7fK6rNeH?e$-r#FU>9(*;2iOHB|#P(ZX*97+QPOOPN{R;1O&I|*xy?apoj zwo=JJN)lQrxsje4Rh0?~679LFL{B~R;wXudu5t(}m)-)2REbmH?0OwzT!Jd~$@@I> zX6C&&Z{GW6eh&n^2r=AHVLK1#tetheWai`Qn13Ma2$A?gUkY#$7`w?p7RQC@v^Ga1Vz*L5gz>DK1zEx~pl zv-M*1W-TyYTT6g&NwJ3ZI^w7)4}zT<^9NlOp48N+ap%!(Tw7~ttfppat^02=s(&p; zHGH?M!?`pl4a9Aq3r0CE(wIN(?IW75riVl|t?H_r()8g0QNKzvH<(aPi*kw*S;>i- zE>oQ-BDt9$SzXPfJG_6fz{V1c+r&cyY5{F@0=tiNTaJ>Yyetc#q2N(edB!;u;bM7r zlnkk9u`9wEZc4P_l&%>rIh!SEC4voJrbC*>5>R|eyHE9B(FoQ0FK3jDcu`JVC292< z>Cfi+FRBTefw2)aLA3r3k{;>5pbqqFs!rOoa^jji1mYn|NIFII!#V^(J3J_>DXAh3 zQfco}Y407`oZ*xuI-I7gz8m4wLcj{>8Sfj9-HJ`0o6(D*gG>CO4G^POBE$C2zKz?! zwvWtKzUeHcCM8*9t4OJ+k!_;w*wP7+=F5@p&b!Xi%ykj|yG13FCJ~-8o74%+zypR$ zyP@ids}B+NFfsGOa3;(+)Ioa}d|C^z0$MG_;p8G8Uge$RJ-2!$K3n8>+VT@ie6y+K zmC%&n1a0_1$$dl+fFz1@ISeU)yJ8O=h9F64`8ts#!z)SGGsZzbo+uqNncr5ZYU0& zMZ=`GvuCG9k{xIj<)LqnW2Ce2nUrL?Uqy4JKFoWwhy4HNVfhR=#rXx+ci1i_S)x|a zD4duLiM=S|iWv?l@4kF%ubJ|qo)HJcK~^2n4oMm!x~%I|9T?Uj{{S+@{Gl510l*h) z(RCT-z|U|=QC2Szzac0YKvqfB4Z(&V!wZTN!<3S=Zs0^k@$q^tODF>x%8r?`0x?{f zfh0-jMt!wIP%gYiQiBH1O2W)}Q8deyim))>Si@J4Ymb2y(26fOeeB)`vpaqWe-~a2 z?OhJF%!gVQLT!(_=0oj8e|s_1zS1mCoca3Vsy{Rpn~W9ro>>j=|GsVcz_IxQ#}*E} zzYspL9PXMAcRhFD<|y7oIM9us`2$lulRaObFRzNe#i}*So0ZVsi8HIgcb0=K^TC!` z{XySCu(K$1S|DNr2l$UzF_|~dU1)5NucNr~=Gg{0D>1fQv^CNA-!L|s-)&`pXkjV4 zTp_nuew5HR`U>juY{We72uJtBjtmF8_!#ooN4f{vc>#{p%&d0kY3~685Eg<{b(3{7 zp4p?tU9F2kTS-vdb!1WKTowFNzDeJ7&yo-}qZo_0tPEr2Oj)(iCdNd@n2~B^-N|H( z?SoHi05*nJf;$R5FF2p`=th&nxzEC3Hm5ew)&lzO7-hPut+g^4W9&ttu`3Cz@q%T6 z$>CAp%F3*E@;%HYzvYRssZ&RRukJzcCjlxJoQXO^mIy=s3i zh11r3_95*6T`Z%uqrlf47~|(0hxrWw;rcaRC}7yO&D>3eo6GK|d3V!-ySc!x`;jkD zaKGR=-15Tfz~N0V!uuzLWns^}uxAm8n|veQXZJS|bT->C_H0R@^OScM$eo@7{)2Z; zTpIh-Y!t}Er904E7m(LCwR>{+S4{=i3UZIv-Kv`o%#0L+EsLmi9n5j#!YyIUyH*P)bV@A ji>RTHU-NjY?FDCHecK|6tZglMikyYybn6moGE?FovP^_- diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_slp_switch.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_slp_switch.cpython-312.pyc deleted file mode 100644 index dfbd448cf720fce7e1929df3372e373f467bd436..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1303 zcmc&zPfrs;6rb5nx7&px{sYv58$$vcA_R}d7>y<-B0U&6*le;{cBk7d>vl7NNb{z1{yg)`ahmp&y!0mI^mN-_>kyooH0$OVw)!6 zUpG#N#0w(AhoqBHoHfM*;%hTOw>H+SkDrMeLYMFl9uKfDV&k7i-Nb}c2;pz)-rCR5 zsq)}W#i{!EAsJ#K?-X5O+i?m)c!E+8wR-{VKal~H_vAY}!=J&#mrS;fzDa3#^|%kD zb4@hRKm$!Q)<9!BPxsMe9UcO>p{yut>S3m53vSZYr9E(^zVxelYjSgbb$Jh5uP-0o Tx?P6{VCa}#BXQ4lwUgoxDEkP1 diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets.cpython-312.pyc deleted file mode 100644 index b3ce3c53346bcd9750a16362aa20c3ab0c5665ec..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1719 zcmaJ>&1)M+6rWkmuC%L7qpTB|G)`C-T%)#@NgxzTX-N|sN*y0eT7nRW?e17wMOum3 zl~aopRGPLfl-9T8SV}L&^e^b8_g*ZlAZPKx&<5&DajjcUeKV_-Wry^^yxI5Wea-v5 z*+0&l(GbvYzkluioj~YMF2n?~pus<|JwgsDAO|~A4HtkHBv=V2VG)NsL(DXXVDtg~hh6E-l>nWWf|-j)ZcPxn2ES#tnB+dy7|0* z^PM8Jp83U;DBC0ft?*p5U2UHmm*z+G~>UXP>G^51B5Ir|9xOl z^WW@|?Eb`LXKBxPwQtOHjhP)|wr}LSM!s)c?;6*4jJG=<0TA2TRF7Qdh|E2{^n}bX z9b%k-gI`p@QG}0iII5$k00%>gn^+w^i}(aU`riivZN$qt!uS-ivz+nYplHD_NK_SH z`Wo*ty8x$GVELDUbcu8I1dr{1kMoK}s# z$1Z`6XZ61Xc9_5z|0~Iu40MDu`?A`Rb~)QWd%mL|5e2_?lu_{OM;QsfahO8b=%o7S zTo;|&%=OUJzMSpc?aSG&oZVd8k*{#))0#sK;lwxU7wX3JL;Wd|J6;c6*hNyL`brl? F{s-XZWjO!< diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets2.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_three_greenlets2.cpython-312.pyc deleted file mode 100644 index 284c323861fb0bf8c0775619f738f19438c9d5cd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2573 zcmbVNO>7%Q6rNr0pMTbA*7<8`lT9I17Sw5sdZ{X^(k4(fjnp;)WLdH{yA#Kbv$kf} zDOt5>D-a16@U7ui^@b{>-jNV@F1Bpsu*xA4aNt&wTQ4xP-d(%Jp;bqkw=?tJemgVY z_ul>@iV=X@k3WA^xX1$VJDK>6@dk$n6aWu^1~NdSG`d7(u+7l8hqREwcFpNI-_C$drkR$#(nWK=SeUM|XfGoZgIsQB{4D~zvWACs>?r-3E z6nx8~b_}xJlW)?=L2D^8Us!|kic%=$j5`HWUCo)R2*O+*8@&Wg<6U{VYRcv+lvSlv zk_)<7mh&??RMq7>)2?2Uqo%GkG2P@$1cL46}TUs%o>1rwgFDC(M$$ICu<(k0gFPm?wQzr-_TkRr!^!VHsVAmd%-efRc>7G9IbOTkWRCyICw50q zZpZe7$hYCI!*yw@CAbmOb!KwsYKxgX#7IE(Dgqe;QgsODhjnaTMF~e}peL_QqAa%+ z1MdA50Q_7jx|OiIPDVxX`8k2fn+)3IA;cq$a z2XY;UK@=;?h#bwqn<2@HM(|`5cVhvYF0~)+sI^Ni=1s!tJ1u5pPdd?-rW(>zQ##X@ zQVl89md-b%^H0{B(pCGqr-=H{TMyUjiFAv3Z#O!G1~6oQihshM*QdA`BsPZeT0mwW zqWh+MAd`C({+V&FM`+q5)MGJCE_4pl=#_$i#`J$_9K_p3|81N+yw?(-j%1!ZfTTC{@WLrGf5YIh1)f6w;bNh)gd%lOP zm3rcAi%I!pSutON>q+hTs}6yUQ{EfW1L!E3xbaDeh}ZoTfzkF^2-6j_tEyo8Bd6 ze;{GU_9HQyKVW$3^nt`u!v}Gm8h$ziD9IMuV6XuOYsnTkxzCQ+pSIbN20K#QXtLu3 z;l-7GhTFP!|Jt_RWKP-CF5tJ~_v5wj&dm1bEinEYU~Rpow7}Tz%i`+X8#^~!V8RQm k$H$w@L>Do-bF)4$(E_sP|KfTdANgFdcIH=b!o8q>0ibvb@c;k- diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_two_greenlets.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/fail_switch_two_greenlets.cpython-312.pyc deleted file mode 100644 index 10955f46326fa36d959af382356aba9590652c13..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1690 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 diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_contextvars.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_contextvars.cpython-312.pyc deleted file mode 100644 index cce107a50eaf8eef61bdf2f729a6bcc4baac3638..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15399 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 K`Na+j;Qs;!Aap+f diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_cpp.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_cpp.cpython-312.pyc deleted file mode 100644 index cf25f8b45c350fd73310ddb3e6057aca14c6c3aa..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4087 zcmcInU2GHC6~6Ohdu+x@9P(pm0EZM7vykoX7KtL*-4wQv1_H}(aDzq@&rLGMnKAc{ zm)Nq^tVFwxgcNvbwklHhaVeGBKJ}q}SXF(f)R(vkwKWj65>LEk39D6k>bWx>PXI&r zrPtEEXYM`soO91T=R4Pbk47U1+MmAsx&9xZ9xIR0w!PEeN!fk{mh&bT}2}5RZ-{O}c@!FqZBK zuD(e^Y&2^2t*PxG~E5{Kk^H#&B2xS`jJV9)UN7+;2Zu^rW>X6e!RNR+EFvrF@su`Nx(lIku&&I5CX{LSc>J-o ze$Q)QN8@Z9y7`8P3Mk2sK4wQtN_K###uT=tDT1+)#TH=)wYwWvpT;>BCJMc-Tj^;# z=#6SFhYigRVfYBnj_BE{KBJM2lNHmBR%6lF>i(4M7^9vZ&@ z8|e61Q!^8%)bunq^s9J0R~SE~r-=z`AM0sM#|N?T@%Tx7Vw~z0-k(#`SJf$)_9VZ> zETvA*0(3A}Ak0CYwmb@RG|fRbhw5EDmGHw@&#h~}h*yRlcEoEP{nd{Cm7ND_ zJ4dQJN0vKIR)*?wbm7GOiCZ7t9I4CQHMy@U_kH%!eR;=P2z73KDxvP}pXvYX+B1Kw zGE})%mpeR`UyuEA>{qETV)x}YzYw~{AtY?DT@$IvM<%6N}|<_yb@TPmv^(cQV1vFuUtUiCmqkb!T80w%h zptNT_hkhE$Z1?#5u`Gej{ryITV^IiYTqe@|4U+aBZ0Cw*7`2BIi zzLx9f&;WWHQSLg_o6CrU-%Wj1xQwoI$>1nq;a2RPRP`*&5rFhYUPHDN19@w5|G^}0 zOV~)8nr=)%np+wsq`2+n9SNN#NZv4jE!k!BCbb1xpthJ(t!Z1#n7UzyG@Tgg3}A|) zOzK&zD0T$T;&k3pC$iX%=u|hTr5b4rdo9A)7(1}ZE{2>{b&9DSq4|j%F#!Xp-FZ~a zX0dk0{rr#+lh`sqM*6B^=B-@bvV-T&A3J^FJV26nID6j|X5}k~(-zhgf~`DZC&W~l zG>M{4K(T{)PleoX-MiR#yRR;{EgYXezHoZ}^qmvS@~*l(upSUPBbDHq zgkoKbZMWM#5B|CMx7{meFRZ+O@r&T^+JDo&G(NG^p00@XZT(Bq)_OEni*Bz*x8Et& zb{wwmIJ^=)^5y1-a!*YjsLBKNj;{Ka#KYeHTJN4}@1CziVzliW0kyri)`r?*3s>f^ zRQP(luNEJ!#)p^VBb8`fk{6=$(OVOXGq-0xzp^5|+n8DEIq-Y($HE^9E7DNCJ+@Go zFDyy%Z+b!4QuN3(%0TKbZEqagCj50<;MnfabMF9#s1BaXu`JET4iZxBl>xyAQWNUXhCs-&`m~iiPx5Mt=smthJts9NB z_*V0VzukuYpn@_`oJ0SX_L}F`9 z8Tk=(cah^|2aX6UJx<3Ulnz0613lqGfqhRpy8>ID4Dx}Wa8D#YFvh_R2D|6RK2<(Z zZk=1&`u=j@gRjI$AhI4n!69zVA>)kEmCgjSf@VJSgaE)n?a zipVhQ+Mu&tOeO3|>ssyD=Qu%RlE?^KVNm9n|4cgUI_5oc2Id%|Pp^qPRg It&X1m0yg2x9RL6T diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_extension_interface.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_extension_interface.cpython-312.pyc deleted file mode 100644 index 13e0dd31ed99bd3e276e30f22eb43a5458e75560..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7447 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 F{{bSiv)BLt diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_greenlet.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_greenlet.cpython-312.pyc deleted file mode 100644 index be08f4978d3dd8a8569f2bad9444dde9a496f83d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 76147 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
RR40uc@l-fNED%VS;42Rh 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 C5ryIa diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_leaks.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_leaks.cpython-312.pyc deleted file mode 100644 index ac77a6aee411fa6fe01400f280c8cdcf32188563..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 19895 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 Ap#T5? diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_stack_saved.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_stack_saved.cpython-312.pyc deleted file mode 100644 index a5e5b01ca1403ca12acd4bd5c9261f1cbdaf04dd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1342 zcmah}%}*0S6o0eb)}?GIs6Zh?2zXgOYPcCg8VxZJksgd*nkJiNcPK5}7H76-Nl7>~ z+M)?3({kd9;ZN}5-BSak&8jC)+)Rpz;p9xW)gmY_*|+cG_daIc+x_J8d4bV~kFVkt z1n`Y3Epj)7dXNbvFu*`2M8FUXmx+=>L_i?G92jCA3^x`ITn;|s$|A*c!j$f){ zaKL#wXsX*{4ic>cwLywv!qcWUMFSl)b0H{*%g7M&(k5!FbTI^S=%CrjrjQd79Ty?3 zUN&8u%|x5*gI5$A|!rK zgqfPetd~uviA2D%qp%{_9{M6~=_|IZQcKfURjRFHgDsGl)Z!`1;ag`*v-hIUC?+)e zByFTe7BzhZr{b$Px|WSD#&wcrzV*0{X*7sa>(R&Yr6`SCcxsI$)MCsWBN(SlY~8Wg zcF_iJFBe?P+TwC1rEiS!^$T*;Isc5A)-{ucf*iyLNqZUlxdk=3duO4zP!_8Jr4mq% zQ1g9*!Q=_a`>*4c^W)l7(623OJ&oR#ttXY^SZd;_-TqZv5 zLopk@B#okcoktkmCo=bR|J~??7;E^8zrX4qt@uX|{kIEZO_Hlppdtmz zH>(5Vm4Wg72S-xyR74#UrylV2R^_3JJXAIg<*@>)$-bTW;{2OOOyqHc6=|?MUDl4I zkv4rVv>V!=K9Z(tQpb)|l-_#(6zSO;-W}fWIhH0!7i%Z1h_;}=5t@$yg?W%s0m;ci7Fe)h6~@c)MsF~}uexRK>d lnsMrd$Sr2$n@0N?`|d&reFpzm=sWdvp~=GCAHZIXz+aqR55WKc diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_throw.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_throw.cpython-312.pyc deleted file mode 100644 index a8fbd55c7824245991659e6f0c6a12a67d4e14b8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7350 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 diff --git a/.env/Lib/site-packages/greenlet/tests/__pycache__/test_version.cpython-312.pyc b/.env/Lib/site-packages/greenlet/tests/__pycache__/test_version.cpython-312.pyc deleted file mode 100644 index 2beab4adf0e618506d82b49378b865beacfad9b2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2549 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 K=K+BqH}*f%c_r5X diff --git a/.env/Lib/site-packages/greenlet/tests/_test_extension.c b/.env/Lib/site-packages/greenlet/tests/_test_extension.c deleted file mode 100644 index 612b735..0000000 --- a/.env/Lib/site-packages/greenlet/tests/_test_extension.c +++ /dev/null @@ -1,258 +0,0 @@ -/* This is a set of functions used by test_extension_interface.py to test the - * Greenlet C API. - */ - -#include "../greenlet.h" - -#ifndef Py_RETURN_NONE -# define Py_RETURN_NONE return Py_INCREF(Py_None), Py_None -#endif - -#define TEST_MODULE_NAME "_test_extension" - -// CAUTION: MSVC is stupidly picky: -// -// "The compiler ignores, without warning, any __declspec keywords -// placed after * or & and in front of the variable identifier in a -// declaration." -// (https://docs.microsoft.com/en-us/cpp/cpp/declspec?view=msvc-160) -// -// So pointer return types must be handled differently (because of the -// trailing *), or you get inscrutable compiler warnings like "error -// C2059: syntax error: ''" -// -// In C23, there is a standard syntax for attributes, and -// GCC defines an attribute to use with this: [[gnu:noinline]]. -// In the future, this is expected to become standard. - -#if defined(__GNUC__) || defined(__clang__) -/* We used to check for GCC 4+ or 3.4+, but those compilers are - laughably out of date. Just assume they support it. */ -# define UNUSED(x) UNUSED_ ## x __attribute__((__unused__)) -#elif defined(_MSC_VER) -/* We used to check for && (_MSC_VER >= 1300) but that's also out of date. */ -# define UNUSED(x) UNUSED_ ## x -#endif - -static PyObject* -test_switch(PyObject* UNUSED(self), PyObject* greenlet) -{ - PyObject* result = NULL; - - if (greenlet == NULL || !PyGreenlet_Check(greenlet)) { - PyErr_BadArgument(); - return NULL; - } - - result = PyGreenlet_Switch((PyGreenlet*)greenlet, NULL, NULL); - if (result == NULL) { - if (!PyErr_Occurred()) { - PyErr_SetString(PyExc_AssertionError, - "greenlet.switch() failed for some reason."); - } - return NULL; - } - Py_INCREF(result); - return result; -} - -static PyObject* -test_switch_kwargs(PyObject* UNUSED(self), PyObject* args, PyObject* kwargs) -{ - PyGreenlet* g = NULL; - PyObject* result = NULL; - - PyArg_ParseTuple(args, "O!", &PyGreenlet_Type, &g); - - if (g == NULL || !PyGreenlet_Check(g)) { - PyErr_BadArgument(); - return NULL; - } - - result = PyGreenlet_Switch(g, NULL, kwargs); - if (result == NULL) { - if (!PyErr_Occurred()) { - PyErr_SetString(PyExc_AssertionError, - "greenlet.switch() failed for some reason."); - } - return NULL; - } - Py_XINCREF(result); - return result; -} - -static PyObject* -test_getcurrent(PyObject* UNUSED(self)) -{ - PyGreenlet* g = PyGreenlet_GetCurrent(); - if (g == NULL || !PyGreenlet_Check(g) || !PyGreenlet_ACTIVE(g)) { - PyErr_SetString(PyExc_AssertionError, - "getcurrent() returned an invalid greenlet"); - Py_XDECREF(g); - return NULL; - } - Py_DECREF(g); - Py_RETURN_NONE; -} - -static PyObject* -test_setparent(PyObject* UNUSED(self), PyObject* arg) -{ - PyGreenlet* current; - PyGreenlet* greenlet = NULL; - - if (arg == NULL || !PyGreenlet_Check(arg)) { - PyErr_BadArgument(); - return NULL; - } - if ((current = PyGreenlet_GetCurrent()) == NULL) { - return NULL; - } - greenlet = (PyGreenlet*)arg; - if (PyGreenlet_SetParent(greenlet, current)) { - Py_DECREF(current); - return NULL; - } - Py_DECREF(current); - if (PyGreenlet_Switch(greenlet, NULL, NULL) == NULL) { - return NULL; - } - Py_RETURN_NONE; -} - -static PyObject* -test_new_greenlet(PyObject* UNUSED(self), PyObject* callable) -{ - PyObject* result = NULL; - PyGreenlet* greenlet = PyGreenlet_New(callable, NULL); - - if (!greenlet) { - return NULL; - } - - result = PyGreenlet_Switch(greenlet, NULL, NULL); - Py_CLEAR(greenlet); - if (result == NULL) { - return NULL; - } - - Py_INCREF(result); - return result; -} - -static PyObject* -test_raise_dead_greenlet(PyObject* UNUSED(self)) -{ - PyErr_SetString(PyExc_GreenletExit, "test GreenletExit exception."); - return NULL; -} - -static PyObject* -test_raise_greenlet_error(PyObject* UNUSED(self)) -{ - PyErr_SetString(PyExc_GreenletError, "test greenlet.error exception"); - return NULL; -} - -static PyObject* -test_throw(PyObject* UNUSED(self), PyGreenlet* g) -{ - const char msg[] = "take that sucka!"; - PyObject* msg_obj = Py_BuildValue("s", msg); - PyGreenlet_Throw(g, PyExc_ValueError, msg_obj, NULL); - Py_DECREF(msg_obj); - if (PyErr_Occurred()) { - return NULL; - } - Py_RETURN_NONE; -} - -static PyObject* -test_throw_exact(PyObject* UNUSED(self), PyObject* args) -{ - PyGreenlet* g = NULL; - PyObject* typ = NULL; - PyObject* val = NULL; - PyObject* tb = NULL; - - if (!PyArg_ParseTuple(args, "OOOO:throw", &g, &typ, &val, &tb)) { - return NULL; - } - - PyGreenlet_Throw(g, typ, val, tb); - if (PyErr_Occurred()) { - return NULL; - } - Py_RETURN_NONE; -} - -static PyMethodDef test_methods[] = { - {"test_switch", - (PyCFunction)test_switch, - METH_O, - "Switch to the provided greenlet sending provided arguments, and \n" - "return the results."}, - {"test_switch_kwargs", - (PyCFunction)test_switch_kwargs, - METH_VARARGS | METH_KEYWORDS, - "Switch to the provided greenlet sending the provided keyword args."}, - {"test_getcurrent", - (PyCFunction)test_getcurrent, - METH_NOARGS, - "Test PyGreenlet_GetCurrent()"}, - {"test_setparent", - (PyCFunction)test_setparent, - METH_O, - "Se the parent of the provided greenlet and switch to it."}, - {"test_new_greenlet", - (PyCFunction)test_new_greenlet, - METH_O, - "Test PyGreenlet_New()"}, - {"test_raise_dead_greenlet", - (PyCFunction)test_raise_dead_greenlet, - METH_NOARGS, - "Just raise greenlet.GreenletExit"}, - {"test_raise_greenlet_error", - (PyCFunction)test_raise_greenlet_error, - METH_NOARGS, - "Just raise greenlet.error"}, - {"test_throw", - (PyCFunction)test_throw, - METH_O, - "Throw a ValueError at the provided greenlet"}, - {"test_throw_exact", - (PyCFunction)test_throw_exact, - METH_VARARGS, - "Throw exactly the arguments given at the provided greenlet"}, - {NULL, NULL, 0, NULL} -}; - - -#define INITERROR return NULL - -static struct PyModuleDef moduledef = {PyModuleDef_HEAD_INIT, - TEST_MODULE_NAME, - NULL, - 0, - test_methods, - NULL, - NULL, - NULL, - NULL}; - -PyMODINIT_FUNC -PyInit__test_extension(void) -{ - PyObject* module = NULL; - module = PyModule_Create(&moduledef); - - if (module == NULL) { - return NULL; - } - - PyGreenlet_Import(); -#ifdef Py_GIL_DISABLED - PyUnstable_Module_SetGIL(module, Py_MOD_GIL_NOT_USED); -#endif - return module; -} diff --git a/.env/Lib/site-packages/greenlet/tests/_test_extension.cp312-win_amd64.pyd b/.env/Lib/site-packages/greenlet/tests/_test_extension.cp312-win_amd64.pyd deleted file mode 100644 index e07bbe5d83de35846f53fcff6b96215833ef6444..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14336 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 Q?l}v&f7PD;vuynT0D$^Cp#T5? diff --git a/.env/Lib/site-packages/greenlet/tests/_test_extension_cpp.cpp b/.env/Lib/site-packages/greenlet/tests/_test_extension_cpp.cpp deleted file mode 100644 index bc3f178..0000000 --- a/.env/Lib/site-packages/greenlet/tests/_test_extension_cpp.cpp +++ /dev/null @@ -1,229 +0,0 @@ -/* This is a set of functions used to test C++ exceptions are not - * broken during greenlet switches - */ - -#include "../greenlet.h" -#include "../greenlet_compiler_compat.hpp" -#include -#include - -struct exception_t { - int depth; - exception_t(int depth) : depth(depth) {} -}; - -/* Functions are called via pointers to prevent inlining */ -static void (*p_test_exception_throw_nonstd)(int depth); -static void (*p_test_exception_throw_std)(); -static PyObject* (*p_test_exception_switch_recurse)(int depth, int left); - -static void -test_exception_throw_nonstd(int depth) -{ - throw exception_t(depth); -} - -static void -test_exception_throw_std() -{ - throw std::runtime_error("Thrown from an extension."); -} - -static PyObject* -test_exception_switch_recurse(int depth, int left) -{ - if (left > 0) { - return p_test_exception_switch_recurse(depth, left - 1); - } - - PyObject* result = NULL; - PyGreenlet* self = PyGreenlet_GetCurrent(); - if (self == NULL) - return NULL; - - try { - if (PyGreenlet_Switch(PyGreenlet_GET_PARENT(self), NULL, NULL) == NULL) { - Py_DECREF(self); - return NULL; - } - p_test_exception_throw_nonstd(depth); - PyErr_SetString(PyExc_RuntimeError, - "throwing C++ exception didn't work"); - } - catch (const exception_t& e) { - if (e.depth != depth) - PyErr_SetString(PyExc_AssertionError, "depth mismatch"); - else - result = PyLong_FromLong(depth); - } - catch (...) { - PyErr_SetString(PyExc_RuntimeError, "unexpected C++ exception"); - } - - Py_DECREF(self); - return result; -} - -/* test_exception_switch(int depth) - * - recurses depth times - * - switches to parent inside try/catch block - * - throws an exception that (expected to be caught in the same function) - * - verifies depth matches (exceptions shouldn't be caught in other greenlets) - */ -static PyObject* -test_exception_switch(PyObject* UNUSED(self), PyObject* args) -{ - int depth; - if (!PyArg_ParseTuple(args, "i", &depth)) - return NULL; - return p_test_exception_switch_recurse(depth, depth); -} - - -static PyObject* -py_test_exception_throw_nonstd(PyObject* UNUSED(self), PyObject* args) -{ - if (!PyArg_ParseTuple(args, "")) - return NULL; - p_test_exception_throw_nonstd(0); - PyErr_SetString(PyExc_AssertionError, "unreachable code running after throw"); - return NULL; -} - -static PyObject* -py_test_exception_throw_std(PyObject* UNUSED(self), PyObject* args) -{ - if (!PyArg_ParseTuple(args, "")) - return NULL; - p_test_exception_throw_std(); - PyErr_SetString(PyExc_AssertionError, "unreachable code running after throw"); - return NULL; -} - -static PyObject* -py_test_call(PyObject* UNUSED(self), PyObject* arg) -{ - PyObject* noargs = PyTuple_New(0); - PyObject* ret = PyObject_Call(arg, noargs, nullptr); - Py_DECREF(noargs); - return ret; -} - - - -/* test_exception_switch_and_do_in_g2(g2func) - * - creates new greenlet g2 to run g2func - * - switches to g2 inside try/catch block - * - verifies that no exception has been caught - * - * it is used together with test_exception_throw to verify that unhandled - * exceptions thrown in one greenlet do not propagate to other greenlet nor - * segfault the process. - */ -static PyObject* -test_exception_switch_and_do_in_g2(PyObject* UNUSED(self), PyObject* args) -{ - PyObject* g2func = NULL; - PyObject* result = NULL; - - if (!PyArg_ParseTuple(args, "O", &g2func)) - return NULL; - PyGreenlet* g2 = PyGreenlet_New(g2func, NULL); - if (!g2) { - return NULL; - } - - try { - result = PyGreenlet_Switch(g2, NULL, NULL); - if (!result) { - return NULL; - } - } - catch (const exception_t& e) { - /* if we are here the memory can be already corrupted and the program - * might crash before below py-level exception might become printed. - * -> print something to stderr to make it clear that we had entered - * this catch block. - * See comments in inner_bootstrap() - */ -#if defined(WIN32) || defined(_WIN32) - fprintf(stderr, "C++ exception unexpectedly caught in g1\n"); - PyErr_SetString(PyExc_AssertionError, "C++ exception unexpectedly caught in g1"); - Py_XDECREF(result); - return NULL; -#else - throw; -#endif - } - - Py_XDECREF(result); - Py_RETURN_NONE; -} - -static PyMethodDef test_methods[] = { - {"test_exception_switch", - (PyCFunction)&test_exception_switch, - METH_VARARGS, - "Switches to parent twice, to test exception handling and greenlet " - "switching."}, - {"test_exception_switch_and_do_in_g2", - (PyCFunction)&test_exception_switch_and_do_in_g2, - METH_VARARGS, - "Creates new greenlet g2 to run g2func and switches to it inside try/catch " - "block. Used together with test_exception_throw to verify that unhandled " - "C++ exceptions thrown in a greenlet doe not corrupt memory."}, - {"test_exception_throw_nonstd", - (PyCFunction)&py_test_exception_throw_nonstd, - METH_VARARGS, - "Throws non-standard C++ exception. Calling this function directly should abort the process." - }, - {"test_exception_throw_std", - (PyCFunction)&py_test_exception_throw_std, - METH_VARARGS, - "Throws standard C++ exception. Calling this function directly should abort the process." - }, - {"test_call", - (PyCFunction)&py_test_call, - METH_O, - "Call the given callable. Unlike calling it directly, this creates a " - "new C-level stack frame, which may be helpful in testing." - }, - {NULL, NULL, 0, NULL} -}; - - -static struct PyModuleDef moduledef = {PyModuleDef_HEAD_INIT, - "greenlet.tests._test_extension_cpp", - NULL, - 0, - test_methods, - NULL, - NULL, - NULL, - NULL}; - -PyMODINIT_FUNC -PyInit__test_extension_cpp(void) -{ - PyObject* module = NULL; - - module = PyModule_Create(&moduledef); - - if (module == NULL) { - return NULL; - } - - PyGreenlet_Import(); - if (_PyGreenlet_API == NULL) { - return NULL; - } - - p_test_exception_throw_nonstd = test_exception_throw_nonstd; - p_test_exception_throw_std = test_exception_throw_std; - p_test_exception_switch_recurse = test_exception_switch_recurse; -#ifdef Py_GIL_DISABLED - PyUnstable_Module_SetGIL(module, Py_MOD_GIL_NOT_USED); -#endif - - return module; -} diff --git a/.env/Lib/site-packages/greenlet/tests/fail_clearing_run_switches.py b/.env/Lib/site-packages/greenlet/tests/fail_clearing_run_switches.py deleted file mode 100644 index 6dd1492..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_clearing_run_switches.py +++ /dev/null @@ -1,47 +0,0 @@ -# -*- coding: utf-8 -*- -""" -If we have a run callable passed to the constructor or set as an -attribute, but we don't actually use that (because ``__getattribute__`` -or the like interferes), then when we clear callable before beginning -to run, there's an opportunity for Python code to run. - -""" -import greenlet - -g = None -main = greenlet.getcurrent() - -results = [] - -class RunCallable: - - def __del__(self): - results.append(('RunCallable', '__del__')) - main.switch('from RunCallable') - - -class G(greenlet.greenlet): - - def __getattribute__(self, name): - if name == 'run': - results.append(('G.__getattribute__', 'run')) - return run_func - return object.__getattribute__(self, name) - - -def run_func(): - results.append(('run_func', 'enter')) - - -g = G(RunCallable()) -# Try to start G. It will get to the point where it deletes -# its run callable C++ variable in inner_bootstrap. That triggers -# the __del__ method, which switches back to main before g -# actually even starts running. -x = g.switch() -results.append(('main: g.switch()', x)) -# In the C++ code, this results in g->g_switch() appearing to return, even though -# it has yet to run. -print('In main with', x, flush=True) -g.switch() -print('RESULTS', results) diff --git a/.env/Lib/site-packages/greenlet/tests/fail_cpp_exception.py b/.env/Lib/site-packages/greenlet/tests/fail_cpp_exception.py deleted file mode 100644 index fa4dc2e..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_cpp_exception.py +++ /dev/null @@ -1,33 +0,0 @@ -# -*- coding: utf-8 -*- -""" -Helper for testing a C++ exception throw aborts the process. - -Takes one argument, the name of the function in :mod:`_test_extension_cpp` to call. -""" -import sys -import greenlet -from greenlet.tests import _test_extension_cpp -print('fail_cpp_exception is running') - -def run_unhandled_exception_in_greenlet_aborts(): - def _(): - _test_extension_cpp.test_exception_switch_and_do_in_g2( - _test_extension_cpp.test_exception_throw_nonstd - ) - g1 = greenlet.greenlet(_) - g1.switch() - - -func_name = sys.argv[1] -try: - func = getattr(_test_extension_cpp, func_name) -except AttributeError: - if func_name == run_unhandled_exception_in_greenlet_aborts.__name__: - func = run_unhandled_exception_in_greenlet_aborts - elif func_name == 'run_as_greenlet_target': - g = greenlet.greenlet(_test_extension_cpp.test_exception_throw_std) - func = g.switch - else: - raise -print('raising', func, flush=True) -func() diff --git a/.env/Lib/site-packages/greenlet/tests/fail_initialstub_already_started.py b/.env/Lib/site-packages/greenlet/tests/fail_initialstub_already_started.py deleted file mode 100644 index c1a44ef..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_initialstub_already_started.py +++ /dev/null @@ -1,78 +0,0 @@ -""" -Testing initialstub throwing an already started exception. -""" - -import greenlet - -a = None -b = None -c = None -main = greenlet.getcurrent() - -# If we switch into a dead greenlet, -# we go looking for its parents. -# if a parent is not yet started, we start it. - -results = [] - -def a_run(*args): - #results.append('A') - results.append(('Begin A', args)) - - -def c_run(): - results.append('Begin C') - b.switch('From C') - results.append('C done') - -class A(greenlet.greenlet): pass - -class B(greenlet.greenlet): - doing_it = False - def __getattribute__(self, name): - if name == 'run' and not self.doing_it: - assert greenlet.getcurrent() is c - self.doing_it = True - results.append('Switch to b from B.__getattribute__ in ' - + type(greenlet.getcurrent()).__name__) - b.switch() - results.append('B.__getattribute__ back from main in ' - + type(greenlet.getcurrent()).__name__) - if name == 'run': - name = '_B_run' - return object.__getattribute__(self, name) - - def _B_run(self, *arg): - results.append(('Begin B', arg)) - results.append('_B_run switching to main') - main.switch('From B') - -class C(greenlet.greenlet): - pass -a = A(a_run) -b = B(parent=a) -c = C(c_run, b) - -# Start a child; while running, it will start B, -# but starting B will ALSO start B. -result = c.switch() -results.append(('main from c', result)) - -# Switch back to C, which was in the middle of switching -# already. This will throw the ``GreenletStartedWhileInPython`` -# exception, which results in parent A getting started (B is finished) -c.switch() - -results.append(('A dead?', a.dead, 'B dead?', b.dead, 'C dead?', c.dead)) - -# A and B should both be dead now. -assert a.dead -assert b.dead -assert not c.dead - -result = c.switch() -results.append(('main from c.2', result)) -# Now C is dead -assert c.dead - -print("RESULTS:", results) diff --git a/.env/Lib/site-packages/greenlet/tests/fail_slp_switch.py b/.env/Lib/site-packages/greenlet/tests/fail_slp_switch.py deleted file mode 100644 index 0990526..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_slp_switch.py +++ /dev/null @@ -1,29 +0,0 @@ -# -*- coding: utf-8 -*- -""" -A test helper for seeing what happens when slp_switch() -fails. -""" -# pragma: no cover - -import greenlet - - -print('fail_slp_switch is running', flush=True) - -runs = [] -def func(): - runs.append(1) - greenlet.getcurrent().parent.switch() - runs.append(2) - greenlet.getcurrent().parent.switch() - runs.append(3) - -g = greenlet._greenlet.UnswitchableGreenlet(func) -g.switch() -assert runs == [1] -g.switch() -assert runs == [1, 2] -g.force_slp_switch_error = True - -# This should crash. -g.switch() diff --git a/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets.py b/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets.py deleted file mode 100644 index e151b19..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets.py +++ /dev/null @@ -1,44 +0,0 @@ -""" -Uses a trace function to switch greenlets at unexpected times. - -In the trace function, we switch from the current greenlet to another -greenlet, which switches -""" -import greenlet - -g1 = None -g2 = None - -switch_to_g2 = False - -def tracefunc(*args): - print('TRACE', *args) - global switch_to_g2 - if switch_to_g2: - switch_to_g2 = False - g2.switch() - print('\tLEAVE TRACE', *args) - -def g1_run(): - print('In g1_run') - global switch_to_g2 - switch_to_g2 = True - from_parent = greenlet.getcurrent().parent.switch() - print('Return to g1_run') - print('From parent', from_parent) - -def g2_run(): - #g1.switch() - greenlet.getcurrent().parent.switch() - -greenlet.settrace(tracefunc) - -g1 = greenlet.greenlet(g1_run) -g2 = greenlet.greenlet(g2_run) - -# This switch didn't actually finish! -# And if it did, it would raise TypeError -# because g1_run() doesn't take any arguments. -g1.switch(1) -print('Back in main') -g1.switch(2) diff --git a/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets2.py b/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets2.py deleted file mode 100644 index 1f6b66b..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_switch_three_greenlets2.py +++ /dev/null @@ -1,55 +0,0 @@ -""" -Like fail_switch_three_greenlets, but the call into g1_run would actually be -valid. -""" -import greenlet - -g1 = None -g2 = None - -switch_to_g2 = True - -results = [] - -def tracefunc(*args): - results.append(('trace', args[0])) - print('TRACE', *args) - global switch_to_g2 - if switch_to_g2: - switch_to_g2 = False - g2.switch('g2 from tracefunc') - print('\tLEAVE TRACE', *args) - -def g1_run(arg): - results.append(('g1 arg', arg)) - print('In g1_run') - from_parent = greenlet.getcurrent().parent.switch('from g1_run') - results.append(('g1 from parent', from_parent)) - return 'g1 done' - -def g2_run(arg): - #g1.switch() - results.append(('g2 arg', arg)) - parent = greenlet.getcurrent().parent.switch('from g2_run') - global switch_to_g2 - switch_to_g2 = False - results.append(('g2 from parent', parent)) - return 'g2 done' - - -greenlet.settrace(tracefunc) - -g1 = greenlet.greenlet(g1_run) -g2 = greenlet.greenlet(g2_run) - -x = g1.switch('g1 from main') -results.append(('main g1', x)) -print('Back in main', x) -x = g1.switch('g2 from main') -results.append(('main g2', x)) -print('back in amain again', x) -x = g1.switch('g1 from main 2') -results.append(('main g1.2', x)) -x = g2.switch() -results.append(('main g2.2', x)) -print("RESULTS:", results) diff --git a/.env/Lib/site-packages/greenlet/tests/fail_switch_two_greenlets.py b/.env/Lib/site-packages/greenlet/tests/fail_switch_two_greenlets.py deleted file mode 100644 index 3e52345..0000000 --- a/.env/Lib/site-packages/greenlet/tests/fail_switch_two_greenlets.py +++ /dev/null @@ -1,41 +0,0 @@ -""" -Uses a trace function to switch greenlets at unexpected times. - -In the trace function, we switch from the current greenlet to another -greenlet, which switches -""" -import greenlet - -g1 = None -g2 = None - -switch_to_g2 = False - -def tracefunc(*args): - print('TRACE', *args) - global switch_to_g2 - if switch_to_g2: - switch_to_g2 = False - g2.switch() - print('\tLEAVE TRACE', *args) - -def g1_run(): - print('In g1_run') - global switch_to_g2 - switch_to_g2 = True - greenlet.getcurrent().parent.switch() - print('Return to g1_run') - print('Falling off end of g1_run') - -def g2_run(): - g1.switch() - print('Falling off end of g2') - -greenlet.settrace(tracefunc) - -g1 = greenlet.greenlet(g1_run) -g2 = greenlet.greenlet(g2_run) - -g1.switch() -print('Falling off end of main') -g2.switch() diff --git a/.env/Lib/site-packages/greenlet/tests/leakcheck.py b/.env/Lib/site-packages/greenlet/tests/leakcheck.py deleted file mode 100644 index f45361e..0000000 --- a/.env/Lib/site-packages/greenlet/tests/leakcheck.py +++ /dev/null @@ -1,336 +0,0 @@ -# Copyright (c) 2018 gevent community -# Copyright (c) 2021 greenlet community -# -# This was originally part of gevent's test suite. The main author -# (Jason Madden) vendored a copy of it into greenlet. -# -# Permission is hereby granted, free of charge, to any person obtaining a copy -# of this software and associated documentation files (the "Software"), to deal -# in the Software without restriction, including without limitation the rights -# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -# copies of the Software, and to permit persons to whom the Software is -# furnished to do so, subject to the following conditions: -# -# The above copyright notice and this permission notice shall be included in -# all copies or substantial portions of the Software. -# -# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -# THE SOFTWARE. -from __future__ import print_function - -import os -import sys -import gc - -from functools import wraps -import unittest - - -import objgraph - -# graphviz 0.18 (Nov 7 2021), available only on Python 3.6 and newer, -# has added type hints (sigh). It wants to use ``typing.Literal`` for -# some stuff, but that's only available on Python 3.9+. If that's not -# found, it creates a ``unittest.mock.MagicMock`` object and annotates -# with that. These are GC'able objects, and doing almost *anything* -# with them results in an explosion of objects. For example, trying to -# compare them for equality creates new objects. This causes our -# leakchecks to fail, with reports like: -# -# greenlet.tests.leakcheck.LeakCheckError: refcount increased by [337, 1333, 343, 430, 530, 643, 769] -# _Call 1820 +546 -# dict 4094 +76 -# MagicProxy 585 +73 -# tuple 2693 +66 -# _CallList 24 +3 -# weakref 1441 +1 -# function 5996 +1 -# type 736 +1 -# cell 592 +1 -# MagicMock 8 +1 -# -# To avoid this, we *could* filter this type of object out early. In -# principle it could leak, but we don't use mocks in greenlet, so it -# doesn't leak from us. However, a further issue is that ``MagicMock`` -# objects have subobjects that are also GC'able, like ``_Call``, and -# those create new mocks of their own too. So we'd have to filter them -# as well, and they're not public. That's OK, we can workaround the -# problem by being very careful to never compare by equality or other -# user-defined operators, only using object identity or other builtin -# functions. - -RUNNING_ON_GITHUB_ACTIONS = os.environ.get('GITHUB_ACTIONS') -RUNNING_ON_TRAVIS = os.environ.get('TRAVIS') or RUNNING_ON_GITHUB_ACTIONS -RUNNING_ON_APPVEYOR = os.environ.get('APPVEYOR') -RUNNING_ON_CI = RUNNING_ON_TRAVIS or RUNNING_ON_APPVEYOR -RUNNING_ON_MANYLINUX = os.environ.get('GREENLET_MANYLINUX') -SKIP_LEAKCHECKS = RUNNING_ON_MANYLINUX or os.environ.get('GREENLET_SKIP_LEAKCHECKS') -SKIP_FAILING_LEAKCHECKS = os.environ.get('GREENLET_SKIP_FAILING_LEAKCHECKS') -ONLY_FAILING_LEAKCHECKS = os.environ.get('GREENLET_ONLY_FAILING_LEAKCHECKS') - -def ignores_leakcheck(func): - """ - Ignore the given object during leakchecks. - - Can be applied to a method, in which case the method will run, but - will not be subject to leak checks. - - If applied to a class, the entire class will be skipped during leakchecks. This - is intended to be used for classes that are very slow and cause problems such as - test timeouts; typically it will be used for classes that are subclasses of a base - class and specify variants of behaviour (such as pool sizes). - """ - func.ignore_leakcheck = True - return func - -def fails_leakcheck(func): - """ - Mark that the function is known to leak. - """ - func.fails_leakcheck = True - if SKIP_FAILING_LEAKCHECKS: - func = unittest.skip("Skipping known failures")(func) - return func - -class LeakCheckError(AssertionError): - pass - -if hasattr(sys, 'getobjects'): - # In a Python build with ``--with-trace-refs``, make objgraph - # trace *all* the objects, not just those that are tracked by the - # GC - class _MockGC(object): - def get_objects(self): - return sys.getobjects(0) # pylint:disable=no-member - def __getattr__(self, name): - return getattr(gc, name) - objgraph.gc = _MockGC() - fails_strict_leakcheck = fails_leakcheck -else: - def fails_strict_leakcheck(func): - """ - Decorator for a function that is known to fail when running - strict (``sys.getobjects()``) leakchecks. - - This type of leakcheck finds all objects, even those, such as - strings, which are not tracked by the garbage collector. - """ - return func - -class ignores_types_in_strict_leakcheck(object): - def __init__(self, types): - self.types = types - def __call__(self, func): - func.leakcheck_ignore_types = self.types - return func - -class _RefCountChecker(object): - - # Some builtin things that we ignore - # XXX: Those things were ignored by gevent, but they're important here, - # presumably. - IGNORED_TYPES = () #(tuple, dict, types.FrameType, types.TracebackType) - - # Names of types that should be ignored. Use this when we cannot - # or don't want to import the class directly. - IGNORED_TYPE_NAMES = ( - # This appears in Python3.14 with the JIT enabled. It - # doesn't seem to be directly exposed to Python; the only way to get - # one is to cause code to get jitted and then look for all objects - # and find one with this name. But they multiply as code - # executes and gets jitted, in ways we don't want to rely on. - # So just ignore it. - 'uop_executor', - ) - - def __init__(self, testcase, function): - self.testcase = testcase - self.function = function - self.deltas = [] - self.peak_stats = {} - self.ignored_types = () - - # The very first time we are called, we have already been - # self.setUp() by the test runner, so we don't need to do it again. - self.needs_setUp = False - - def _include_object_p(self, obj): - # pylint:disable=too-many-return-statements - # - # See the comment block at the top. We must be careful to - # avoid invoking user-defined operations. - if obj is self: - return False - kind = type(obj) - # ``self._include_object_p == obj`` returns NotImplemented - # for non-function objects, which causes the interpreter - # to try to reverse the order of arguments...which leads - # to the explosion of mock objects. We don't want that, so we implement - # the check manually. - if kind == type(self._include_object_p): # pylint: disable=unidiomatic-typecheck - try: - # pylint:disable=not-callable - exact_method_equals = self._include_object_p.__eq__(obj) - except AttributeError: - # Python 2.7 methods may only have __cmp__, and that raises a - # TypeError for non-method arguments - # pylint:disable=no-member - exact_method_equals = self._include_object_p.__cmp__(obj) == 0 - - if exact_method_equals is not NotImplemented and exact_method_equals: - return False - - # Similarly, we need to check identity in our __dict__ to avoid mock explosions. - for x in self.__dict__.values(): - if obj is x: - return False - - - if ( - kind in self.ignored_types - or kind in self.IGNORED_TYPES - or kind.__name__ in self.IGNORED_TYPE_NAMES - ): - return False - - - return True - - def _growth(self): - return objgraph.growth(limit=None, peak_stats=self.peak_stats, - filter=self._include_object_p) - - def _report_diff(self, growth): - if not growth: - return "" - - lines = [] - width = max(len(name) for name, _, _ in growth) - for name, count, delta in growth: - lines.append('%-*s%9d %+9d' % (width, name, count, delta)) - - diff = '\n'.join(lines) - return diff - - - def _run_test(self, args, kwargs): - gc_enabled = gc.isenabled() - gc.disable() - - if self.needs_setUp: - self.testcase.setUp() - self.testcase.skipTearDown = False - try: - self.function(self.testcase, *args, **kwargs) - finally: - self.testcase.tearDown() - self.testcase.doCleanups() - self.testcase.skipTearDown = True - self.needs_setUp = True - if gc_enabled: - gc.enable() - - def _growth_after(self): - # Grab post snapshot - # pylint:disable=no-member - if 'urlparse' in sys.modules: - sys.modules['urlparse'].clear_cache() - if 'urllib.parse' in sys.modules: - sys.modules['urllib.parse'].clear_cache() - - return self._growth() - - def _check_deltas(self, growth): - # Return false when we have decided there is no leak, - # true if we should keep looping, raises an assertion - # if we have decided there is a leak. - - deltas = self.deltas - if not deltas: - # We haven't run yet, no data, keep looping - return True - - if gc.garbage: - raise LeakCheckError("Generated uncollectable garbage %r" % (gc.garbage,)) - - - # the following configurations are classified as "no leak" - # [0, 0] - # [x, 0, 0] - # [... a, b, c, d] where a+b+c+d = 0 - # - # the following configurations are classified as "leak" - # [... z, z, z] where z > 0 - - if deltas[-2:] == [0, 0] and len(deltas) in (2, 3): - return False - - if deltas[-3:] == [0, 0, 0]: - return False - - if len(deltas) >= 4 and sum(deltas[-4:]) == 0: - return False - - if len(deltas) >= 3 and deltas[-1] > 0 and deltas[-1] == deltas[-2] and deltas[-2] == deltas[-3]: - diff = self._report_diff(growth) - raise LeakCheckError('refcount increased by %r\n%s' % (deltas, diff)) - - # OK, we don't know for sure yet. Let's search for more - if sum(deltas[-3:]) <= 0 or sum(deltas[-4:]) <= 0 or deltas[-4:].count(0) >= 2: - # this is suspicious, so give a few more runs - limit = 11 - else: - limit = 7 - if len(deltas) >= limit: - raise LeakCheckError('refcount increased by %r\n%s' - % (deltas, - self._report_diff(growth))) - - # We couldn't decide yet, keep going - return True - - def __call__(self, args, kwargs): - for _ in range(3): - gc.collect() - - expect_failure = getattr(self.function, 'fails_leakcheck', False) - if expect_failure: - self.testcase.expect_greenlet_leak = True - self.ignored_types = getattr(self.function, "leakcheck_ignore_types", ()) - - # Capture state before; the incremental will be - # updated by each call to _growth_after - growth = self._growth() - - try: - while self._check_deltas(growth): - self._run_test(args, kwargs) - - growth = self._growth_after() - - self.deltas.append(sum((stat[2] for stat in growth))) - except LeakCheckError: - if not expect_failure: - raise - else: - if expect_failure: - raise LeakCheckError("Expected %s to leak but it did not." % (self.function,)) - -def wrap_refcount(method): - if getattr(method, 'ignore_leakcheck', False) or SKIP_LEAKCHECKS: - return method - - @wraps(method) - def wrapper(self, *args, **kwargs): # pylint:disable=too-many-branches - if getattr(self, 'ignore_leakcheck', False): - raise unittest.SkipTest("This class ignored during leakchecks") - if ONLY_FAILING_LEAKCHECKS and not getattr(method, 'fails_leakcheck', False): - raise unittest.SkipTest("Only running tests that fail leakchecks.") - return _RefCountChecker(self, method)(args, kwargs) - - return wrapper diff --git a/.env/Lib/site-packages/greenlet/tests/test_contextvars.py b/.env/Lib/site-packages/greenlet/tests/test_contextvars.py deleted file mode 100644 index b0d1ccf..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_contextvars.py +++ /dev/null @@ -1,312 +0,0 @@ -from __future__ import print_function - -import gc -import sys -import unittest - -from functools import partial -from unittest import skipUnless -from unittest import skipIf - -from greenlet import greenlet -from greenlet import getcurrent -from . import TestCase -from . import PY314 - -try: - from contextvars import Context - from contextvars import ContextVar - from contextvars import copy_context - # From the documentation: - # - # Important: Context Variables should be created at the top module - # level and never in closures. Context objects hold strong - # references to context variables which prevents context variables - # from being properly garbage collected. - ID_VAR = ContextVar("id", default=None) - VAR_VAR = ContextVar("var", default=None) - ContextVar = None -except ImportError: - Context = ContextVar = copy_context = None - -# We don't support testing if greenlet's built-in context var support is disabled. -@skipUnless(Context is not None, "ContextVar not supported") -class ContextVarsTests(TestCase): - def _new_ctx_run(self, *args, **kwargs): - return copy_context().run(*args, **kwargs) - - def _increment(self, greenlet_id, callback, counts, expect): - ctx_var = ID_VAR - if expect is None: - self.assertIsNone(ctx_var.get()) - else: - self.assertEqual(ctx_var.get(), expect) - ctx_var.set(greenlet_id) - for _ in range(2): - counts[ctx_var.get()] += 1 - callback() - - def _test_context(self, propagate_by): - # pylint:disable=too-many-branches - ID_VAR.set(0) - - callback = getcurrent().switch - counts = dict((i, 0) for i in range(5)) - - lets = [ - greenlet(partial( - partial( - copy_context().run, - self._increment - ) if propagate_by == "run" else self._increment, - greenlet_id=i, - callback=callback, - counts=counts, - expect=( - i - 1 if propagate_by == "share" else - 0 if propagate_by in ("set", "run") else None - ) - )) - for i in range(1, 5) - ] - - for let in lets: - if propagate_by == "set": - let.gr_context = copy_context() - elif propagate_by == "share": - let.gr_context = getcurrent().gr_context - - for i in range(2): - counts[ID_VAR.get()] += 1 - for let in lets: - let.switch() - - if propagate_by == "run": - # Must leave each context.run() in reverse order of entry - for let in reversed(lets): - let.switch() - else: - # No context.run(), so fine to exit in any order. - for let in lets: - let.switch() - - for let in lets: - self.assertTrue(let.dead) - # When using run(), we leave the run() as the greenlet dies, - # and there's no context "underneath". When not using run(), - # gr_context still reflects the context the greenlet was - # running in. - if propagate_by == 'run': - self.assertIsNone(let.gr_context) - else: - self.assertIsNotNone(let.gr_context) - - - if propagate_by == "share": - self.assertEqual(counts, {0: 1, 1: 1, 2: 1, 3: 1, 4: 6}) - else: - self.assertEqual(set(counts.values()), set([2])) - - def test_context_propagated_by_context_run(self): - self._new_ctx_run(self._test_context, "run") - - def test_context_propagated_by_setting_attribute(self): - self._new_ctx_run(self._test_context, "set") - - def test_context_not_propagated(self): - self._new_ctx_run(self._test_context, None) - - def test_context_shared(self): - self._new_ctx_run(self._test_context, "share") - - def test_break_ctxvars(self): - let1 = greenlet(copy_context().run) - let2 = greenlet(copy_context().run) - let1.switch(getcurrent().switch) - let2.switch(getcurrent().switch) - # Since let2 entered the current context and let1 exits its own, the - # interpreter emits: - # RuntimeError: cannot exit context: thread state references a different context object - let1.switch() - - def test_not_broken_if_using_attribute_instead_of_context_run(self): - let1 = greenlet(getcurrent().switch) - let2 = greenlet(getcurrent().switch) - let1.gr_context = copy_context() - let2.gr_context = copy_context() - let1.switch() - let2.switch() - let1.switch() - let2.switch() - - def test_context_assignment_while_running(self): - # pylint:disable=too-many-statements - ID_VAR.set(None) - - def target(): - self.assertIsNone(ID_VAR.get()) - self.assertIsNone(gr.gr_context) - - # Context is created on first use - ID_VAR.set(1) - self.assertIsInstance(gr.gr_context, Context) - self.assertEqual(ID_VAR.get(), 1) - self.assertEqual(gr.gr_context[ID_VAR], 1) - - # Clearing the context makes it get re-created as another - # empty context when next used - old_context = gr.gr_context - gr.gr_context = None # assign None while running - self.assertIsNone(ID_VAR.get()) - self.assertIsNone(gr.gr_context) - ID_VAR.set(2) - self.assertIsInstance(gr.gr_context, Context) - self.assertEqual(ID_VAR.get(), 2) - self.assertEqual(gr.gr_context[ID_VAR], 2) - - new_context = gr.gr_context - getcurrent().parent.switch((old_context, new_context)) - # parent switches us back to old_context - - self.assertEqual(ID_VAR.get(), 1) - gr.gr_context = new_context # assign non-None while running - self.assertEqual(ID_VAR.get(), 2) - - getcurrent().parent.switch() - # parent switches us back to no context - self.assertIsNone(ID_VAR.get()) - self.assertIsNone(gr.gr_context) - gr.gr_context = old_context - self.assertEqual(ID_VAR.get(), 1) - - getcurrent().parent.switch() - # parent switches us back to no context - self.assertIsNone(ID_VAR.get()) - self.assertIsNone(gr.gr_context) - - gr = greenlet(target) - - with self.assertRaisesRegex(AttributeError, "can't delete context attribute"): - del gr.gr_context - - self.assertIsNone(gr.gr_context) - old_context, new_context = gr.switch() - self.assertIs(new_context, gr.gr_context) - self.assertEqual(old_context[ID_VAR], 1) - self.assertEqual(new_context[ID_VAR], 2) - self.assertEqual(new_context.run(ID_VAR.get), 2) - gr.gr_context = old_context # assign non-None while suspended - gr.switch() - self.assertIs(gr.gr_context, new_context) - gr.gr_context = None # assign None while suspended - gr.switch() - self.assertIs(gr.gr_context, old_context) - gr.gr_context = None - gr.switch() - self.assertIsNone(gr.gr_context) - - # Make sure there are no reference leaks - gr = None - gc.collect() - # Python 3.14 elides reference counting operations - # in some cases. See https://github.com/python/cpython/pull/130708 - self.assertEqual(sys.getrefcount(old_context), 2 if not PY314 else 1) - self.assertEqual(sys.getrefcount(new_context), 2 if not PY314 else 1) - - def test_context_assignment_different_thread(self): - import threading - VAR_VAR.set(None) - ctx = Context() - - is_running = threading.Event() - should_suspend = threading.Event() - did_suspend = threading.Event() - should_exit = threading.Event() - holder = [] - - def greenlet_in_thread_fn(): - VAR_VAR.set(1) - is_running.set() - should_suspend.wait(10) - VAR_VAR.set(2) - getcurrent().parent.switch() - holder.append(VAR_VAR.get()) - - def thread_fn(): - gr = greenlet(greenlet_in_thread_fn) - gr.gr_context = ctx - holder.append(gr) - gr.switch() - did_suspend.set() - should_exit.wait(10) - gr.switch() - del gr - greenlet() # trigger cleanup - - thread = threading.Thread(target=thread_fn, daemon=True) - thread.start() - is_running.wait(10) - gr = holder[0] - - # Can't access or modify context if the greenlet is running - # in a different thread - with self.assertRaisesRegex(ValueError, "running in a different"): - getattr(gr, 'gr_context') - with self.assertRaisesRegex(ValueError, "running in a different"): - gr.gr_context = None - - should_suspend.set() - did_suspend.wait(10) - - # OK to access and modify context if greenlet is suspended - self.assertIs(gr.gr_context, ctx) - self.assertEqual(gr.gr_context[VAR_VAR], 2) - gr.gr_context = None - - should_exit.set() - thread.join(10) - - self.assertEqual(holder, [gr, None]) - - # Context can still be accessed/modified when greenlet is dead: - self.assertIsNone(gr.gr_context) - gr.gr_context = ctx - self.assertIs(gr.gr_context, ctx) - - # Otherwise we leak greenlets on some platforms. - # XXX: Should be able to do this automatically - del holder[:] - gr = None - thread = None - - def test_context_assignment_wrong_type(self): - g = greenlet() - with self.assertRaisesRegex(TypeError, - "greenlet context must be a contextvars.Context or None"): - g.gr_context = self - - -@skipIf(Context is not None, "ContextVar supported") -class NoContextVarsTests(TestCase): - def test_contextvars_errors(self): - let1 = greenlet(getcurrent().switch) - self.assertFalse(hasattr(let1, 'gr_context')) - with self.assertRaises(AttributeError): - getattr(let1, 'gr_context') - - with self.assertRaises(AttributeError): - let1.gr_context = None - - let1.switch() - - with self.assertRaises(AttributeError): - getattr(let1, 'gr_context') - - with self.assertRaises(AttributeError): - let1.gr_context = None - - del let1 - - -if __name__ == '__main__': - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_cpp.py b/.env/Lib/site-packages/greenlet/tests/test_cpp.py deleted file mode 100644 index 2d0cc9c..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_cpp.py +++ /dev/null @@ -1,73 +0,0 @@ -from __future__ import print_function -from __future__ import absolute_import - -import subprocess -import unittest - -import greenlet -from . import _test_extension_cpp -from . import TestCase -from . import WIN - -class CPPTests(TestCase): - def test_exception_switch(self): - greenlets = [] - for i in range(4): - g = greenlet.greenlet(_test_extension_cpp.test_exception_switch) - g.switch(i) - greenlets.append(g) - for i, g in enumerate(greenlets): - self.assertEqual(g.switch(), i) - - def _do_test_unhandled_exception(self, target): - import os - import sys - script = os.path.join( - os.path.dirname(__file__), - 'fail_cpp_exception.py', - ) - args = [sys.executable, script, target.__name__ if not isinstance(target, str) else target] - __traceback_info__ = args - with self.assertRaises(subprocess.CalledProcessError) as exc: - subprocess.check_output( - args, - encoding='utf-8', - stderr=subprocess.STDOUT - ) - - ex = exc.exception - expected_exit = self.get_expected_returncodes_for_aborted_process() - self.assertIn(ex.returncode, expected_exit) - self.assertIn('fail_cpp_exception is running', ex.output) - return ex.output - - - def test_unhandled_nonstd_exception_aborts(self): - # verify that plain unhandled throw aborts - self._do_test_unhandled_exception(_test_extension_cpp.test_exception_throw_nonstd) - - def test_unhandled_std_exception_aborts(self): - # verify that plain unhandled throw aborts - self._do_test_unhandled_exception(_test_extension_cpp.test_exception_throw_std) - - @unittest.skipIf(WIN, "XXX: This does not crash on Windows") - # Meaning the exception is getting lost somewhere... - def test_unhandled_std_exception_as_greenlet_function_aborts(self): - # verify that plain unhandled throw aborts - output = self._do_test_unhandled_exception('run_as_greenlet_target') - self.assertIn( - # We really expect this to be prefixed with "greenlet: Unhandled C++ exception:" - # as added by our handler for std::exception (see TUserGreenlet.cpp), but - # that's not correct everywhere --- our handler never runs before std::terminate - # gets called (for example, on arm32). - 'Thrown from an extension.', - output - ) - - def test_unhandled_exception_in_greenlet_aborts(self): - # verify that unhandled throw called in greenlet aborts too - self._do_test_unhandled_exception('run_unhandled_exception_in_greenlet_aborts') - - -if __name__ == '__main__': - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_extension_interface.py b/.env/Lib/site-packages/greenlet/tests/test_extension_interface.py deleted file mode 100644 index 34b6656..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_extension_interface.py +++ /dev/null @@ -1,115 +0,0 @@ -from __future__ import print_function -from __future__ import absolute_import - -import sys - -import greenlet -from . import _test_extension -from . import TestCase - -# pylint:disable=c-extension-no-member - -class CAPITests(TestCase): - def test_switch(self): - self.assertEqual( - 50, _test_extension.test_switch(greenlet.greenlet(lambda: 50))) - - def test_switch_kwargs(self): - def adder(x, y): - return x * y - g = greenlet.greenlet(adder) - self.assertEqual(6, _test_extension.test_switch_kwargs(g, x=3, y=2)) - - def test_setparent(self): - # pylint:disable=disallowed-name - def foo(): - def bar(): - greenlet.getcurrent().parent.switch() - - # This final switch should go back to the main greenlet, since - # the test_setparent() function in the C extension should have - # reparented this greenlet. - greenlet.getcurrent().parent.switch() - raise AssertionError("Should never have reached this code") - child = greenlet.greenlet(bar) - child.switch() - greenlet.getcurrent().parent.switch(child) - greenlet.getcurrent().parent.throw( - AssertionError("Should never reach this code")) - foo_child = greenlet.greenlet(foo).switch() - self.assertEqual(None, _test_extension.test_setparent(foo_child)) - - def test_getcurrent(self): - _test_extension.test_getcurrent() - - def test_new_greenlet(self): - self.assertEqual(-15, _test_extension.test_new_greenlet(lambda: -15)) - - def test_raise_greenlet_dead(self): - self.assertRaises( - greenlet.GreenletExit, _test_extension.test_raise_dead_greenlet) - - def test_raise_greenlet_error(self): - self.assertRaises( - greenlet.error, _test_extension.test_raise_greenlet_error) - - def test_throw(self): - seen = [] - - def foo(): # pylint:disable=disallowed-name - try: - greenlet.getcurrent().parent.switch() - except ValueError: - seen.append(sys.exc_info()[1]) - except greenlet.GreenletExit: - raise AssertionError - g = greenlet.greenlet(foo) - g.switch() - _test_extension.test_throw(g) - self.assertEqual(len(seen), 1) - self.assertTrue( - isinstance(seen[0], ValueError), - "ValueError was not raised in foo()") - self.assertEqual( - str(seen[0]), - 'take that sucka!', - "message doesn't match") - - def test_non_traceback_param(self): - with self.assertRaises(TypeError) as exc: - _test_extension.test_throw_exact( - greenlet.getcurrent(), - Exception, - Exception(), - self - ) - self.assertEqual(str(exc.exception), - "throw() third argument must be a traceback object") - - def test_instance_of_wrong_type(self): - with self.assertRaises(TypeError) as exc: - _test_extension.test_throw_exact( - greenlet.getcurrent(), - Exception(), - BaseException(), - None, - ) - - self.assertEqual(str(exc.exception), - "instance exception may not have a separate value") - - def test_not_throwable(self): - with self.assertRaises(TypeError) as exc: - _test_extension.test_throw_exact( - greenlet.getcurrent(), - "abc", - None, - None, - ) - self.assertEqual(str(exc.exception), - "exceptions must be classes, or instances, not str") - - -if __name__ == '__main__': - import unittest - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_gc.py b/.env/Lib/site-packages/greenlet/tests/test_gc.py deleted file mode 100644 index 898452f..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_gc.py +++ /dev/null @@ -1,86 +0,0 @@ -import gc - -import weakref - -import greenlet - - -from . import TestCase -from .leakcheck import fails_leakcheck -# These only work with greenlet gc support -# which is no longer optional. -assert greenlet.GREENLET_USE_GC - -class TestGC(TestCase): - def test_dead_circular_ref(self): - o = weakref.ref(greenlet.greenlet(greenlet.getcurrent).switch()) - gc.collect() - if o() is not None: - import sys - print("O IS NOT NONE.", sys.getrefcount(o())) - self.assertIsNone(o()) - self.assertFalse(gc.garbage, gc.garbage) - - def test_circular_greenlet(self): - class circular_greenlet(greenlet.greenlet): - self = None - o = circular_greenlet() - o.self = o - o = weakref.ref(o) - gc.collect() - self.assertIsNone(o()) - self.assertFalse(gc.garbage, gc.garbage) - - def test_inactive_ref(self): - class inactive_greenlet(greenlet.greenlet): - def __init__(self): - greenlet.greenlet.__init__(self, run=self.run) - - def run(self): - pass - o = inactive_greenlet() - o = weakref.ref(o) - gc.collect() - self.assertIsNone(o()) - self.assertFalse(gc.garbage, gc.garbage) - - @fails_leakcheck - def test_finalizer_crash(self): - # This test is designed to crash when active greenlets - # are made garbage collectable, until the underlying - # problem is resolved. How does it work: - # - order of object creation is important - # - array is created first, so it is moved to unreachable first - # - we create a cycle between a greenlet and this array - # - we create an object that participates in gc, is only - # referenced by a greenlet, and would corrupt gc lists - # on destruction, the easiest is to use an object with - # a finalizer - # - because array is the first object in unreachable it is - # cleared first, which causes all references to greenlet - # to disappear and causes greenlet to be destroyed, but since - # it is still live it causes a switch during gc, which causes - # an object with finalizer to be destroyed, which causes stack - # corruption and then a crash - - class object_with_finalizer(object): - def __del__(self): - pass - array = [] - parent = greenlet.getcurrent() - def greenlet_body(): - greenlet.getcurrent().object = object_with_finalizer() - try: - parent.switch() - except greenlet.GreenletExit: - print("Got greenlet exit!") - finally: - del greenlet.getcurrent().object - g = greenlet.greenlet(greenlet_body) - g.array = array - array.append(g) - g.switch() - del array - del g - greenlet.getcurrent() - gc.collect() diff --git a/.env/Lib/site-packages/greenlet/tests/test_generator.py b/.env/Lib/site-packages/greenlet/tests/test_generator.py deleted file mode 100644 index ca4a644..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_generator.py +++ /dev/null @@ -1,59 +0,0 @@ - -from greenlet import greenlet - -from . import TestCase - -class genlet(greenlet): - parent = None - def __init__(self, *args, **kwds): - self.args = args - self.kwds = kwds - - def run(self): - fn, = self.fn - fn(*self.args, **self.kwds) - - def __iter__(self): - return self - - def __next__(self): - self.parent = greenlet.getcurrent() - result = self.switch() - if self: - return result - - raise StopIteration - - next = __next__ - - -def Yield(value): - g = greenlet.getcurrent() - while not isinstance(g, genlet): - if g is None: - raise RuntimeError('yield outside a genlet') - g = g.parent - g.parent.switch(value) - - -def generator(func): - class Generator(genlet): - fn = (func,) - return Generator - -# ____________________________________________________________ - - -class GeneratorTests(TestCase): - def test_generator(self): - seen = [] - - def g(n): - for i in range(n): - seen.append(i) - Yield(i) - g = generator(g) - for _ in range(3): - for j in g(5): - seen.append(j) - self.assertEqual(seen, 3 * [0, 0, 1, 1, 2, 2, 3, 3, 4, 4]) diff --git a/.env/Lib/site-packages/greenlet/tests/test_generator_nested.py b/.env/Lib/site-packages/greenlet/tests/test_generator_nested.py deleted file mode 100644 index 8d752a6..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_generator_nested.py +++ /dev/null @@ -1,168 +0,0 @@ - -from greenlet import greenlet -from . import TestCase -from .leakcheck import fails_leakcheck - -class genlet(greenlet): - parent = None - def __init__(self, *args, **kwds): - self.args = args - self.kwds = kwds - self.child = None - - def run(self): - # Note the function is packed in a tuple - # to avoid creating a bound method for it. - fn, = self.fn - fn(*self.args, **self.kwds) - - def __iter__(self): - return self - - def set_child(self, child): - self.child = child - - def __next__(self): - if self.child: - child = self.child - while child.child: - tmp = child - child = child.child - tmp.child = None - - result = child.switch() - else: - self.parent = greenlet.getcurrent() - result = self.switch() - - if self: - return result - - raise StopIteration - - next = __next__ - -def Yield(value, level=1): - g = greenlet.getcurrent() - - while level != 0: - if not isinstance(g, genlet): - raise RuntimeError('yield outside a genlet') - if level > 1: - g.parent.set_child(g) - g = g.parent - level -= 1 - - g.switch(value) - - -def Genlet(func): - class TheGenlet(genlet): - fn = (func,) - return TheGenlet - -# ____________________________________________________________ - - -def g1(n, seen): - for i in range(n): - seen.append(i + 1) - yield i - - -def g2(n, seen): - for i in range(n): - seen.append(i + 1) - Yield(i) - -g2 = Genlet(g2) - - -def nested(i): - Yield(i) - - -def g3(n, seen): - for i in range(n): - seen.append(i + 1) - nested(i) -g3 = Genlet(g3) - - -def a(n): - if n == 0: - return - for ii in ax(n - 1): - Yield(ii) - Yield(n) -ax = Genlet(a) - - -def perms(l): - if len(l) > 1: - for e in l: - # No syntactical sugar for generator expressions - x = [Yield([e] + p) for p in perms([x for x in l if x != e])] - assert x - else: - Yield(l) -perms = Genlet(perms) - - -def gr1(n): - for ii in range(1, n): - Yield(ii) - Yield(ii * ii, 2) - -gr1 = Genlet(gr1) - - -def gr2(n, seen): - for ii in gr1(n): - seen.append(ii) - -gr2 = Genlet(gr2) - - -class NestedGeneratorTests(TestCase): - def test_layered_genlets(self): - seen = [] - for ii in gr2(5, seen): - seen.append(ii) - self.assertEqual(seen, [1, 1, 2, 4, 3, 9, 4, 16]) - - @fails_leakcheck - def test_permutations(self): - gen_perms = perms(list(range(4))) - permutations = list(gen_perms) - self.assertEqual(len(permutations), 4 * 3 * 2 * 1) - self.assertIn([0, 1, 2, 3], permutations) - self.assertIn([3, 2, 1, 0], permutations) - res = [] - for ii in zip(perms(list(range(4))), perms(list(range(3)))): - res.append(ii) - self.assertEqual( - res, - [([0, 1, 2, 3], [0, 1, 2]), ([0, 1, 3, 2], [0, 2, 1]), - ([0, 2, 1, 3], [1, 0, 2]), ([0, 2, 3, 1], [1, 2, 0]), - ([0, 3, 1, 2], [2, 0, 1]), ([0, 3, 2, 1], [2, 1, 0])]) - # XXX Test to make sure we are working as a generator expression - - def test_genlet_simple(self): - for g in g1, g2, g3: - seen = [] - for _ in range(3): - for j in g(5, seen): - seen.append(j) - self.assertEqual(seen, 3 * [1, 0, 2, 1, 3, 2, 4, 3, 5, 4]) - - def test_genlet_bad(self): - try: - Yield(10) - except RuntimeError: - pass - - def test_nested_genlets(self): - seen = [] - for ii in ax(5): - seen.append(ii) diff --git a/.env/Lib/site-packages/greenlet/tests/test_greenlet.py b/.env/Lib/site-packages/greenlet/tests/test_greenlet.py deleted file mode 100644 index 5a8139b..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_greenlet.py +++ /dev/null @@ -1,1365 +0,0 @@ -import gc -import sys -import time -import threading -import unittest - -from abc import ABCMeta -from abc import abstractmethod - -import greenlet -from greenlet import greenlet as RawGreenlet -from . import TestCase -from . import RUNNING_ON_MANYLINUX -from . import PY313 -from . import PY314 -from . import RUNNING_ON_FREETHREAD_BUILD -from .leakcheck import fails_leakcheck - - -# We manually manage locks in many tests -# pylint:disable=consider-using-with -# pylint:disable=too-many-public-methods -# This module is quite large. -# TODO: Refactor into separate test files. For example, -# put all the regression tests that used to produce -# crashes in test_greenlet_no_crash; put tests that DO deliberately crash -# the interpreter into test_greenlet_crash. -# pylint:disable=too-many-lines - -class SomeError(Exception): - pass - - -def fmain(seen): - try: - greenlet.getcurrent().parent.switch() - except: - seen.append(sys.exc_info()[0]) - raise - raise SomeError - - -def send_exception(g, exc): - # note: send_exception(g, exc) can be now done with g.throw(exc). - # the purpose of this test is to explicitly check the propagation rules. - def crasher(exc): - raise exc - g1 = RawGreenlet(crasher, parent=g) - g1.switch(exc) - - -class TestGreenlet(TestCase): - - def _do_simple_test(self): - lst = [] - - def f(): - lst.append(1) - greenlet.getcurrent().parent.switch() - lst.append(3) - g = RawGreenlet(f) - lst.append(0) - g.switch() - lst.append(2) - g.switch() - lst.append(4) - self.assertEqual(lst, list(range(5))) - - def test_simple(self): - self._do_simple_test() - - def test_switch_no_run_raises_AttributeError(self): - g = RawGreenlet() - with self.assertRaises(AttributeError) as exc: - g.switch() - - self.assertIn("run", str(exc.exception)) - - def test_throw_no_run_raises_AttributeError(self): - g = RawGreenlet() - with self.assertRaises(AttributeError) as exc: - g.throw(SomeError) - - self.assertIn("run", str(exc.exception)) - - def test_parent_equals_None(self): - g = RawGreenlet(parent=None) - self.assertIsNotNone(g) - self.assertIs(g.parent, greenlet.getcurrent()) - - def test_run_equals_None(self): - g = RawGreenlet(run=None) - self.assertIsNotNone(g) - self.assertIsNone(g.run) - - def test_two_children(self): - lst = [] - - def f(): - lst.append(1) - greenlet.getcurrent().parent.switch() - lst.extend([1, 1]) - g = RawGreenlet(f) - h = RawGreenlet(f) - g.switch() - self.assertEqual(len(lst), 1) - h.switch() - self.assertEqual(len(lst), 2) - h.switch() - self.assertEqual(len(lst), 4) - self.assertEqual(h.dead, True) - g.switch() - self.assertEqual(len(lst), 6) - self.assertEqual(g.dead, True) - - def test_two_recursive_children(self): - lst = [] - - def f(): - lst.append('b') - greenlet.getcurrent().parent.switch() - - def g(): - lst.append('a') - g = RawGreenlet(f) - g.switch() - lst.append('c') - self.assertEqual(sys.getrefcount(g), 2 if not PY314 else 1) - g = RawGreenlet(g) - # Python 3.14 elides reference counting operations - # in some cases. See https://github.com/python/cpython/pull/130708 - self.assertEqual(sys.getrefcount(g), 2 if not PY314 else 1) - g.switch() - self.assertEqual(lst, ['a', 'b', 'c']) - # Just the one in this frame, plus the one on the stack we pass to the function - self.assertEqual(sys.getrefcount(g), 2 if not PY314 else 1) - - def test_threads(self): - success = [] - - def f(): - self._do_simple_test() - success.append(True) - ths = [threading.Thread(target=f) for i in range(10)] - for th in ths: - th.start() - for th in ths: - th.join(10) - self.assertEqual(len(success), len(ths)) - - def test_exception(self): - seen = [] - g1 = RawGreenlet(fmain) - g2 = RawGreenlet(fmain) - g1.switch(seen) - g2.switch(seen) - g2.parent = g1 - - self.assertEqual(seen, []) - #with self.assertRaises(SomeError): - # p("***Switching back") - # g2.switch() - # Creating this as a bound method can reveal bugs that - # are hidden on newer versions of Python that avoid creating - # bound methods for direct expressions; IOW, don't use the `with` - # form! - self.assertRaises(SomeError, g2.switch) - self.assertEqual(seen, [SomeError]) - - value = g2.switch() - self.assertEqual(value, ()) - self.assertEqual(seen, [SomeError]) - - value = g2.switch(25) - self.assertEqual(value, 25) - self.assertEqual(seen, [SomeError]) - - - def test_send_exception(self): - seen = [] - g1 = RawGreenlet(fmain) - g1.switch(seen) - self.assertRaises(KeyError, send_exception, g1, KeyError) - self.assertEqual(seen, [KeyError]) - - def test_dealloc(self): - seen = [] - g1 = RawGreenlet(fmain) - g2 = RawGreenlet(fmain) - g1.switch(seen) - g2.switch(seen) - self.assertEqual(seen, []) - del g1 - gc.collect() - self.assertEqual(seen, [greenlet.GreenletExit]) - del g2 - gc.collect() - self.assertEqual(seen, [greenlet.GreenletExit, greenlet.GreenletExit]) - - def test_dealloc_catches_GreenletExit_throws_other(self): - def run(): - try: - greenlet.getcurrent().parent.switch() - except greenlet.GreenletExit: - raise SomeError from None - - g = RawGreenlet(run) - g.switch() - # Destroying the only reference to the greenlet causes it - # to get GreenletExit; when it in turn raises, even though we're the parent - # we don't get the exception, it just gets printed. - # When we run on 3.8 only, we can use sys.unraisablehook - oldstderr = sys.stderr - from io import StringIO - stderr = sys.stderr = StringIO() - try: - del g - finally: - sys.stderr = oldstderr - - v = stderr.getvalue() - self.assertIn("Exception", v) - self.assertIn('ignored', v) - self.assertIn("SomeError", v) - - - @unittest.skipIf( - PY313 and RUNNING_ON_MANYLINUX, - "Sometimes flaky (getting one GreenletExit in the second list)" - # Probably due to funky timing interactions? - # TODO: FIXME Make that work. - ) - - def test_dealloc_other_thread(self): - seen = [] - someref = [] - - bg_glet_created_running_and_no_longer_ref_in_bg = threading.Event() - fg_ref_released = threading.Event() - bg_should_be_clear = threading.Event() - ok_to_exit_bg_thread = threading.Event() - - def f(): - g1 = RawGreenlet(fmain) - g1.switch(seen) - someref.append(g1) - del g1 - gc.collect() - bg_glet_created_running_and_no_longer_ref_in_bg.set() - fg_ref_released.wait(3) - - RawGreenlet() # trigger release - bg_should_be_clear.set() - ok_to_exit_bg_thread.wait(3) - RawGreenlet() # One more time - - t = threading.Thread(target=f) - t.start() - bg_glet_created_running_and_no_longer_ref_in_bg.wait(10) - - self.assertEqual(seen, []) - self.assertEqual(len(someref), 1) - del someref[:] - if not RUNNING_ON_FREETHREAD_BUILD: - # The free-threaded GC is very different. In 3.14rc1, - # the free-threaded GC traverses ``g1``, realizes it is - # not referenced from anywhere else IT cares about, - # calls ``tp_clear`` and then ``green_dealloc``. This causes - # the greenlet to lose its reference to the main greenlet and thread - # in which it was running, which means we can no longer throw an - # exception into it, preventing the rest of this test from working. - # Standard 3.14 traverses the object but doesn't ``tp_clear`` or - # ``green_dealloc`` it. - gc.collect() - # g1 is not released immediately because it's from another thread; - # switching back to that thread will allocate a greenlet and thus - # trigger deletion actions. - self.assertEqual(seen, []) - fg_ref_released.set() - bg_should_be_clear.wait(3) - try: - self.assertEqual(seen, [greenlet.GreenletExit]) - finally: - ok_to_exit_bg_thread.set() - t.join(10) - del seen[:] - del someref[:] - - def test_frame(self): - def f1(): - f = sys._getframe(0) # pylint:disable=protected-access - self.assertEqual(f.f_back, None) - greenlet.getcurrent().parent.switch(f) - return "meaning of life" - g = RawGreenlet(f1) - frame = g.switch() - self.assertTrue(frame is g.gr_frame) - self.assertTrue(g) - - from_g = g.switch() - self.assertFalse(g) - self.assertEqual(from_g, 'meaning of life') - self.assertEqual(g.gr_frame, None) - - def test_thread_bug(self): - def runner(x): - g = RawGreenlet(lambda: time.sleep(x)) - g.switch() - t1 = threading.Thread(target=runner, args=(0.2,)) - t2 = threading.Thread(target=runner, args=(0.3,)) - t1.start() - t2.start() - t1.join(10) - t2.join(10) - - def test_switch_kwargs(self): - def run(a, b): - self.assertEqual(a, 4) - self.assertEqual(b, 2) - return 42 - x = RawGreenlet(run).switch(a=4, b=2) - self.assertEqual(x, 42) - - def test_switch_kwargs_to_parent(self): - def run(x): - greenlet.getcurrent().parent.switch(x=x) - greenlet.getcurrent().parent.switch(2, x=3) - return x, x ** 2 - g = RawGreenlet(run) - self.assertEqual({'x': 3}, g.switch(3)) - self.assertEqual(((2,), {'x': 3}), g.switch()) - self.assertEqual((3, 9), g.switch()) - - def test_switch_to_another_thread(self): - data = {} - created_event = threading.Event() - done_event = threading.Event() - - def run(): - data['g'] = RawGreenlet(lambda: None) - created_event.set() - done_event.wait(10) - thread = threading.Thread(target=run) - thread.start() - created_event.wait(10) - with self.assertRaises(greenlet.error): - data['g'].switch() - done_event.set() - thread.join(10) - # XXX: Should handle this automatically - data.clear() - - def test_exc_state(self): - def f(): - try: - raise ValueError('fun') - except: # pylint:disable=bare-except - exc_info = sys.exc_info() - RawGreenlet(h).switch() - self.assertEqual(exc_info, sys.exc_info()) - - def h(): - self.assertEqual(sys.exc_info(), (None, None, None)) - - RawGreenlet(f).switch() - - def test_instance_dict(self): - def f(): - greenlet.getcurrent().test = 42 - def deldict(g): - del g.__dict__ - def setdict(g, value): - g.__dict__ = value - g = RawGreenlet(f) - self.assertEqual(g.__dict__, {}) - g.switch() - self.assertEqual(g.test, 42) - self.assertEqual(g.__dict__, {'test': 42}) - g.__dict__ = g.__dict__ - self.assertEqual(g.__dict__, {'test': 42}) - self.assertRaises(TypeError, deldict, g) - self.assertRaises(TypeError, setdict, g, 42) - - def test_running_greenlet_has_no_run(self): - has_run = [] - def func(): - has_run.append( - hasattr(greenlet.getcurrent(), 'run') - ) - - g = RawGreenlet(func) - g.switch() - self.assertEqual(has_run, [False]) - - def test_deepcopy(self): - import copy - self.assertRaises(TypeError, copy.copy, RawGreenlet()) - self.assertRaises(TypeError, copy.deepcopy, RawGreenlet()) - - def test_parent_restored_on_kill(self): - hub = RawGreenlet(lambda: None) - main = greenlet.getcurrent() - result = [] - def worker(): - try: - # Wait to be killed by going back to the test. - main.switch() - except greenlet.GreenletExit: - # Resurrect and switch to parent - result.append(greenlet.getcurrent().parent) - result.append(greenlet.getcurrent()) - hub.switch() - g = RawGreenlet(worker, parent=hub) - g.switch() - # delete the only reference, thereby raising GreenletExit - del g - self.assertTrue(result) - self.assertIs(result[0], main) - self.assertIs(result[1].parent, hub) - # Delete them, thereby breaking the cycle between the greenlet - # and the frame, which otherwise would never be collectable - # XXX: We should be able to automatically fix this. - del result[:] - hub = None - main = None - - def test_parent_return_failure(self): - # No run causes AttributeError on switch - g1 = RawGreenlet() - # Greenlet that implicitly switches to parent - g2 = RawGreenlet(lambda: None, parent=g1) - # AttributeError should propagate to us, no fatal errors - with self.assertRaises(AttributeError): - g2.switch() - - def test_throw_exception_not_lost(self): - class mygreenlet(RawGreenlet): - def __getattribute__(self, name): - try: - raise Exception # pylint:disable=broad-exception-raised - except: # pylint:disable=bare-except - pass - return RawGreenlet.__getattribute__(self, name) - g = mygreenlet(lambda: None) - self.assertRaises(SomeError, g.throw, SomeError()) - - @fails_leakcheck - def _do_test_throw_to_dead_thread_doesnt_crash(self, wait_for_cleanup=False): - result = [] - def worker(): - greenlet.getcurrent().parent.switch() - - def creator(): - g = RawGreenlet(worker) - g.switch() - result.append(g) - if wait_for_cleanup: - # Let this greenlet eventually be cleaned up. - g.switch() - greenlet.getcurrent() - t = threading.Thread(target=creator) - t.start() - t.join(10) - del t - # But, depending on the operating system, the thread - # deallocator may not actually have run yet! So we can't be - # sure about the error message unless we wait. - if wait_for_cleanup: - self.wait_for_pending_cleanups() - with self.assertRaises(greenlet.error) as exc: - result[0].throw(SomeError) - - if not wait_for_cleanup: - s = str(exc.exception) - self.assertTrue( - s == "cannot switch to a different thread (which happens to have exited)" - or 'Cannot switch' in s - ) - else: - self.assertEqual( - str(exc.exception), - "cannot switch to a different thread (which happens to have exited)", - ) - - if hasattr(result[0].gr_frame, 'clear'): - # The frame is actually executing (it thinks), we can't clear it. - with self.assertRaises(RuntimeError): - result[0].gr_frame.clear() - # Unfortunately, this doesn't actually clear the references, they're in the - # fast local array. - if not wait_for_cleanup: - # f_locals has no clear method in Python 3.13 - if hasattr(result[0].gr_frame.f_locals, 'clear'): - result[0].gr_frame.f_locals.clear() - else: - self.assertIsNone(result[0].gr_frame) - - del creator - worker = None - del result[:] - # XXX: we ought to be able to automatically fix this. - # See issue 252 - self.expect_greenlet_leak = True # direct us not to wait for it to go away - - @fails_leakcheck - def test_throw_to_dead_thread_doesnt_crash(self): - self._do_test_throw_to_dead_thread_doesnt_crash() - - def test_throw_to_dead_thread_doesnt_crash_wait(self): - self._do_test_throw_to_dead_thread_doesnt_crash(True) - - @fails_leakcheck - def test_recursive_startup(self): - class convoluted(RawGreenlet): - def __init__(self): - RawGreenlet.__init__(self) - self.count = 0 - def __getattribute__(self, name): - if name == 'run' and self.count == 0: - self.count = 1 - self.switch(43) - return RawGreenlet.__getattribute__(self, name) - def run(self, value): - while True: - self.parent.switch(value) - g = convoluted() - self.assertEqual(g.switch(42), 43) - # Exits the running greenlet, otherwise it leaks - # XXX: We should be able to automatically fix this - #g.throw(greenlet.GreenletExit) - #del g - self.expect_greenlet_leak = True - - def test_threaded_updatecurrent(self): - # released when main thread should execute - lock1 = threading.Lock() - lock1.acquire() - # released when another thread should execute - lock2 = threading.Lock() - lock2.acquire() - class finalized(object): - def __del__(self): - # happens while in green_updatecurrent() in main greenlet - # should be very careful not to accidentally call it again - # at the same time we must make sure another thread executes - lock2.release() - lock1.acquire() - # now ts_current belongs to another thread - def deallocator(): - greenlet.getcurrent().parent.switch() - def fthread(): - lock2.acquire() - greenlet.getcurrent() - del g[0] - lock1.release() - lock2.acquire() - greenlet.getcurrent() - lock1.release() - main = greenlet.getcurrent() - g = [RawGreenlet(deallocator)] - g[0].bomb = finalized() - g[0].switch() - t = threading.Thread(target=fthread) - t.start() - # let another thread grab ts_current and deallocate g[0] - lock2.release() - lock1.acquire() - # this is the corner stone - # getcurrent() will notice that ts_current belongs to another thread - # and start the update process, which would notice that g[0] should - # be deallocated, and that will execute an object's finalizer. Now, - # that object will let another thread run so it can grab ts_current - # again, which would likely crash the interpreter if there's no - # check for this case at the end of green_updatecurrent(). This test - # passes if getcurrent() returns correct result, but it's likely - # to randomly crash if it's not anyway. - self.assertEqual(greenlet.getcurrent(), main) - # wait for another thread to complete, just in case - t.join(10) - - def test_dealloc_switch_args_not_lost(self): - seen = [] - def worker(): - # wait for the value - value = greenlet.getcurrent().parent.switch() - # delete all references to ourself - del worker[0] - initiator.parent = greenlet.getcurrent().parent - # switch to main with the value, but because - # ts_current is the last reference to us we - # return here immediately, where we resurrect ourself. - try: - greenlet.getcurrent().parent.switch(value) - finally: - seen.append(greenlet.getcurrent()) - def initiator(): - return 42 # implicitly falls thru to parent - - worker = [RawGreenlet(worker)] - - worker[0].switch() # prime worker - initiator = RawGreenlet(initiator, worker[0]) - value = initiator.switch() - self.assertTrue(seen) - self.assertEqual(value, 42) - - def test_tuple_subclass(self): - # The point of this test is to see what happens when a custom - # tuple subclass is used as an object passed directly to the C - # function ``green_switch``; part of ``green_switch`` checks - # the ``len()`` of the ``args`` tuple, and that can call back - # into Python. Here, when it calls back into Python, we - # recursively enter ``green_switch`` again. - - # This test is really only relevant on Python 2. The builtin - # `apply` function directly passes the given args tuple object - # to the underlying function, whereas the Python 3 version - # unpacks and repacks into an actual tuple. This could still - # happen using the C API on Python 3 though. We should write a - # builtin version of apply() ourself. - def _apply(func, a, k): - func(*a, **k) - - class mytuple(tuple): - def __len__(self): - greenlet.getcurrent().switch() - return tuple.__len__(self) - args = mytuple() - kwargs = dict(a=42) - def switchapply(): - _apply(greenlet.getcurrent().parent.switch, args, kwargs) - g = RawGreenlet(switchapply) - self.assertEqual(g.switch(), kwargs) - - def test_abstract_subclasses(self): - AbstractSubclass = ABCMeta( - 'AbstractSubclass', - (RawGreenlet,), - {'run': abstractmethod(lambda self: None)}) - - class BadSubclass(AbstractSubclass): - pass - - class GoodSubclass(AbstractSubclass): - def run(self): - pass - - GoodSubclass() # should not raise - self.assertRaises(TypeError, BadSubclass) - - def test_implicit_parent_with_threads(self): - if not gc.isenabled(): - return # cannot test with disabled gc - N = gc.get_threshold()[0] - if N < 50: - return # cannot test with such a small N - def attempt(): - lock1 = threading.Lock() - lock1.acquire() - lock2 = threading.Lock() - lock2.acquire() - recycled = [False] - def another_thread(): - lock1.acquire() # wait for gc - greenlet.getcurrent() # update ts_current - lock2.release() # release gc - t = threading.Thread(target=another_thread) - t.start() - class gc_callback(object): - def __del__(self): - lock1.release() - lock2.acquire() - recycled[0] = True - class garbage(object): - def __init__(self): - self.cycle = self - self.callback = gc_callback() - l = [] - x = range(N*2) - current = greenlet.getcurrent() - g = garbage() - for _ in x: - g = None # lose reference to garbage - if recycled[0]: - # gc callback called prematurely - t.join(10) - return False - last = RawGreenlet() - if recycled[0]: - break # yes! gc called in green_new - l.append(last) # increase allocation counter - else: - # gc callback not called when expected - gc.collect() - if recycled[0]: - t.join(10) - return False - self.assertEqual(last.parent, current) - for g in l: - self.assertEqual(g.parent, current) - return True - for _ in range(5): - if attempt(): - break - - def test_issue_245_reference_counting_subclass_no_threads(self): - # https://github.com/python-greenlet/greenlet/issues/245 - # Before the fix, this crashed pretty reliably on - # Python 3.10, at least on macOS; but much less reliably on other - # interpreters (memory layout must have changed). - # The threaded test crashed more reliably on more interpreters. - from greenlet import getcurrent - from greenlet import GreenletExit - - class Greenlet(RawGreenlet): - pass - - initial_refs = sys.getrefcount(Greenlet) - # This has to be an instance variable because - # Python 2 raises a SyntaxError if we delete a local - # variable referenced in an inner scope. - self.glets = [] # pylint:disable=attribute-defined-outside-init - - def greenlet_main(): - try: - getcurrent().parent.switch() - except GreenletExit: - self.glets.append(getcurrent()) - - # Before the - for _ in range(10): - Greenlet(greenlet_main).switch() - - del self.glets - if RUNNING_ON_FREETHREAD_BUILD: - # Free-threaded builds make types immortal, which gives us - # weird numbers here, and we actually do APPEAR to end - # up with one more reference than we started with, at least on 3.14. - # If we change the code in green_dealloc to avoid increffing the type - # (which fixed this initial bug), then our leakchecks find other objects - # that have leaked, including a tuple, a dict, and a type. So that's not the - # right solution. Instead we change the test: - # XXX: FIXME: Is there a better way? - self.assertGreaterEqual(sys.getrefcount(Greenlet), initial_refs) - else: - self.assertEqual(sys.getrefcount(Greenlet), initial_refs) - - @unittest.skipIf( - PY313 and RUNNING_ON_MANYLINUX, - "The manylinux images appear to hang on this test on 3.13rc2" - # Or perhaps I just got tired of waiting for the 450s timeout. - # Still, it shouldn't take anywhere near that long. Does not reproduce in - # Ubuntu images, on macOS or Windows. - ) - def test_issue_245_reference_counting_subclass_threads(self): - # https://github.com/python-greenlet/greenlet/issues/245 - from threading import Thread - from threading import Event - - from greenlet import getcurrent - - class MyGreenlet(RawGreenlet): - pass - - glets = [] - ref_cleared = Event() - - def greenlet_main(): - getcurrent().parent.switch() - - def thread_main(greenlet_running_event): - mine = MyGreenlet(greenlet_main) - glets.append(mine) - # The greenlets being deleted must be active - mine.switch() - # Don't keep any reference to it in this thread - del mine - # Let main know we published our greenlet. - greenlet_running_event.set() - # Wait for main to let us know the references are - # gone and the greenlet objects no longer reachable - ref_cleared.wait(10) - # The creating thread must call getcurrent() (or a few other - # greenlet APIs) because that's when the thread-local list of dead - # greenlets gets cleared. - getcurrent() - - # We start with 3 references to the subclass: - # - This module - # - Its __mro__ - # - The __subclassess__ attribute of greenlet - # - (If we call gc.get_referents(), we find four entries, including - # some other tuple ``(greenlet)`` that I'm not sure about but must be part - # of the machinery.) - # - # On Python 3.10 it's often enough to just run 3 threads; on Python 2.7, - # more threads are needed, and the results are still - # non-deterministic. Presumably the memory layouts are different - initial_refs = sys.getrefcount(MyGreenlet) - thread_ready_events = [] - thread_count = initial_refs + 45 - if RUNNING_ON_FREETHREAD_BUILD: - # types are immortal, so this is a HUGE number most likely, - # and we can't create that many threads. - thread_count = 50 - for _ in range(thread_count): - event = Event() - thread = Thread(target=thread_main, args=(event,)) - thread_ready_events.append(event) - thread.start() - - - for done_event in thread_ready_events: - done_event.wait(10) - - - del glets[:] - ref_cleared.set() - # Let any other thread run; it will crash the interpreter - # if not fixed (or silently corrupt memory and we possibly crash - # later). - self.wait_for_pending_cleanups() - self.assertEqual(sys.getrefcount(MyGreenlet), initial_refs) - - def test_falling_off_end_switches_to_unstarted_parent_raises_error(self): - def no_args(): - return 13 - - parent_never_started = RawGreenlet(no_args) - - def leaf(): - return 42 - - child = RawGreenlet(leaf, parent_never_started) - - # Because the run function takes to arguments - with self.assertRaises(TypeError): - child.switch() - - def test_falling_off_end_switches_to_unstarted_parent_works(self): - def one_arg(x): - return (x, 24) - - parent_never_started = RawGreenlet(one_arg) - - def leaf(): - return 42 - - child = RawGreenlet(leaf, parent_never_started) - - result = child.switch() - self.assertEqual(result, (42, 24)) - - def test_switch_to_dead_greenlet_with_unstarted_perverse_parent(self): - class Parent(RawGreenlet): - def __getattribute__(self, name): - if name == 'run': - raise SomeError - - - parent_never_started = Parent() - seen = [] - child = RawGreenlet(lambda: seen.append(42), parent_never_started) - # Because we automatically start the parent when the child is - # finished - with self.assertRaises(SomeError): - child.switch() - - self.assertEqual(seen, [42]) - - with self.assertRaises(SomeError): - child.switch() - self.assertEqual(seen, [42]) - - def test_switch_to_dead_greenlet_reparent(self): - seen = [] - parent_never_started = RawGreenlet(lambda: seen.append(24)) - child = RawGreenlet(lambda: seen.append(42)) - - child.switch() - self.assertEqual(seen, [42]) - - child.parent = parent_never_started - # This actually is the same as switching to the parent. - result = child.switch() - self.assertIsNone(result) - self.assertEqual(seen, [42, 24]) - - def test_can_access_f_back_of_suspended_greenlet(self): - # This tests our frame rewriting to work around Python 3.12+ having - # some interpreter frames on the C stack. It will crash in the absence - # of that logic. - main = greenlet.getcurrent() - - def outer(): - inner() - - def inner(): - main.switch(sys._getframe(0)) - - hub = RawGreenlet(outer) - # start it - hub.switch() - - # start another greenlet to make sure we aren't relying on - # anything in `hub` still being on the C stack - unrelated = RawGreenlet(lambda: None) - unrelated.switch() - - # now it is suspended - self.assertIsNotNone(hub.gr_frame) - self.assertEqual(hub.gr_frame.f_code.co_name, "inner") - self.assertIsNotNone(hub.gr_frame.f_back) - self.assertEqual(hub.gr_frame.f_back.f_code.co_name, "outer") - # The next line is what would crash - self.assertIsNone(hub.gr_frame.f_back.f_back) - - def test_get_stack_with_nested_c_calls(self): - from functools import partial - from . import _test_extension_cpp - - def recurse(v): - if v > 0: - return v * _test_extension_cpp.test_call(partial(recurse, v - 1)) - return greenlet.getcurrent().parent.switch() - - gr = RawGreenlet(recurse) - gr.switch(5) - frame = gr.gr_frame - for i in range(5): - self.assertEqual(frame.f_locals["v"], i) - frame = frame.f_back - self.assertEqual(frame.f_locals["v"], 5) - self.assertIsNone(frame.f_back) - self.assertEqual(gr.switch(10), 1200) # 1200 = 5! * 10 - - def test_frames_always_exposed(self): - # On Python 3.12 this will crash if we don't set the - # gr_frames_always_exposed attribute. More background: - # https://github.com/python-greenlet/greenlet/issues/388 - main = greenlet.getcurrent() - - def outer(): - inner(sys._getframe(0)) - - def inner(frame): - main.switch(frame) - - gr = RawGreenlet(outer) - frame = gr.switch() - - # Do something else to clobber the part of the C stack used by `gr`, - # so we can't skate by on "it just happened to still be there" - unrelated = RawGreenlet(lambda: None) - unrelated.switch() - - self.assertEqual(frame.f_code.co_name, "outer") - # The next line crashes on 3.12 if we haven't exposed the frames. - self.assertIsNone(frame.f_back) - - -class TestGreenletSetParentErrors(TestCase): - def test_threaded_reparent(self): - data = {} - created_event = threading.Event() - done_event = threading.Event() - - def run(): - data['g'] = RawGreenlet(lambda: None) - created_event.set() - done_event.wait(10) - - def blank(): - greenlet.getcurrent().parent.switch() - - thread = threading.Thread(target=run) - thread.start() - created_event.wait(10) - g = RawGreenlet(blank) - g.switch() - with self.assertRaises(ValueError) as exc: - g.parent = data['g'] - done_event.set() - thread.join(10) - - self.assertEqual(str(exc.exception), "parent cannot be on a different thread") - - def test_unexpected_reparenting(self): - another = [] - def worker(): - g = RawGreenlet(lambda: None) - another.append(g) - g.switch() - t = threading.Thread(target=worker) - t.start() - t.join(10) - # The first time we switch (running g_initialstub(), which is - # when we look up the run attribute) we attempt to change the - # parent to one from another thread (which also happens to be - # dead). ``g_initialstub()`` should detect this and raise a - # greenlet error. - # - # EXCEPT: With the fix for #252, this is actually detected - # sooner, when setting the parent itself. Prior to that fix, - # the main greenlet from the background thread kept a valid - # value for ``run_info``, and appeared to be a valid parent - # until we actually started the greenlet. But now that it's - # cleared, this test is catching whether ``green_setparent`` - # can detect the dead thread. - # - # Further refactoring once again changes this back to a greenlet.error - # - # We need to wait for the cleanup to happen, but we're - # deliberately leaking a main greenlet here. - self.wait_for_pending_cleanups(initial_main_greenlets=self.main_greenlets_before_test + 1) - - class convoluted(RawGreenlet): - def __getattribute__(self, name): - if name == 'run': - self.parent = another[0] # pylint:disable=attribute-defined-outside-init - return RawGreenlet.__getattribute__(self, name) - g = convoluted(lambda: None) - with self.assertRaises(greenlet.error) as exc: - g.switch() - self.assertEqual(str(exc.exception), - "cannot switch to a different thread (which happens to have exited)") - del another[:] - - def test_unexpected_reparenting_thread_running(self): - # Like ``test_unexpected_reparenting``, except the background thread is - # actually still alive. - another = [] - switched_to_greenlet = threading.Event() - keep_main_alive = threading.Event() - def worker(): - g = RawGreenlet(lambda: None) - another.append(g) - g.switch() - switched_to_greenlet.set() - keep_main_alive.wait(10) - class convoluted(RawGreenlet): - def __getattribute__(self, name): - if name == 'run': - self.parent = another[0] # pylint:disable=attribute-defined-outside-init - return RawGreenlet.__getattribute__(self, name) - - t = threading.Thread(target=worker) - t.start() - - switched_to_greenlet.wait(10) - try: - g = convoluted(lambda: None) - - with self.assertRaises(greenlet.error) as exc: - g.switch() - self.assertIn("Cannot switch to a different thread", str(exc.exception)) - self.assertIn("Expected", str(exc.exception)) - self.assertIn("Current", str(exc.exception)) - finally: - keep_main_alive.set() - t.join(10) - # XXX: Should handle this automatically. - del another[:] - - def test_cannot_delete_parent(self): - worker = RawGreenlet(lambda: None) - self.assertIs(worker.parent, greenlet.getcurrent()) - - with self.assertRaises(AttributeError) as exc: - del worker.parent - self.assertEqual(str(exc.exception), "can't delete attribute") - - def test_cannot_delete_parent_of_main(self): - with self.assertRaises(AttributeError) as exc: - del greenlet.getcurrent().parent - self.assertEqual(str(exc.exception), "can't delete attribute") - - - def test_main_greenlet_parent_is_none(self): - # assuming we're in a main greenlet here. - self.assertIsNone(greenlet.getcurrent().parent) - - def test_set_parent_wrong_types(self): - def bg(): - # Go back to main. - greenlet.getcurrent().parent.switch() - - def check(glet): - for p in None, 1, self, "42": - with self.assertRaises(TypeError) as exc: - glet.parent = p - - self.assertEqual( - str(exc.exception), - "GreenletChecker: Expected any type of greenlet, not " + type(p).__name__) - - # First, not running - g = RawGreenlet(bg) - self.assertFalse(g) - check(g) - - # Then when running. - g.switch() - self.assertTrue(g) - check(g) - - # Let it finish - g.switch() - - - def test_trivial_cycle(self): - glet = RawGreenlet(lambda: None) - with self.assertRaises(ValueError) as exc: - glet.parent = glet - self.assertEqual(str(exc.exception), "cyclic parent chain") - - def test_trivial_cycle_main(self): - # This used to produce a ValueError, but we catch it earlier than that now. - with self.assertRaises(AttributeError) as exc: - greenlet.getcurrent().parent = greenlet.getcurrent() - self.assertEqual(str(exc.exception), "cannot set the parent of a main greenlet") - - def test_deeper_cycle(self): - g1 = RawGreenlet(lambda: None) - g2 = RawGreenlet(lambda: None) - g3 = RawGreenlet(lambda: None) - - g1.parent = g2 - g2.parent = g3 - with self.assertRaises(ValueError) as exc: - g3.parent = g1 - self.assertEqual(str(exc.exception), "cyclic parent chain") - - -class TestRepr(TestCase): - - if not hasattr(TestCase, 'assertEndsWith'): # Added in 3.14 - def assertEndsWith(self, s, suffix, msg=None): - self.assertTrue(s.endswith(suffix), (s, suffix, msg)) - - def test_main_while_running(self): - r = repr(greenlet.getcurrent()) - self.assertEndsWith(r, " current active started main>") - - def test_main_in_background(self): - main = greenlet.getcurrent() - def run(): - return repr(main) - - g = RawGreenlet(run) - r = g.switch() - self.assertEndsWith(r, ' suspended active started main>') - - def test_initial(self): - r = repr(RawGreenlet()) - self.assertEndsWith(r, ' pending>') - - def test_main_from_other_thread(self): - main = greenlet.getcurrent() - - class T(threading.Thread): - original_main = thread_main = None - main_glet = None - def run(self): - self.original_main = repr(main) - self.main_glet = greenlet.getcurrent() - self.thread_main = repr(self.main_glet) - - t = T() - t.start() - t.join(10) - - self.assertEndsWith(t.original_main, ' suspended active started main>') - self.assertEndsWith(t.thread_main, ' current active started main>') - # give the machinery time to notice the death of the thread, - # and clean it up. Note that we don't use - # ``expect_greenlet_leak`` or wait_for_pending_cleanups, - # because at this point we know we have an extra greenlet - # still reachable. - for _ in range(3): - time.sleep(0.001) - - # In the past, main greenlets, even from dead threads, never - # really appear dead. We have fixed that, and we also report - # that the thread is dead in the repr. (Do this multiple times - # to make sure that we don't self-modify and forget our state - # in the C++ code). - for _ in range(3): - self.assertTrue(t.main_glet.dead) - r = repr(t.main_glet) - self.assertEndsWith(r, ' (thread exited) dead>') - - def test_dead(self): - g = RawGreenlet(lambda: None) - g.switch() - self.assertEndsWith(repr(g), ' dead>') - self.assertNotIn('suspended', repr(g)) - self.assertNotIn('started', repr(g)) - self.assertNotIn('active', repr(g)) - - def test_formatting_produces_native_str(self): - # https://github.com/python-greenlet/greenlet/issues/218 - # %s formatting on Python 2 was producing unicode, not str. - - g_dead = RawGreenlet(lambda: None) - g_not_started = RawGreenlet(lambda: None) - g_cur = greenlet.getcurrent() - - for g in g_dead, g_not_started, g_cur: - - self.assertIsInstance( - '%s' % (g,), - str - ) - self.assertIsInstance( - '%r' % (g,), - str, - ) - - -class TestMainGreenlet(TestCase): - # Tests some implementation details, and relies on some - # implementation details. - - def _check_current_is_main(self): - # implementation detail - assert 'main' in repr(greenlet.getcurrent()) - - t = type(greenlet.getcurrent()) - assert 'main' not in repr(t) - return t - - def test_main_greenlet_type_can_be_subclassed(self): - main_type = self._check_current_is_main() - subclass = type('subclass', (main_type,), {}) - self.assertIsNotNone(subclass) - - def test_main_greenlet_is_greenlet(self): - self._check_current_is_main() - self.assertIsInstance(greenlet.getcurrent(), RawGreenlet) - - - -class TestBrokenGreenlets(TestCase): - # Tests for things that used to, or still do, terminate the interpreter. - # This often means doing unsavory things. - - def test_failed_to_initialstub(self): - def func(): - raise AssertionError("Never get here") - - - g = greenlet._greenlet.UnswitchableGreenlet(func) - g.force_switch_error = True - - with self.assertRaisesRegex(SystemError, - "Failed to switch stacks into a greenlet for the first time."): - g.switch() - - def test_failed_to_switch_into_running(self): - runs = [] - def func(): - runs.append(1) - greenlet.getcurrent().parent.switch() - runs.append(2) - greenlet.getcurrent().parent.switch() - runs.append(3) # pragma: no cover - - g = greenlet._greenlet.UnswitchableGreenlet(func) - g.switch() - self.assertEqual(runs, [1]) - g.switch() - self.assertEqual(runs, [1, 2]) - g.force_switch_error = True - - with self.assertRaisesRegex(SystemError, - "Failed to switch stacks into a running greenlet."): - g.switch() - - # If we stopped here, we would fail the leakcheck, because we've left - # the ``inner_bootstrap()`` C frame and its descendents hanging around, - # which have a bunch of Python references. They'll never get cleaned up - # if we don't let the greenlet finish. - g.force_switch_error = False - g.switch() - self.assertEqual(runs, [1, 2, 3]) - - def test_failed_to_slp_switch_into_running(self): - ex = self.assertScriptRaises('fail_slp_switch.py') - - self.assertIn('fail_slp_switch is running', ex.output) - self.assertIn(ex.returncode, self.get_expected_returncodes_for_aborted_process()) - - def test_reentrant_switch_two_greenlets(self): - # Before we started capturing the arguments in g_switch_finish, this could crash. - output = self.run_script('fail_switch_two_greenlets.py') - self.assertIn('In g1_run', output) - self.assertIn('TRACE', output) - self.assertIn('LEAVE TRACE', output) - self.assertIn('Falling off end of main', output) - self.assertIn('Falling off end of g1_run', output) - self.assertIn('Falling off end of g2', output) - - def test_reentrant_switch_three_greenlets(self): - # On debug builds of greenlet, this used to crash with an assertion error; - # on non-debug versions, it ran fine (which it should not do!). - # Now it always crashes correctly with a TypeError - ex = self.assertScriptRaises('fail_switch_three_greenlets.py', exitcodes=(1,)) - - self.assertIn('TypeError', ex.output) - self.assertIn('positional arguments', ex.output) - - def test_reentrant_switch_three_greenlets2(self): - # This actually passed on debug and non-debug builds. It - # should probably have been triggering some debug assertions - # but it didn't. - # - # I think the fixes for the above test also kicked in here. - output = self.run_script('fail_switch_three_greenlets2.py') - self.assertIn( - "RESULTS: [('trace', 'switch'), " - "('trace', 'switch'), ('g2 arg', 'g2 from tracefunc'), " - "('trace', 'switch'), ('main g1', 'from g2_run'), ('trace', 'switch'), " - "('g1 arg', 'g1 from main'), ('trace', 'switch'), ('main g2', 'from g1_run'), " - "('trace', 'switch'), ('g1 from parent', 'g1 from main 2'), ('trace', 'switch'), " - "('main g1.2', 'g1 done'), ('trace', 'switch'), ('g2 from parent', ()), " - "('trace', 'switch'), ('main g2.2', 'g2 done')]", - output - ) - - def test_reentrant_switch_GreenletAlreadyStartedInPython(self): - output = self.run_script('fail_initialstub_already_started.py') - - self.assertIn( - "RESULTS: ['Begin C', 'Switch to b from B.__getattribute__ in C', " - "('Begin B', ()), '_B_run switching to main', ('main from c', 'From B'), " - "'B.__getattribute__ back from main in C', ('Begin A', (None,)), " - "('A dead?', True, 'B dead?', True, 'C dead?', False), " - "'C done', ('main from c.2', None)]", - output - ) - - def test_reentrant_switch_run_callable_has_del(self): - output = self.run_script('fail_clearing_run_switches.py') - self.assertIn( - "RESULTS [" - "('G.__getattribute__', 'run'), ('RunCallable', '__del__'), " - "('main: g.switch()', 'from RunCallable'), ('run_func', 'enter')" - "]", - output - ) - -class TestModule(TestCase): - - @unittest.skipUnless(hasattr(sys, '_is_gil_enabled'), - "Needs 3.13 and above for sys._is_gil_enabled") - def test_no_gil_on_free_threaded(self): - - if RUNNING_ON_FREETHREAD_BUILD: - self.assertFalse(sys._is_gil_enabled()) - else: - self.assertTrue(sys._is_gil_enabled()) - -if __name__ == '__main__': - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_greenlet_trash.py b/.env/Lib/site-packages/greenlet/tests/test_greenlet_trash.py deleted file mode 100644 index c1fc137..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_greenlet_trash.py +++ /dev/null @@ -1,187 +0,0 @@ -# -*- coding: utf-8 -*- -""" -Tests for greenlets interacting with the CPython trash can API. - -The CPython trash can API is not designed to be re-entered from a -single thread. But this can happen using greenlets, if something -during the object deallocation process switches greenlets, and this second -greenlet then causes the trash can to get entered again. Here, we do this -very explicitly, but in other cases (like gevent) it could be arbitrarily more -complicated: for example, a weakref callback might try to acquire a lock that's -already held by another greenlet; that would allow a greenlet switch to occur. - -See https://github.com/gevent/gevent/issues/1909 - -This test is fragile and relies on details of the CPython -implementation (like most of the rest of this package): - - - We enter the trashcan and deferred deallocation after - ``_PyTrash_UNWIND_LEVEL`` calls. This constant, defined in - CPython's object.c, is generally 50. That's basically how many objects are required to - get us into the deferred deallocation situation. - - - The test fails by hitting an ``assert()`` in object.c; if the - build didn't enable assert, then we don't catch this. - - - If the test fails in that way, the interpreter crashes. -""" -from __future__ import print_function, absolute_import, division - -import unittest - - -class TestTrashCanReEnter(unittest.TestCase): - - def test_it(self): - try: - # pylint:disable-next=no-name-in-module - from greenlet._greenlet import get_tstate_trash_delete_nesting # pylint:disable=unused-import - except ImportError: - import sys - # Python 3.13 has not "trash delete nesting" anymore (but "delete later") - assert sys.version_info[:2] >= (3, 13) - self.skipTest("get_tstate_trash_delete_nesting is not available.") - - # Try several times to trigger it, because it isn't 100% - # reliable. - for _ in range(10): - self.check_it() - - def check_it(self): # pylint:disable=too-many-statements - import greenlet - from greenlet._greenlet import get_tstate_trash_delete_nesting # pylint:disable=no-name-in-module - main = greenlet.getcurrent() - - assert get_tstate_trash_delete_nesting() == 0 - - # We expect to be in deferred deallocation after this many - # deallocations have occurred. TODO: I wish we had a better way to do - # this --- that was before get_tstate_trash_delete_nesting; perhaps - # we can use that API to do better? - TRASH_UNWIND_LEVEL = 50 - # How many objects to put in a container; it's the container that - # queues objects for deferred deallocation. - OBJECTS_PER_CONTAINER = 500 - - class Dealloc: # define the class here because we alter class variables each time we run. - """ - An object with a ``__del__`` method. When it starts getting deallocated - from a deferred trash can run, it switches greenlets, allocates more objects - which then also go in the trash can. If we don't save state appropriately, - nesting gets out of order and we can crash the interpreter. - """ - - #: Has our deallocation actually run and switched greenlets? - #: When it does, this will be set to the current greenlet. This should - #: be happening in the main greenlet, so we check that down below. - SPAWNED = False - - #: Has the background greenlet run? - BG_RAN = False - - BG_GLET = None - - #: How many of these things have ever been allocated. - CREATED = 0 - - #: How many of these things have ever been deallocated. - DESTROYED = 0 - - #: How many were destroyed not in the main greenlet. There should always - #: be some. - #: If the test is broken or things change in the trashcan implementation, - #: this may not be correct. - DESTROYED_BG = 0 - - def __init__(self, sequence_number): - """ - :param sequence_number: The ordinal of this object during - one particular creation run. This is used to detect (guess, really) - when we have entered the trash can's deferred deallocation. - """ - self.i = sequence_number - Dealloc.CREATED += 1 - - def __del__(self): - if self.i == TRASH_UNWIND_LEVEL and not self.SPAWNED: - Dealloc.SPAWNED = greenlet.getcurrent() - other = Dealloc.BG_GLET = greenlet.greenlet(background_greenlet) - x = other.switch() - assert x == 42 - # It's important that we don't switch back to the greenlet, - # we leave it hanging there in an incomplete state. But we don't let it - # get collected, either. If we complete it now, while we're still - # in the scope of the initial trash can, things work out and we - # don't see the problem. We need this greenlet to complete - # at some point in the future, after we've exited this trash can invocation. - del other - elif self.i == 40 and greenlet.getcurrent() is not main: - Dealloc.BG_RAN = True - try: - main.switch(42) - except greenlet.GreenletExit as ex: - # We expect this; all references to us go away - # while we're still running, and we need to finish deleting - # ourself. - Dealloc.BG_RAN = type(ex) - del ex - - # Record the fact that we're dead last of all. This ensures that - # we actually get returned too. - Dealloc.DESTROYED += 1 - if greenlet.getcurrent() is not main: - Dealloc.DESTROYED_BG += 1 - - - def background_greenlet(): - # We direct through a second function, instead of - # directly calling ``make_some()``, so that we have complete - # control over when these objects are destroyed: we need them - # to be destroyed in the context of the background greenlet - t = make_some() - del t # Triggere deletion. - - def make_some(): - t = () - i = OBJECTS_PER_CONTAINER - while i: - # Nest the tuples; it's the recursion that gets us - # into trash. - t = (Dealloc(i), t) - i -= 1 - return t - - - some = make_some() - self.assertEqual(Dealloc.CREATED, OBJECTS_PER_CONTAINER) - self.assertEqual(Dealloc.DESTROYED, 0) - - # If we're going to crash, it should be on the following line. - # We only crash if ``assert()`` is enabled, of course. - del some - - # For non-debug builds of CPython, we won't crash. The best we can do is check - # the nesting level explicitly. - self.assertEqual(0, get_tstate_trash_delete_nesting()) - - # Discard this, raising GreenletExit into where it is waiting. - Dealloc.BG_GLET = None - # The same nesting level maintains. - self.assertEqual(0, get_tstate_trash_delete_nesting()) - - # We definitely cleaned some up in the background - self.assertGreater(Dealloc.DESTROYED_BG, 0) - - # Make sure all the cleanups happened. - self.assertIs(Dealloc.SPAWNED, main) - self.assertTrue(Dealloc.BG_RAN) - self.assertEqual(Dealloc.BG_RAN, greenlet.GreenletExit) - self.assertEqual(Dealloc.CREATED, Dealloc.DESTROYED ) - self.assertEqual(Dealloc.CREATED, OBJECTS_PER_CONTAINER * 2) - - import gc - gc.collect() - - -if __name__ == '__main__': - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_leaks.py b/.env/Lib/site-packages/greenlet/tests/test_leaks.py deleted file mode 100644 index e09da7d..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_leaks.py +++ /dev/null @@ -1,457 +0,0 @@ -# -*- coding: utf-8 -*- -""" -Testing scenarios that may have leaked. -""" -from __future__ import print_function, absolute_import, division - -import sys -import gc - -import time -import weakref -import threading - - -import greenlet -from . import TestCase -from . import PY314 -from . import RUNNING_ON_FREETHREAD_BUILD -from .leakcheck import fails_leakcheck -from .leakcheck import ignores_leakcheck -from .leakcheck import RUNNING_ON_MANYLINUX - - -# pylint:disable=protected-access - -assert greenlet.GREENLET_USE_GC # Option to disable this was removed in 1.0 - -class HasFinalizerTracksInstances(object): - EXTANT_INSTANCES = set() - def __init__(self, msg): - self.msg = sys.intern(msg) - self.EXTANT_INSTANCES.add(id(self)) - def __del__(self): - self.EXTANT_INSTANCES.remove(id(self)) - def __repr__(self): - return "" % ( - id(self), self.msg - ) - @classmethod - def reset(cls): - cls.EXTANT_INSTANCES.clear() - - -def fails_leakcheck_except_on_free_thraded(func): - if RUNNING_ON_FREETHREAD_BUILD: - # These all seem to pass on free threading because - # of the changes to the garbage collector - return func - return fails_leakcheck(func) - - -class TestLeaks(TestCase): - - def test_arg_refs(self): - args = ('a', 'b', 'c') - refcount_before = sys.getrefcount(args) - # pylint:disable=unnecessary-lambda - g = greenlet.greenlet( - lambda *args: greenlet.getcurrent().parent.switch(*args)) - for _ in range(100): - g.switch(*args) - self.assertEqual(sys.getrefcount(args), refcount_before) - - def test_kwarg_refs(self): - kwargs = {} - self.assertEqual(sys.getrefcount(kwargs), 2 if not PY314 else 1) - # pylint:disable=unnecessary-lambda - g = greenlet.greenlet( - lambda **gkwargs: greenlet.getcurrent().parent.switch(**gkwargs)) - for _ in range(100): - g.switch(**kwargs) - # Python 3.14 elides reference counting operations - # in some cases. See https://github.com/python/cpython/pull/130708 - self.assertEqual(sys.getrefcount(kwargs), 2 if not PY314 else 1) - - - @staticmethod - def __recycle_threads(): - # By introducing a thread that does sleep we allow other threads, - # that have triggered their __block condition, but did not have a - # chance to deallocate their thread state yet, to finally do so. - # The way it works is by requiring a GIL switch (different thread), - # which does a GIL release (sleep), which might do a GIL switch - # to finished threads and allow them to clean up. - def worker(): - time.sleep(0.001) - t = threading.Thread(target=worker) - t.start() - time.sleep(0.001) - t.join(10) - - def test_threaded_leak(self): - gg = [] - def worker(): - # only main greenlet present - gg.append(weakref.ref(greenlet.getcurrent())) - for _ in range(2): - t = threading.Thread(target=worker) - t.start() - t.join(10) - del t - greenlet.getcurrent() # update ts_current - self.__recycle_threads() - greenlet.getcurrent() # update ts_current - gc.collect() - greenlet.getcurrent() # update ts_current - for g in gg: - self.assertIsNone(g()) - - def test_threaded_adv_leak(self): - gg = [] - def worker(): - # main and additional *finished* greenlets - ll = greenlet.getcurrent().ll = [] - def additional(): - ll.append(greenlet.getcurrent()) - for _ in range(2): - greenlet.greenlet(additional).switch() - gg.append(weakref.ref(greenlet.getcurrent())) - for _ in range(2): - t = threading.Thread(target=worker) - t.start() - t.join(10) - del t - greenlet.getcurrent() # update ts_current - self.__recycle_threads() - greenlet.getcurrent() # update ts_current - gc.collect() - greenlet.getcurrent() # update ts_current - for g in gg: - self.assertIsNone(g()) - - def assertClocksUsed(self): - used = greenlet._greenlet.get_clocks_used_doing_optional_cleanup() - self.assertGreaterEqual(used, 0) - # we don't lose the value - greenlet._greenlet.enable_optional_cleanup(True) - used2 = greenlet._greenlet.get_clocks_used_doing_optional_cleanup() - self.assertEqual(used, used2) - self.assertGreater(greenlet._greenlet.CLOCKS_PER_SEC, 1) - - def _check_issue251(self, - manually_collect_background=True, - explicit_reference_to_switch=False): - # See https://github.com/python-greenlet/greenlet/issues/251 - # Killing a greenlet (probably not the main one) - # in one thread from another thread would - # result in leaking a list (the ts_delkey list). - # We no longer use lists to hold that stuff, though. - - # For the test to be valid, even empty lists have to be tracked by the - # GC - - assert gc.is_tracked([]) - HasFinalizerTracksInstances.reset() - greenlet.getcurrent() - greenlets_before = self.count_objects(greenlet.greenlet, exact_kind=False) - - background_glet_running = threading.Event() - background_glet_killed = threading.Event() - background_greenlets = [] - - # XXX: Switching this to a greenlet subclass that overrides - # run results in all callers failing the leaktest; that - # greenlet instance is leaked. There's a bound method for - # run() living on the stack of the greenlet in g_initialstub, - # and since we don't manually switch back to the background - # greenlet to let it "fall off the end" and exit the - # g_initialstub function, it never gets cleaned up. Making the - # garbage collector aware of this bound method (making it an - # attribute of the greenlet structure and traversing into it) - # doesn't help, for some reason. - def background_greenlet(): - # Throw control back to the main greenlet. - jd = HasFinalizerTracksInstances("DELETING STACK OBJECT") - greenlet._greenlet.set_thread_local( - 'test_leaks_key', - HasFinalizerTracksInstances("DELETING THREAD STATE")) - # Explicitly keeping 'switch' in a local variable - # breaks this test in all versions - if explicit_reference_to_switch: - s = greenlet.getcurrent().parent.switch - s([jd]) - else: - greenlet.getcurrent().parent.switch([jd]) - - bg_main_wrefs = [] - - def background_thread(): - glet = greenlet.greenlet(background_greenlet) - bg_main_wrefs.append(weakref.ref(glet.parent)) - - background_greenlets.append(glet) - glet.switch() # Be sure it's active. - # Control is ours again. - del glet # Delete one reference from the thread it runs in. - background_glet_running.set() - background_glet_killed.wait(10) - - # To trigger the background collection of the dead - # greenlet, thus clearing out the contents of the list, we - # need to run some APIs. See issue 252. - if manually_collect_background: - greenlet.getcurrent() - - - t = threading.Thread(target=background_thread) - t.start() - background_glet_running.wait(10) - greenlet.getcurrent() - lists_before = self.count_objects(list, exact_kind=True) - - assert len(background_greenlets) == 1 - self.assertFalse(background_greenlets[0].dead) - # Delete the last reference to the background greenlet - # from a different thread. This puts it in the background thread's - # ts_delkey list. - del background_greenlets[:] - background_glet_killed.set() - - # Now wait for the background thread to die. - t.join(10) - del t - # As part of the fix for 252, we need to cycle the ceval.c - # interpreter loop to be sure it has had a chance to process - # the pending call. - self.wait_for_pending_cleanups() - - lists_after = self.count_objects(list, exact_kind=True) - greenlets_after = self.count_objects(greenlet.greenlet, exact_kind=False) - - # On 2.7, we observe that lists_after is smaller than - # lists_before. No idea what lists got cleaned up. All the - # Python 3 versions match exactly. - self.assertLessEqual(lists_after, lists_before) - # On versions after 3.6, we've successfully cleaned up the - # greenlet references thanks to the internal "vectorcall" - # protocol; prior to that, there is a reference path through - # the ``greenlet.switch`` method still on the stack that we - # can't reach to clean up. The C code goes through terrific - # lengths to clean that up. - if not explicit_reference_to_switch \ - and greenlet._greenlet.get_clocks_used_doing_optional_cleanup() is not None: - # If cleanup was disabled, though, we may not find it. - self.assertEqual(greenlets_after, greenlets_before) - if manually_collect_background: - # TODO: Figure out how to make this work! - # The one on the stack is still leaking somehow - # in the non-manually-collect state. - self.assertEqual(HasFinalizerTracksInstances.EXTANT_INSTANCES, set()) - else: - # The explicit reference prevents us from collecting it - # and it isn't always found by the GC either for some - # reason. The entire frame is leaked somehow, on some - # platforms (e.g., MacPorts builds of Python (all - # versions!)), but not on other platforms (the linux and - # windows builds on GitHub actions and Appveyor). So we'd - # like to write a test that proves that the main greenlet - # sticks around, and we can on my machine (macOS 11.6, - # MacPorts builds of everything) but we can't write that - # same test on other platforms. However, hopefully iteration - # done by leakcheck will find it. - pass - - if greenlet._greenlet.get_clocks_used_doing_optional_cleanup() is not None: - self.assertClocksUsed() - - def test_issue251_killing_cross_thread_leaks_list(self): - self._check_issue251() - - def test_issue251_with_cleanup_disabled(self): - greenlet._greenlet.enable_optional_cleanup(False) - try: - self._check_issue251() - finally: - greenlet._greenlet.enable_optional_cleanup(True) - - @fails_leakcheck_except_on_free_thraded - def test_issue251_issue252_need_to_collect_in_background(self): - # Between greenlet 1.1.2 and the next version, this was still - # failing because the leak of the list still exists when we - # don't call a greenlet API before exiting the thread. The - # proximate cause is that neither of the two greenlets from - # the background thread are actually being destroyed, even - # though the GC is in fact visiting both objects. It's not - # clear where that leak is? For some reason the thread-local - # dict holding it isn't being cleaned up. - # - # The leak, I think, is in the CPYthon internal function that - # calls into green_switch(). The argument tuple is still on - # the C stack somewhere and can't be reached? That doesn't - # make sense, because the tuple should be collectable when - # this object goes away. - # - # Note that this test sometimes spuriously passes on Linux, - # for some reason, but I've never seen it pass on macOS. - self._check_issue251(manually_collect_background=False) - - @fails_leakcheck_except_on_free_thraded - def test_issue251_issue252_need_to_collect_in_background_cleanup_disabled(self): - self.expect_greenlet_leak = True - greenlet._greenlet.enable_optional_cleanup(False) - try: - self._check_issue251(manually_collect_background=False) - finally: - greenlet._greenlet.enable_optional_cleanup(True) - - @fails_leakcheck_except_on_free_thraded - def test_issue251_issue252_explicit_reference_not_collectable(self): - self._check_issue251( - manually_collect_background=False, - explicit_reference_to_switch=True) - - UNTRACK_ATTEMPTS = 100 - - def _only_test_some_versions(self): - # We're only looking for this problem specifically on 3.11, - # and this set of tests is relatively fragile, depending on - # OS and memory management details. So we want to run it on 3.11+ - # (obviously) but not every older 3.x version in order to reduce - # false negatives. At the moment, those false results seem to have - # resolved, so we are actually running this on 3.8+ - assert sys.version_info[0] >= 3 - if sys.version_info[:2] < (3, 8): - self.skipTest('Only observed on 3.11') - if RUNNING_ON_MANYLINUX: - self.skipTest("Slow and not worth repeating here") - - @ignores_leakcheck - # Because we're just trying to track raw memory, not objects, and running - # the leakcheck makes an already slow test slower. - def test_untracked_memory_doesnt_increase(self): - # See https://github.com/gevent/gevent/issues/1924 - # and https://github.com/python-greenlet/greenlet/issues/328 - self._only_test_some_versions() - def f(): - return 1 - - ITER = 10000 - def run_it(): - for _ in range(ITER): - greenlet.greenlet(f).switch() - - # Establish baseline - for _ in range(3): - run_it() - - # uss: (Linux, macOS, Windows): aka "Unique Set Size", this is - # the memory which is unique to a process and which would be - # freed if the process was terminated right now. - uss_before = self.get_process_uss() - - for count in range(self.UNTRACK_ATTEMPTS): - uss_before = max(uss_before, self.get_process_uss()) - run_it() - - uss_after = self.get_process_uss() - if uss_after <= uss_before and count > 1: - break - - self.assertLessEqual(uss_after, uss_before) - - def _check_untracked_memory_thread(self, deallocate_in_thread=True): - self._only_test_some_versions() - # Like the above test, but what if there are a bunch of - # unfinished greenlets in a thread that dies? - # Does it matter if we deallocate in the thread or not? - EXIT_COUNT = [0] - - def f(): - try: - greenlet.getcurrent().parent.switch() - except greenlet.GreenletExit: - EXIT_COUNT[0] += 1 - raise - return 1 - - ITER = 10000 - def run_it(): - glets = [] - for _ in range(ITER): - # Greenlet starts, switches back to us. - # We keep a strong reference to the greenlet though so it doesn't - # get a GreenletExit exception. - g = greenlet.greenlet(f) - glets.append(g) - g.switch() - - return glets - - test = self - - class ThreadFunc: - uss_before = uss_after = 0 - glets = () - ITER = 2 - def __call__(self): - self.uss_before = test.get_process_uss() - - for _ in range(self.ITER): - self.glets += tuple(run_it()) - - for g in self.glets: - test.assertIn('suspended active', str(g)) - # Drop them. - if deallocate_in_thread: - self.glets = () - self.uss_after = test.get_process_uss() - - # Establish baseline - uss_before = uss_after = None - for count in range(self.UNTRACK_ATTEMPTS): - EXIT_COUNT[0] = 0 - thread_func = ThreadFunc() - t = threading.Thread(target=thread_func) - t.start() - t.join(30) - self.assertFalse(t.is_alive()) - - if uss_before is None: - uss_before = thread_func.uss_before - - uss_before = max(uss_before, thread_func.uss_before) - if deallocate_in_thread: - self.assertEqual(thread_func.glets, ()) - self.assertEqual(EXIT_COUNT[0], ITER * thread_func.ITER) - - del thread_func # Deallocate the greenlets; but this won't raise into them - del t - if not deallocate_in_thread: - self.assertEqual(EXIT_COUNT[0], 0) - if deallocate_in_thread: - self.wait_for_pending_cleanups() - - uss_after = self.get_process_uss() - # See if we achieve a non-growth state at some point. Break when we do. - if uss_after <= uss_before and count > 1: - break - - self.wait_for_pending_cleanups() - uss_after = self.get_process_uss() - self.assertLessEqual(uss_after, uss_before, "after attempts %d" % (count,)) - - @ignores_leakcheck - # Because we're just trying to track raw memory, not objects, and running - # the leakcheck makes an already slow test slower. - def test_untracked_memory_doesnt_increase_unfinished_thread_dealloc_in_thread(self): - self._check_untracked_memory_thread(deallocate_in_thread=True) - - @ignores_leakcheck - # Because the main greenlets from the background threads do not exit in a timely fashion, - # we fail the object-based leakchecks. - def test_untracked_memory_doesnt_increase_unfinished_thread_dealloc_in_main(self): - self._check_untracked_memory_thread(deallocate_in_thread=False) - -if __name__ == '__main__': - __import__('unittest').main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_stack_saved.py b/.env/Lib/site-packages/greenlet/tests/test_stack_saved.py deleted file mode 100644 index b362bf9..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_stack_saved.py +++ /dev/null @@ -1,19 +0,0 @@ -import greenlet -from . import TestCase - - -class Test(TestCase): - - def test_stack_saved(self): - main = greenlet.getcurrent() - self.assertEqual(main._stack_saved, 0) - - def func(): - main.switch(main._stack_saved) - - g = greenlet.greenlet(func) - x = g.switch() - self.assertGreater(x, 0) - self.assertGreater(g._stack_saved, 0) - g.switch() - self.assertEqual(g._stack_saved, 0) diff --git a/.env/Lib/site-packages/greenlet/tests/test_throw.py b/.env/Lib/site-packages/greenlet/tests/test_throw.py deleted file mode 100644 index f4f9a14..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_throw.py +++ /dev/null @@ -1,128 +0,0 @@ -import sys - - -from greenlet import greenlet -from . import TestCase - -def switch(*args): - return greenlet.getcurrent().parent.switch(*args) - - -class ThrowTests(TestCase): - def test_class(self): - def f(): - try: - switch("ok") - except RuntimeError: - switch("ok") - return - switch("fail") - g = greenlet(f) - res = g.switch() - self.assertEqual(res, "ok") - res = g.throw(RuntimeError) - self.assertEqual(res, "ok") - - def test_val(self): - def f(): - try: - switch("ok") - except RuntimeError: - val = sys.exc_info()[1] - if str(val) == "ciao": - switch("ok") - return - switch("fail") - - g = greenlet(f) - res = g.switch() - self.assertEqual(res, "ok") - res = g.throw(RuntimeError("ciao")) - self.assertEqual(res, "ok") - - g = greenlet(f) - res = g.switch() - self.assertEqual(res, "ok") - res = g.throw(RuntimeError, "ciao") - self.assertEqual(res, "ok") - - def test_kill(self): - def f(): - switch("ok") - switch("fail") - g = greenlet(f) - res = g.switch() - self.assertEqual(res, "ok") - res = g.throw() - self.assertTrue(isinstance(res, greenlet.GreenletExit)) - self.assertTrue(g.dead) - res = g.throw() # immediately eaten by the already-dead greenlet - self.assertTrue(isinstance(res, greenlet.GreenletExit)) - - def test_throw_goes_to_original_parent(self): - main = greenlet.getcurrent() - - def f1(): - try: - main.switch("f1 ready to catch") - except IndexError: - return "caught" - return "normal exit" - - def f2(): - main.switch("from f2") - - g1 = greenlet(f1) - g2 = greenlet(f2, parent=g1) - with self.assertRaises(IndexError): - g2.throw(IndexError) - self.assertTrue(g2.dead) - self.assertTrue(g1.dead) - - g1 = greenlet(f1) - g2 = greenlet(f2, parent=g1) - res = g1.switch() - self.assertEqual(res, "f1 ready to catch") - res = g2.throw(IndexError) - self.assertEqual(res, "caught") - self.assertTrue(g2.dead) - self.assertTrue(g1.dead) - - g1 = greenlet(f1) - g2 = greenlet(f2, parent=g1) - res = g1.switch() - self.assertEqual(res, "f1 ready to catch") - res = g2.switch() - self.assertEqual(res, "from f2") - res = g2.throw(IndexError) - self.assertEqual(res, "caught") - self.assertTrue(g2.dead) - self.assertTrue(g1.dead) - - def test_non_traceback_param(self): - with self.assertRaises(TypeError) as exc: - greenlet.getcurrent().throw( - Exception, - Exception(), - self - ) - self.assertEqual(str(exc.exception), - "throw() third argument must be a traceback object") - - def test_instance_of_wrong_type(self): - with self.assertRaises(TypeError) as exc: - greenlet.getcurrent().throw( - Exception(), - BaseException() - ) - - self.assertEqual(str(exc.exception), - "instance exception may not have a separate value") - - def test_not_throwable(self): - with self.assertRaises(TypeError) as exc: - greenlet.getcurrent().throw( - "abc" - ) - self.assertEqual(str(exc.exception), - "exceptions must be classes, or instances, not str") diff --git a/.env/Lib/site-packages/greenlet/tests/test_tracing.py b/.env/Lib/site-packages/greenlet/tests/test_tracing.py deleted file mode 100644 index 235fbcd..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_tracing.py +++ /dev/null @@ -1,299 +0,0 @@ -from __future__ import print_function -import sys -import sysconfig -import greenlet -import unittest - -from . import TestCase -from . import PY312 - -# https://discuss.python.org/t/cpython-3-12-greenlet-and-tracing-profiling-how-to-not-crash-and-get-correct-results/33144/2 -# When build variables are available, OPT is the best way of detecting -# the build with assertions enabled. Otherwise, fallback to detecting PyDEBUG -# build. -ASSERTION_BUILD_PY312 = ( - PY312 and ( - "-DNDEBUG" not in sysconfig.get_config_var("OPT").split() - if sysconfig.get_config_var("OPT") is not None - else hasattr(sys, 'gettotalrefcount') - ), - "Broken on assertion-enabled builds of Python 3.12" -) - -class SomeError(Exception): - pass - -class GreenletTracer(object): - oldtrace = None - - def __init__(self, error_on_trace=False): - self.actions = [] - self.error_on_trace = error_on_trace - - def __call__(self, *args): - self.actions.append(args) - if self.error_on_trace: - raise SomeError - - def __enter__(self): - self.oldtrace = greenlet.settrace(self) - return self.actions - - def __exit__(self, *args): - greenlet.settrace(self.oldtrace) - - -class TestGreenletTracing(TestCase): - """ - Tests of ``greenlet.settrace()`` - """ - - def test_a_greenlet_tracing(self): - main = greenlet.getcurrent() - def dummy(): - pass - def dummyexc(): - raise SomeError() - - with GreenletTracer() as actions: - g1 = greenlet.greenlet(dummy) - g1.switch() - g2 = greenlet.greenlet(dummyexc) - self.assertRaises(SomeError, g2.switch) - - self.assertEqual(actions, [ - ('switch', (main, g1)), - ('switch', (g1, main)), - ('switch', (main, g2)), - ('throw', (g2, main)), - ]) - - def test_b_exception_disables_tracing(self): - main = greenlet.getcurrent() - def dummy(): - main.switch() - g = greenlet.greenlet(dummy) - g.switch() - with GreenletTracer(error_on_trace=True) as actions: - self.assertRaises(SomeError, g.switch) - self.assertEqual(greenlet.gettrace(), None) - - self.assertEqual(actions, [ - ('switch', (main, g)), - ]) - - def test_set_same_tracer_twice(self): - # https://github.com/python-greenlet/greenlet/issues/332 - # Our logic in asserting that the tracefunction should - # gain a reference was incorrect if the same tracefunction was set - # twice. - tracer = GreenletTracer() - with tracer: - greenlet.settrace(tracer) - - -class PythonTracer(object): - oldtrace = None - - def __init__(self): - self.actions = [] - - def __call__(self, frame, event, arg): - # Record the co_name so we have an idea what function we're in. - self.actions.append((event, frame.f_code.co_name)) - - def __enter__(self): - self.oldtrace = sys.setprofile(self) - return self.actions - - def __exit__(self, *args): - sys.setprofile(self.oldtrace) - -def tpt_callback(): - return 42 - -class TestPythonTracing(TestCase): - """ - Tests of the interaction of ``sys.settrace()`` - with greenlet facilities. - - NOTE: Most of this is probably CPython specific. - """ - - maxDiff = None - - def test_trace_events_trivial(self): - with PythonTracer() as actions: - tpt_callback() - # If we use the sys.settrace instead of setprofile, we get - # this: - - # self.assertEqual(actions, [ - # ('call', 'tpt_callback'), - # ('call', '__exit__'), - # ]) - - self.assertEqual(actions, [ - ('return', '__enter__'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('call', '__exit__'), - ('c_call', '__exit__'), - ]) - - def _trace_switch(self, glet): - with PythonTracer() as actions: - glet.switch() - return actions - - def _check_trace_events_func_already_set(self, glet): - actions = self._trace_switch(glet) - self.assertEqual(actions, [ - ('return', '__enter__'), - ('c_call', '_trace_switch'), - ('call', 'run'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('return', 'run'), - ('c_return', '_trace_switch'), - ('call', '__exit__'), - ('c_call', '__exit__'), - ]) - - def test_trace_events_into_greenlet_func_already_set(self): - def run(): - return tpt_callback() - - self._check_trace_events_func_already_set(greenlet.greenlet(run)) - - def test_trace_events_into_greenlet_subclass_already_set(self): - class X(greenlet.greenlet): - def run(self): - return tpt_callback() - self._check_trace_events_func_already_set(X()) - - def _check_trace_events_from_greenlet_sets_profiler(self, g, tracer): - g.switch() - tpt_callback() - tracer.__exit__() - self.assertEqual(tracer.actions, [ - ('return', '__enter__'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('return', 'run'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('call', '__exit__'), - ('c_call', '__exit__'), - ]) - - - def test_trace_events_from_greenlet_func_sets_profiler(self): - tracer = PythonTracer() - def run(): - tracer.__enter__() - return tpt_callback() - - self._check_trace_events_from_greenlet_sets_profiler(greenlet.greenlet(run), - tracer) - - def test_trace_events_from_greenlet_subclass_sets_profiler(self): - tracer = PythonTracer() - class X(greenlet.greenlet): - def run(self): - tracer.__enter__() - return tpt_callback() - - self._check_trace_events_from_greenlet_sets_profiler(X(), tracer) - - @unittest.skipIf(*ASSERTION_BUILD_PY312) - def test_trace_events_multiple_greenlets_switching(self): - tracer = PythonTracer() - - g1 = None - g2 = None - - def g1_run(): - tracer.__enter__() - tpt_callback() - g2.switch() - tpt_callback() - return 42 - - def g2_run(): - tpt_callback() - tracer.__exit__() - tpt_callback() - g1.switch() - - g1 = greenlet.greenlet(g1_run) - g2 = greenlet.greenlet(g2_run) - - x = g1.switch() - self.assertEqual(x, 42) - tpt_callback() # ensure not in the trace - self.assertEqual(tracer.actions, [ - ('return', '__enter__'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('c_call', 'g1_run'), - ('call', 'g2_run'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('call', '__exit__'), - ('c_call', '__exit__'), - ]) - - @unittest.skipIf(*ASSERTION_BUILD_PY312) - def test_trace_events_multiple_greenlets_switching_siblings(self): - # Like the first version, but get both greenlets running first - # as "siblings" and then establish the tracing. - tracer = PythonTracer() - - g1 = None - g2 = None - - def g1_run(): - greenlet.getcurrent().parent.switch() - tracer.__enter__() - tpt_callback() - g2.switch() - tpt_callback() - return 42 - - def g2_run(): - greenlet.getcurrent().parent.switch() - - tpt_callback() - tracer.__exit__() - tpt_callback() - g1.switch() - - g1 = greenlet.greenlet(g1_run) - g2 = greenlet.greenlet(g2_run) - - # Start g1 - g1.switch() - # And it immediately returns control to us. - # Start g2 - g2.switch() - # Which also returns. Now kick of the real part of the - # test. - x = g1.switch() - self.assertEqual(x, 42) - - tpt_callback() # ensure not in the trace - self.assertEqual(tracer.actions, [ - ('return', '__enter__'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('c_call', 'g1_run'), - ('call', 'tpt_callback'), - ('return', 'tpt_callback'), - ('call', '__exit__'), - ('c_call', '__exit__'), - ]) - - -if __name__ == '__main__': - unittest.main() diff --git a/.env/Lib/site-packages/greenlet/tests/test_version.py b/.env/Lib/site-packages/greenlet/tests/test_version.py deleted file mode 100644 index 96c17cf..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_version.py +++ /dev/null @@ -1,41 +0,0 @@ -#! /usr/bin/env python -from __future__ import absolute_import -from __future__ import print_function - -import sys -import os -from unittest import TestCase as NonLeakingTestCase - -import greenlet - -# No reason to run this multiple times under leakchecks, -# it doesn't do anything. -class VersionTests(NonLeakingTestCase): - def test_version(self): - def find_dominating_file(name): - if os.path.exists(name): - return name - - tried = [] - here = os.path.abspath(os.path.dirname(__file__)) - for i in range(10): - up = ['..'] * i - path = [here] + up + [name] - fname = os.path.join(*path) - fname = os.path.abspath(fname) - tried.append(fname) - if os.path.exists(fname): - return fname - raise AssertionError("Could not find file " + name + "; checked " + str(tried)) - - try: - setup_py = find_dominating_file('setup.py') - except AssertionError as e: - self.skipTest("Unable to find setup.py; must be out of tree. " + str(e)) - - - invoke_setup = "%s %s --version" % (sys.executable, setup_py) - with os.popen(invoke_setup) as f: - sversion = f.read().strip() - - self.assertEqual(sversion, greenlet.__version__) diff --git a/.env/Lib/site-packages/greenlet/tests/test_weakref.py b/.env/Lib/site-packages/greenlet/tests/test_weakref.py deleted file mode 100644 index 05a38a7..0000000 --- a/.env/Lib/site-packages/greenlet/tests/test_weakref.py +++ /dev/null @@ -1,35 +0,0 @@ -import gc -import weakref - - -import greenlet -from . import TestCase - -class WeakRefTests(TestCase): - def test_dead_weakref(self): - def _dead_greenlet(): - g = greenlet.greenlet(lambda: None) - g.switch() - return g - o = weakref.ref(_dead_greenlet()) - gc.collect() - self.assertEqual(o(), None) - - def test_inactive_weakref(self): - o = weakref.ref(greenlet.greenlet()) - gc.collect() - self.assertEqual(o(), None) - - def test_dealloc_weakref(self): - seen = [] - def worker(): - try: - greenlet.getcurrent().parent.switch() - finally: - seen.append(g()) - g = greenlet.greenlet(worker) - g.switch() - g2 = greenlet.greenlet(lambda: None, g) - g = weakref.ref(g2) - g2 = None - self.assertEqual(seen, [None]) diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/INSTALLER b/.env/Lib/site-packages/h11-0.16.0.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/METADATA b/.env/Lib/site-packages/h11-0.16.0.dist-info/METADATA deleted file mode 100644 index 8a2f639..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/METADATA +++ /dev/null @@ -1,202 +0,0 @@ -Metadata-Version: 2.4 -Name: h11 -Version: 0.16.0 -Summary: A pure-Python, bring-your-own-I/O implementation of HTTP/1.1 -Home-page: https://github.com/python-hyper/h11 -Author: Nathaniel J. Smith -Author-email: njs@pobox.com -License: MIT -Classifier: Development Status :: 3 - Alpha -Classifier: Intended Audience :: Developers -Classifier: License :: OSI Approved :: MIT License -Classifier: Programming Language :: Python :: Implementation :: CPython -Classifier: Programming Language :: Python :: Implementation :: PyPy -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Topic :: Internet :: WWW/HTTP -Classifier: Topic :: System :: Networking -Requires-Python: >=3.8 -License-File: LICENSE.txt -Dynamic: author -Dynamic: author-email -Dynamic: classifier -Dynamic: description -Dynamic: home-page -Dynamic: license -Dynamic: license-file -Dynamic: requires-python -Dynamic: summary - -h11 -=== - -.. image:: https://travis-ci.org/python-hyper/h11.svg?branch=master - :target: https://travis-ci.org/python-hyper/h11 - :alt: Automated test status - -.. image:: https://codecov.io/gh/python-hyper/h11/branch/master/graph/badge.svg - :target: https://codecov.io/gh/python-hyper/h11 - :alt: Test coverage - -.. image:: https://readthedocs.org/projects/h11/badge/?version=latest - :target: http://h11.readthedocs.io/en/latest/?badge=latest - :alt: Documentation Status - -This is a little HTTP/1.1 library written from scratch in Python, -heavily inspired by `hyper-h2 `_. - -It's a "bring-your-own-I/O" library; h11 contains no IO code -whatsoever. This means you can hook h11 up to your favorite network -API, and that could be anything you want: synchronous, threaded, -asynchronous, or your own implementation of `RFC 6214 -`_ -- h11 won't judge you. -(Compare this to the current state of the art, where every time a `new -network API `_ comes along then someone -gets to start over reimplementing the entire HTTP protocol from -scratch.) Cory Benfield made an `excellent blog post describing the -benefits of this approach -`_, or if you like video -then here's his `PyCon 2016 talk on the same theme -`_. - -This also means that h11 is not immediately useful out of the box: -it's a toolkit for building programs that speak HTTP, not something -that could directly replace ``requests`` or ``twisted.web`` or -whatever. But h11 makes it much easier to implement something like -``requests`` or ``twisted.web``. - -At a high level, working with h11 goes like this: - -1) First, create an ``h11.Connection`` object to track the state of a - single HTTP/1.1 connection. - -2) When you read data off the network, pass it to - ``conn.receive_data(...)``; you'll get back a list of objects - representing high-level HTTP "events". - -3) When you want to send a high-level HTTP event, create the - corresponding "event" object and pass it to ``conn.send(...)``; - this will give you back some bytes that you can then push out - through the network. - -For example, a client might instantiate and then send a -``h11.Request`` object, then zero or more ``h11.Data`` objects for the -request body (e.g., if this is a POST), and then a -``h11.EndOfMessage`` to indicate the end of the message. Then the -server would then send back a ``h11.Response``, some ``h11.Data``, and -its own ``h11.EndOfMessage``. If either side violates the protocol, -you'll get a ``h11.ProtocolError`` exception. - -h11 is suitable for implementing both servers and clients, and has a -pleasantly symmetric API: the events you send as a client are exactly -the ones that you receive as a server and vice-versa. - -`Here's an example of a tiny HTTP client -`_ - -It also has `a fine manual `_. - -FAQ ---- - -*Whyyyyy?* - -I wanted to play with HTTP in `Curio -`__ and `Trio -`__, which at the time didn't have any -HTTP libraries. So I thought, no big deal, Python has, like, a dozen -different implementations of HTTP, surely I can find one that's -reusable. I didn't find one, but I did find Cory's call-to-arms -blog-post. So I figured, well, fine, if I have to implement HTTP from -scratch, at least I can make sure no-one *else* has to ever again. - -*Should I use it?* - -Maybe. You should be aware that it's a very young project. But, it's -feature complete and has an exhaustive test-suite and complete docs, -so the next step is for people to try using it and see how it goes -:-). If you do then please let us know -- if nothing else we'll want -to talk to you before making any incompatible changes! - -*What are the features/limitations?* - -Roughly speaking, it's trying to be a robust, complete, and non-hacky -implementation of the first "chapter" of the HTTP/1.1 spec: `RFC 7230: -HTTP/1.1 Message Syntax and Routing -`_. That is, it mostly focuses on -implementing HTTP at the level of taking bytes on and off the wire, -and the headers related to that, and tries to be anal about spec -conformance. It doesn't know about higher-level concerns like URL -routing, conditional GETs, cross-origin cookie policies, or content -negotiation. But it does know how to take care of framing, -cross-version differences in keep-alive handling, and the "obsolete -line folding" rule, so you can focus your energies on the hard / -interesting parts for your application, and it tries to support the -full specification in the sense that any useful HTTP/1.1 conformant -application should be able to use h11. - -It's pure Python, and has no dependencies outside of the standard -library. - -It has a test suite with 100.0% coverage for both statements and -branches. - -Currently it supports Python 3 (testing on 3.8-3.12) and PyPy 3. -The last Python 2-compatible version was h11 0.11.x. -(Originally it had a Cython wrapper for `http-parser -`_ and a beautiful nested state -machine implemented with ``yield from`` to postprocess the output. But -I had to take these out -- the new *parser* needs fewer lines-of-code -than the old *parser wrapper*, is written in pure Python, uses no -exotic language syntax, and has more features. It's sad, really; that -old state machine was really slick. I just need a few sentences here -to mourn that.) - -I don't know how fast it is. I haven't benchmarked or profiled it yet, -so it's probably got a few pointless hot spots, and I've been trying -to err on the side of simplicity and robustness instead of -micro-optimization. But at the architectural level I tried hard to -avoid fundamentally bad decisions, e.g., I believe that all the -parsing algorithms remain linear-time even in the face of pathological -input like slowloris, and there are no byte-by-byte loops. (I also -believe that it maintains bounded memory usage in the face of -arbitrary/pathological input.) - -The whole library is ~800 lines-of-code. You can read and understand -the whole thing in less than an hour. Most of the energy invested in -this so far has been spent on trying to keep things simple by -minimizing special-cases and ad hoc state manipulation; even though it -is now quite small and simple, I'm still annoyed that I haven't -figured out how to make it even smaller and simpler. (Unfortunately, -HTTP does not lend itself to simplicity.) - -The API is ~feature complete and I don't expect the general outlines -to change much, but you can't judge an API's ergonomics until you -actually document and use it, so I'd expect some changes in the -details. - -*How do I try it?* - -.. code-block:: sh - - $ pip install h11 - $ git clone git@github.com:python-hyper/h11 - $ cd h11/examples - $ python basic-client.py - -and go from there. - -*License?* - -MIT - -*Code of conduct?* - -Contributors are requested to follow our `code of conduct -`_ in -all project spaces. diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/RECORD b/.env/Lib/site-packages/h11-0.16.0.dist-info/RECORD deleted file mode 100644 index fa60234..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/RECORD +++ /dev/null @@ -1,29 +0,0 @@ -h11-0.16.0.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -h11-0.16.0.dist-info/METADATA,sha256=KPMmCYrAn8unm48YD5YIfIQf4kViFct7hyqcfVzRnWQ,8348 -h11-0.16.0.dist-info/RECORD,, -h11-0.16.0.dist-info/WHEEL,sha256=CmyFI0kx5cdEMTLiONQRbGQwjIoR1aIYB7eCAQ4KPJ0,91 -h11-0.16.0.dist-info/licenses/LICENSE.txt,sha256=N9tbuFkm2yikJ6JYZ_ELEjIAOuob5pzLhRE4rbjm82E,1124 -h11-0.16.0.dist-info/top_level.txt,sha256=F7dC4jl3zeh8TGHEPaWJrMbeuoWbS379Gwdi-Yvdcis,4 -h11/__init__.py,sha256=iO1KzkSO42yZ6ffg-VMgbx_ZVTWGUY00nRYEWn-s3kY,1507 -h11/__pycache__/__init__.cpython-312.pyc,, -h11/__pycache__/_abnf.cpython-312.pyc,, -h11/__pycache__/_connection.cpython-312.pyc,, -h11/__pycache__/_events.cpython-312.pyc,, -h11/__pycache__/_headers.cpython-312.pyc,, -h11/__pycache__/_readers.cpython-312.pyc,, -h11/__pycache__/_receivebuffer.cpython-312.pyc,, -h11/__pycache__/_state.cpython-312.pyc,, -h11/__pycache__/_util.cpython-312.pyc,, -h11/__pycache__/_version.cpython-312.pyc,, -h11/__pycache__/_writers.cpython-312.pyc,, -h11/_abnf.py,sha256=ybixr0xsupnkA6GFAyMubuXF6Tc1lb_hF890NgCsfNc,4815 -h11/_connection.py,sha256=k9YRVf6koZqbttBW36xSWaJpWdZwa-xQVU9AHEo9DuI,26863 -h11/_events.py,sha256=I97aXoal1Wu7dkL548BANBUCkOIbe-x5CioYA9IBY14,11792 -h11/_headers.py,sha256=P7D-lBNxHwdLZPLimmYwrPG-9ZkjElvvJZJdZAgSP-4,10412 -h11/_readers.py,sha256=a4RypORUCC3d0q_kxPuBIM7jTD8iLt5X91TH0FsduN4,8590 -h11/_receivebuffer.py,sha256=xrspsdsNgWFxRfQcTXxR8RrdjRXXTK0Io5cQYWpJ1Ws,5252 -h11/_state.py,sha256=_5LG_BGR8FCcFQeBPH-TMHgm_-B-EUcWCnQof_9XjFE,13231 -h11/_util.py,sha256=LWkkjXyJaFlAy6Lt39w73UStklFT5ovcvo0TkY7RYuk,4888 -h11/_version.py,sha256=GVSsbPSPDcOuF6ptfIiXnVJoaEm3ygXbMnqlr_Giahw,686 -h11/_writers.py,sha256=oFKm6PtjeHfbj4RLX7VB7KDc1gIY53gXG3_HR9ltmTA,5081 -h11/py.typed,sha256=sow9soTwP9T_gEAQSVh7Gb8855h04Nwmhs2We-JRgZM,7 diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/WHEEL b/.env/Lib/site-packages/h11-0.16.0.dist-info/WHEEL deleted file mode 100644 index 1eb3c49..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/WHEEL +++ /dev/null @@ -1,5 +0,0 @@ -Wheel-Version: 1.0 -Generator: setuptools (78.1.0) -Root-Is-Purelib: true -Tag: py3-none-any - diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/licenses/LICENSE.txt b/.env/Lib/site-packages/h11-0.16.0.dist-info/licenses/LICENSE.txt deleted file mode 100644 index 8f080ea..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/licenses/LICENSE.txt +++ /dev/null @@ -1,22 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2016 Nathaniel J. Smith and other contributors - -Permission is hereby granted, free of charge, to any person obtaining -a copy of this software and associated documentation files (the -"Software"), to deal in the Software without restriction, including -without limitation the rights to use, copy, modify, merge, publish, -distribute, sublicense, and/or sell copies of the Software, and to -permit persons to whom the Software is furnished to do so, subject to -the following conditions: - -The above copyright notice and this permission notice shall be -included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/.env/Lib/site-packages/h11-0.16.0.dist-info/top_level.txt b/.env/Lib/site-packages/h11-0.16.0.dist-info/top_level.txt deleted file mode 100644 index 0d24def..0000000 --- a/.env/Lib/site-packages/h11-0.16.0.dist-info/top_level.txt +++ /dev/null @@ -1 +0,0 @@ -h11 diff --git a/.env/Lib/site-packages/h11/__init__.py b/.env/Lib/site-packages/h11/__init__.py deleted file mode 100644 index 989e92c..0000000 --- a/.env/Lib/site-packages/h11/__init__.py +++ /dev/null @@ -1,62 +0,0 @@ -# A highish-level implementation of the HTTP/1.1 wire protocol (RFC 7230), -# containing no networking code at all, loosely modelled on hyper-h2's generic -# implementation of HTTP/2 (and in particular the h2.connection.H2Connection -# class). There's still a bunch of subtle details you need to get right if you -# want to make this actually useful, because it doesn't implement all the -# semantics to check that what you're asking to write to the wire is sensible, -# but at least it gets you out of dealing with the wire itself. - -from h11._connection import Connection, NEED_DATA, PAUSED -from h11._events import ( - ConnectionClosed, - Data, - EndOfMessage, - Event, - InformationalResponse, - Request, - Response, -) -from h11._state import ( - CLIENT, - CLOSED, - DONE, - ERROR, - IDLE, - MIGHT_SWITCH_PROTOCOL, - MUST_CLOSE, - SEND_BODY, - SEND_RESPONSE, - SERVER, - SWITCHED_PROTOCOL, -) -from h11._util import LocalProtocolError, ProtocolError, RemoteProtocolError -from h11._version import __version__ - -PRODUCT_ID = "python-h11/" + __version__ - - -__all__ = ( - "Connection", - "NEED_DATA", - "PAUSED", - "ConnectionClosed", - "Data", - "EndOfMessage", - "Event", - "InformationalResponse", - "Request", - "Response", - "CLIENT", - "CLOSED", - "DONE", - "ERROR", - "IDLE", - "MUST_CLOSE", - "SEND_BODY", - "SEND_RESPONSE", - "SERVER", - "SWITCHED_PROTOCOL", - "ProtocolError", - "LocalProtocolError", - "RemoteProtocolError", -) diff --git a/.env/Lib/site-packages/h11/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 1f120f6b68b50eacba5b818e5cc4589835beea77..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1067 zcma))O>fgM7{`;oByG~HZ|k~s40f0hJ#5pCObBT-5v$ZRRoa3uk*ttLvnkio8pj%N{qpMLN(=<$yX{#NL)b#e#{Q>MhZx34guxAHtfhdVpl&3c%ocN%6GXl-zzz#uZ@81f9P zd_tL_z@RWJFccX|3}uEY!y-eCVTqy6u)?s+(4gox*GZA?z6G7%@A#JuA}8aBJy7%? zo`{spk77GU|Mpj54*q4519IudK+8(dO88 zj%Zy*7Ts!{=Qe&QjgK!+l?QL7b75Idy%WO?s`v{n%M?8T diff --git a/.env/Lib/site-packages/h11/__pycache__/_abnf.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_abnf.cpython-312.pyc deleted file mode 100644 index fef9b625cf53f63b0514898a3b6941a1b5b1dae0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1773 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 diff --git a/.env/Lib/site-packages/h11/__pycache__/_events.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_events.cpython-312.pyc deleted file mode 100644 index de6e085188ea6d22ce5c0daab27797d1ec50098f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13213 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 a+A=G)J$8k~;6!mzz>C^gy+8&0Sp6^L$eM)! diff --git a/.env/Lib/site-packages/h11/__pycache__/_headers.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_headers.cpython-312.pyc deleted file mode 100644 index 719cc43e049a414a23eba86f76512d70d20b1223..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7975 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 diff --git a/.env/Lib/site-packages/h11/__pycache__/_receivebuffer.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_receivebuffer.cpython-312.pyc deleted file mode 100644 index db60b3fb2e3a319ff733161d0a547fee217b479d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4691 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

}gkwkSW2{iJJIe5LDL-Z(Y=- zqKk_wrRf65VuaR@feHv3}mn?(eEehEn-#q5hmMka>9`NEGC;vkAv z55(21js7;2Kd8`r&kEy>!_ErPOAz?Y#6E^Dyr@Tv1k?&oIniKGqYamsukxUy_r5A~pX>KKRaP=5=?<-Zn9eDS2_(_1^?4Z@Eox<@k1J z>4t_aoLS)t-niv6+FOyNYx!aUwz-Eh%ULbh!_t}60-a0TN`cOWD=RxSE%`SvA!j}O zw3BrGbQm6RR7qaESAmZU3tdZw2R+LMFx-QmUtR5b&V6xQZtG3CPs#REDNElA{~vb# z?r&wwSc>z*+L*?fH+sFnn9;L_vNw98QzsL9rivJN)OLNzB=0z|IxbfnO>>nC-X+iS zrPVIkeh5I>x^1~euGpF8s^sd{YyiL}|V5PfZNEKJDe#--aClfhaJM?q6f+T6T+jG}4F_T>@T+$DGSr!CbB zpUCER*>eOWT~xB_uNzN(D6MuFigQ~R%(Px^?A##R7+vNRLs!(4Y)Rg@*8%(Z#={P| zd}oUFtaPois~5khc)`B7_`3?Zb2znoL}o|cz&SL#R>pd{VXw?~tk%ef&&r+WUK{wN z|IYBj_)`WKEl_$M-h89jQot5qFX@e=*T_9Qfb}%oQspho-^Nj8<4HCl~_zu!| IM)e*42RT>iB>(^b diff --git a/.env/Lib/site-packages/h11/__pycache__/_util.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_util.cpython-312.pyc deleted file mode 100644 index b9ac8097db34fa53b4cf2c41428f466647baaebc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4711 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 Tb6Y*nw+OJmIKGb!_rZSvrr(EH diff --git a/.env/Lib/site-packages/h11/__pycache__/_version.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_version.cpython-312.pyc deleted file mode 100644 index 83be77be01f9d2456de05e7a4fc03d9f9b5faad8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 205 zcmX@j%ge<81iKHkWv*jlV0aATzyKqZ@tFh2n9h*G5XDf*sL52tW}s(irf1-%$#{!9 zKE5oqs5mn}FFt-H!)K6&U;fTkF`>mk(U_q8lzau>#N>?Byv*#>n1ad}-^}Eqe4t!e zW^!tAj9zM9S&UC+QcQ7XNvdu^Vsdt3I#4*n&`=*{tX@IoFAkgB{FKt1RJ$TJp!pzI Y74raz56p~=j2{#j_{AH!i&%gH0DD|Gg#Z8m diff --git a/.env/Lib/site-packages/h11/__pycache__/_writers.cpython-312.pyc b/.env/Lib/site-packages/h11/__pycache__/_writers.cpython-312.pyc deleted file mode 100644 index 78ea228c46f3971593cf05bebb30cd00c5b9056c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6277 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 Y2VTg6M=IFuP5ivP(7ZwH9M2H_Kf`rS=Kufz diff --git a/.env/Lib/site-packages/h11/_abnf.py b/.env/Lib/site-packages/h11/_abnf.py deleted file mode 100644 index 933587f..0000000 --- a/.env/Lib/site-packages/h11/_abnf.py +++ /dev/null @@ -1,132 +0,0 @@ -# We use native strings for all the re patterns, to take advantage of string -# formatting, and then convert to bytestrings when compiling the final re -# objects. - -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#whitespace -# OWS = *( SP / HTAB ) -# ; optional whitespace -OWS = r"[ \t]*" - -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#rule.token.separators -# token = 1*tchar -# -# tchar = "!" / "#" / "$" / "%" / "&" / "'" / "*" -# / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~" -# / DIGIT / ALPHA -# ; any VCHAR, except delimiters -token = r"[-!#$%&'*+.^_`|~0-9a-zA-Z]+" - -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#header.fields -# field-name = token -field_name = token - -# The standard says: -# -# field-value = *( field-content / obs-fold ) -# field-content = field-vchar [ 1*( SP / HTAB ) field-vchar ] -# field-vchar = VCHAR / obs-text -# obs-fold = CRLF 1*( SP / HTAB ) -# ; obsolete line folding -# ; see Section 3.2.4 -# -# https://tools.ietf.org/html/rfc5234#appendix-B.1 -# -# VCHAR = %x21-7E -# ; visible (printing) characters -# -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#rule.quoted-string -# obs-text = %x80-FF -# -# However, the standard definition of field-content is WRONG! It disallows -# fields containing a single visible character surrounded by whitespace, -# e.g. "foo a bar". -# -# See: https://www.rfc-editor.org/errata_search.php?rfc=7230&eid=4189 -# -# So our definition of field_content attempts to fix it up... -# -# Also, we allow lots of control characters, because apparently people assume -# that they're legal in practice (e.g., google analytics makes cookies with -# \x01 in them!): -# https://github.com/python-hyper/h11/issues/57 -# We still don't allow NUL or whitespace, because those are often treated as -# meta-characters and letting them through can lead to nasty issues like SSRF. -vchar = r"[\x21-\x7e]" -vchar_or_obs_text = r"[^\x00\s]" -field_vchar = vchar_or_obs_text -field_content = r"{field_vchar}+(?:[ \t]+{field_vchar}+)*".format(**globals()) - -# We handle obs-fold at a different level, and our fixed-up field_content -# already grows to swallow the whole value, so ? instead of * -field_value = r"({field_content})?".format(**globals()) - -# header-field = field-name ":" OWS field-value OWS -header_field = ( - r"(?P{field_name})" - r":" - r"{OWS}" - r"(?P{field_value})" - r"{OWS}".format(**globals()) -) - -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#request.line -# -# request-line = method SP request-target SP HTTP-version CRLF -# method = token -# HTTP-version = HTTP-name "/" DIGIT "." DIGIT -# HTTP-name = %x48.54.54.50 ; "HTTP", case-sensitive -# -# request-target is complicated (see RFC 7230 sec 5.3) -- could be path, full -# URL, host+port (for connect), or even "*", but in any case we are guaranteed -# that it contists of the visible printing characters. -method = token -request_target = r"{vchar}+".format(**globals()) -http_version = r"HTTP/(?P[0-9]\.[0-9])" -request_line = ( - r"(?P{method})" - r" " - r"(?P{request_target})" - r" " - r"{http_version}".format(**globals()) -) - -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#status.line -# -# status-line = HTTP-version SP status-code SP reason-phrase CRLF -# status-code = 3DIGIT -# reason-phrase = *( HTAB / SP / VCHAR / obs-text ) -status_code = r"[0-9]{3}" -reason_phrase = r"([ \t]|{vchar_or_obs_text})*".format(**globals()) -status_line = ( - r"{http_version}" - r" " - r"(?P{status_code})" - # However, there are apparently a few too many servers out there that just - # leave out the reason phrase: - # https://github.com/scrapy/scrapy/issues/345#issuecomment-281756036 - # https://github.com/seanmonstar/httparse/issues/29 - # so make it optional. ?: is a non-capturing group. - r"(?: (?P{reason_phrase}))?".format(**globals()) -) - -HEXDIG = r"[0-9A-Fa-f]" -# Actually -# -# chunk-size = 1*HEXDIG -# -# but we impose an upper-limit to avoid ridiculosity. len(str(2**64)) == 20 -chunk_size = r"({HEXDIG}){{1,20}}".format(**globals()) -# Actually -# -# chunk-ext = *( ";" chunk-ext-name [ "=" chunk-ext-val ] ) -# -# but we aren't parsing the things so we don't really care. -chunk_ext = r";.*" -chunk_header = ( - r"(?P{chunk_size})" - r"(?P{chunk_ext})?" - r"{OWS}\r\n".format( - **globals() - ) # Even though the specification does not allow for extra whitespaces, - # we are lenient with trailing whitespaces because some servers on the wild use it. -) diff --git a/.env/Lib/site-packages/h11/_connection.py b/.env/Lib/site-packages/h11/_connection.py deleted file mode 100644 index e37d82a..0000000 --- a/.env/Lib/site-packages/h11/_connection.py +++ /dev/null @@ -1,659 +0,0 @@ -# This contains the main Connection class. Everything in h11 revolves around -# this. -from typing import ( - Any, - Callable, - cast, - Dict, - List, - Optional, - overload, - Tuple, - Type, - Union, -) - -from ._events import ( - ConnectionClosed, - Data, - EndOfMessage, - Event, - InformationalResponse, - Request, - Response, -) -from ._headers import get_comma_header, has_expect_100_continue, set_comma_header -from ._readers import READERS, ReadersType -from ._receivebuffer import ReceiveBuffer -from ._state import ( - _SWITCH_CONNECT, - _SWITCH_UPGRADE, - CLIENT, - ConnectionState, - DONE, - ERROR, - MIGHT_SWITCH_PROTOCOL, - SEND_BODY, - SERVER, - SWITCHED_PROTOCOL, -) -from ._util import ( # Import the internal things we need - LocalProtocolError, - RemoteProtocolError, - Sentinel, -) -from ._writers import WRITERS, WritersType - -# Everything in __all__ gets re-exported as part of the h11 public API. -__all__ = ["Connection", "NEED_DATA", "PAUSED"] - - -class NEED_DATA(Sentinel, metaclass=Sentinel): - pass - - -class PAUSED(Sentinel, metaclass=Sentinel): - pass - - -# If we ever have this much buffered without it making a complete parseable -# event, we error out. The only time we really buffer is when reading the -# request/response line + headers together, so this is effectively the limit on -# the size of that. -# -# Some precedents for defaults: -# - node.js: 80 * 1024 -# - tomcat: 8 * 1024 -# - IIS: 16 * 1024 -# - Apache: <8 KiB per line> -DEFAULT_MAX_INCOMPLETE_EVENT_SIZE = 16 * 1024 - - -# RFC 7230's rules for connection lifecycles: -# - If either side says they want to close the connection, then the connection -# must close. -# - HTTP/1.1 defaults to keep-alive unless someone says Connection: close -# - HTTP/1.0 defaults to close unless both sides say Connection: keep-alive -# (and even this is a mess -- e.g. if you're implementing a proxy then -# sending Connection: keep-alive is forbidden). -# -# We simplify life by simply not supporting keep-alive with HTTP/1.0 peers. So -# our rule is: -# - If someone says Connection: close, we will close -# - If someone uses HTTP/1.0, we will close. -def _keep_alive(event: Union[Request, Response]) -> bool: - connection = get_comma_header(event.headers, b"connection") - if b"close" in connection: - return False - if getattr(event, "http_version", b"1.1") < b"1.1": - return False - return True - - -def _body_framing( - request_method: bytes, event: Union[Request, Response] -) -> Tuple[str, Union[Tuple[()], Tuple[int]]]: - # Called when we enter SEND_BODY to figure out framing information for - # this body. - # - # These are the only two events that can trigger a SEND_BODY state: - assert type(event) in (Request, Response) - # Returns one of: - # - # ("content-length", count) - # ("chunked", ()) - # ("http/1.0", ()) - # - # which are (lookup key, *args) for constructing body reader/writer - # objects. - # - # Reference: https://tools.ietf.org/html/rfc7230#section-3.3.3 - # - # Step 1: some responses always have an empty body, regardless of what the - # headers say. - if type(event) is Response: - if ( - event.status_code in (204, 304) - or request_method == b"HEAD" - or (request_method == b"CONNECT" and 200 <= event.status_code < 300) - ): - return ("content-length", (0,)) - # Section 3.3.3 also lists another case -- responses with status_code - # < 200. For us these are InformationalResponses, not Responses, so - # they can't get into this function in the first place. - assert event.status_code >= 200 - - # Step 2: check for Transfer-Encoding (T-E beats C-L): - transfer_encodings = get_comma_header(event.headers, b"transfer-encoding") - if transfer_encodings: - assert transfer_encodings == [b"chunked"] - return ("chunked", ()) - - # Step 3: check for Content-Length - content_lengths = get_comma_header(event.headers, b"content-length") - if content_lengths: - return ("content-length", (int(content_lengths[0]),)) - - # Step 4: no applicable headers; fallback/default depends on type - if type(event) is Request: - return ("content-length", (0,)) - else: - return ("http/1.0", ()) - - -################################################################ -# -# The main Connection class -# -################################################################ - - -class Connection: - """An object encapsulating the state of an HTTP connection. - - Args: - our_role: If you're implementing a client, pass :data:`h11.CLIENT`. If - you're implementing a server, pass :data:`h11.SERVER`. - - max_incomplete_event_size (int): - The maximum number of bytes we're willing to buffer of an - incomplete event. In practice this mostly sets a limit on the - maximum size of the request/response line + headers. If this is - exceeded, then :meth:`next_event` will raise - :exc:`RemoteProtocolError`. - - """ - - def __init__( - self, - our_role: Type[Sentinel], - max_incomplete_event_size: int = DEFAULT_MAX_INCOMPLETE_EVENT_SIZE, - ) -> None: - self._max_incomplete_event_size = max_incomplete_event_size - # State and role tracking - if our_role not in (CLIENT, SERVER): - raise ValueError(f"expected CLIENT or SERVER, not {our_role!r}") - self.our_role = our_role - self.their_role: Type[Sentinel] - if our_role is CLIENT: - self.their_role = SERVER - else: - self.their_role = CLIENT - self._cstate = ConnectionState() - - # Callables for converting data->events or vice-versa given the - # current state - self._writer = self._get_io_object(self.our_role, None, WRITERS) - self._reader = self._get_io_object(self.their_role, None, READERS) - - # Holds any unprocessed received data - self._receive_buffer = ReceiveBuffer() - # If this is true, then it indicates that the incoming connection was - # closed *after* the end of whatever's in self._receive_buffer: - self._receive_buffer_closed = False - - # Extra bits of state that don't fit into the state machine. - # - # These two are only used to interpret framing headers for figuring - # out how to read/write response bodies. their_http_version is also - # made available as a convenient public API. - self.their_http_version: Optional[bytes] = None - self._request_method: Optional[bytes] = None - # This is pure flow-control and doesn't at all affect the set of legal - # transitions, so no need to bother ConnectionState with it: - self.client_is_waiting_for_100_continue = False - - @property - def states(self) -> Dict[Type[Sentinel], Type[Sentinel]]: - """A dictionary like:: - - {CLIENT: , SERVER: } - - See :ref:`state-machine` for details. - - """ - return dict(self._cstate.states) - - @property - def our_state(self) -> Type[Sentinel]: - """The current state of whichever role we are playing. See - :ref:`state-machine` for details. - """ - return self._cstate.states[self.our_role] - - @property - def their_state(self) -> Type[Sentinel]: - """The current state of whichever role we are NOT playing. See - :ref:`state-machine` for details. - """ - return self._cstate.states[self.their_role] - - @property - def they_are_waiting_for_100_continue(self) -> bool: - return self.their_role is CLIENT and self.client_is_waiting_for_100_continue - - def start_next_cycle(self) -> None: - """Attempt to reset our connection state for a new request/response - cycle. - - If both client and server are in :data:`DONE` state, then resets them - both to :data:`IDLE` state in preparation for a new request/response - cycle on this same connection. Otherwise, raises a - :exc:`LocalProtocolError`. - - See :ref:`keepalive-and-pipelining`. - - """ - old_states = dict(self._cstate.states) - self._cstate.start_next_cycle() - self._request_method = None - # self.their_http_version gets left alone, since it presumably lasts - # beyond a single request/response cycle - assert not self.client_is_waiting_for_100_continue - self._respond_to_state_changes(old_states) - - def _process_error(self, role: Type[Sentinel]) -> None: - old_states = dict(self._cstate.states) - self._cstate.process_error(role) - self._respond_to_state_changes(old_states) - - def _server_switch_event(self, event: Event) -> Optional[Type[Sentinel]]: - if type(event) is InformationalResponse and event.status_code == 101: - return _SWITCH_UPGRADE - if type(event) is Response: - if ( - _SWITCH_CONNECT in self._cstate.pending_switch_proposals - and 200 <= event.status_code < 300 - ): - return _SWITCH_CONNECT - return None - - # All events go through here - def _process_event(self, role: Type[Sentinel], event: Event) -> None: - # First, pass the event through the state machine to make sure it - # succeeds. - old_states = dict(self._cstate.states) - if role is CLIENT and type(event) is Request: - if event.method == b"CONNECT": - self._cstate.process_client_switch_proposal(_SWITCH_CONNECT) - if get_comma_header(event.headers, b"upgrade"): - self._cstate.process_client_switch_proposal(_SWITCH_UPGRADE) - server_switch_event = None - if role is SERVER: - server_switch_event = self._server_switch_event(event) - self._cstate.process_event(role, type(event), server_switch_event) - - # Then perform the updates triggered by it. - - if type(event) is Request: - self._request_method = event.method - - if role is self.their_role and type(event) in ( - Request, - Response, - InformationalResponse, - ): - event = cast(Union[Request, Response, InformationalResponse], event) - self.their_http_version = event.http_version - - # Keep alive handling - # - # RFC 7230 doesn't really say what one should do if Connection: close - # shows up on a 1xx InformationalResponse. I think the idea is that - # this is not supposed to happen. In any case, if it does happen, we - # ignore it. - if type(event) in (Request, Response) and not _keep_alive( - cast(Union[Request, Response], event) - ): - self._cstate.process_keep_alive_disabled() - - # 100-continue - if type(event) is Request and has_expect_100_continue(event): - self.client_is_waiting_for_100_continue = True - if type(event) in (InformationalResponse, Response): - self.client_is_waiting_for_100_continue = False - if role is CLIENT and type(event) in (Data, EndOfMessage): - self.client_is_waiting_for_100_continue = False - - self._respond_to_state_changes(old_states, event) - - def _get_io_object( - self, - role: Type[Sentinel], - event: Optional[Event], - io_dict: Union[ReadersType, WritersType], - ) -> Optional[Callable[..., Any]]: - # event may be None; it's only used when entering SEND_BODY - state = self._cstate.states[role] - if state is SEND_BODY: - # Special case: the io_dict has a dict of reader/writer factories - # that depend on the request/response framing. - framing_type, args = _body_framing( - cast(bytes, self._request_method), cast(Union[Request, Response], event) - ) - return io_dict[SEND_BODY][framing_type](*args) # type: ignore[index] - else: - # General case: the io_dict just has the appropriate reader/writer - # for this state - return io_dict.get((role, state)) # type: ignore[return-value] - - # This must be called after any action that might have caused - # self._cstate.states to change. - def _respond_to_state_changes( - self, - old_states: Dict[Type[Sentinel], Type[Sentinel]], - event: Optional[Event] = None, - ) -> None: - # Update reader/writer - if self.our_state != old_states[self.our_role]: - self._writer = self._get_io_object(self.our_role, event, WRITERS) - if self.their_state != old_states[self.their_role]: - self._reader = self._get_io_object(self.their_role, event, READERS) - - @property - def trailing_data(self) -> Tuple[bytes, bool]: - """Data that has been received, but not yet processed, represented as - a tuple with two elements, where the first is a byte-string containing - the unprocessed data itself, and the second is a bool that is True if - the receive connection was closed. - - See :ref:`switching-protocols` for discussion of why you'd want this. - """ - return (bytes(self._receive_buffer), self._receive_buffer_closed) - - def receive_data(self, data: bytes) -> None: - """Add data to our internal receive buffer. - - This does not actually do any processing on the data, just stores - it. To trigger processing, you have to call :meth:`next_event`. - - Args: - data (:term:`bytes-like object`): - The new data that was just received. - - Special case: If *data* is an empty byte-string like ``b""``, - then this indicates that the remote side has closed the - connection (end of file). Normally this is convenient, because - standard Python APIs like :meth:`file.read` or - :meth:`socket.recv` use ``b""`` to indicate end-of-file, while - other failures to read are indicated using other mechanisms - like raising :exc:`TimeoutError`. When using such an API you - can just blindly pass through whatever you get from ``read`` - to :meth:`receive_data`, and everything will work. - - But, if you have an API where reading an empty string is a - valid non-EOF condition, then you need to be aware of this and - make sure to check for such strings and avoid passing them to - :meth:`receive_data`. - - Returns: - Nothing, but after calling this you should call :meth:`next_event` - to parse the newly received data. - - Raises: - RuntimeError: - Raised if you pass an empty *data*, indicating EOF, and then - pass a non-empty *data*, indicating more data that somehow - arrived after the EOF. - - (Calling ``receive_data(b"")`` multiple times is fine, - and equivalent to calling it once.) - - """ - if data: - if self._receive_buffer_closed: - raise RuntimeError("received close, then received more data?") - self._receive_buffer += data - else: - self._receive_buffer_closed = True - - def _extract_next_receive_event( - self, - ) -> Union[Event, Type[NEED_DATA], Type[PAUSED]]: - state = self.their_state - # We don't pause immediately when they enter DONE, because even in - # DONE state we can still process a ConnectionClosed() event. But - # if we have data in our buffer, then we definitely aren't getting - # a ConnectionClosed() immediately and we need to pause. - if state is DONE and self._receive_buffer: - return PAUSED - if state is MIGHT_SWITCH_PROTOCOL or state is SWITCHED_PROTOCOL: - return PAUSED - assert self._reader is not None - event = self._reader(self._receive_buffer) - if event is None: - if not self._receive_buffer and self._receive_buffer_closed: - # In some unusual cases (basically just HTTP/1.0 bodies), EOF - # triggers an actual protocol event; in that case, we want to - # return that event, and then the state will change and we'll - # get called again to generate the actual ConnectionClosed(). - if hasattr(self._reader, "read_eof"): - event = self._reader.read_eof() - else: - event = ConnectionClosed() - if event is None: - event = NEED_DATA - return event # type: ignore[no-any-return] - - def next_event(self) -> Union[Event, Type[NEED_DATA], Type[PAUSED]]: - """Parse the next event out of our receive buffer, update our internal - state, and return it. - - This is a mutating operation -- think of it like calling :func:`next` - on an iterator. - - Returns: - : One of three things: - - 1) An event object -- see :ref:`events`. - - 2) The special constant :data:`NEED_DATA`, which indicates that - you need to read more data from your socket and pass it to - :meth:`receive_data` before this method will be able to return - any more events. - - 3) The special constant :data:`PAUSED`, which indicates that we - are not in a state where we can process incoming data (usually - because the peer has finished their part of the current - request/response cycle, and you have not yet called - :meth:`start_next_cycle`). See :ref:`flow-control` for details. - - Raises: - RemoteProtocolError: - The peer has misbehaved. You should close the connection - (possibly after sending some kind of 4xx response). - - Once this method returns :class:`ConnectionClosed` once, then all - subsequent calls will also return :class:`ConnectionClosed`. - - If this method raises any exception besides :exc:`RemoteProtocolError` - then that's a bug -- if it happens please file a bug report! - - If this method raises any exception then it also sets - :attr:`Connection.their_state` to :data:`ERROR` -- see - :ref:`error-handling` for discussion. - - """ - - if self.their_state is ERROR: - raise RemoteProtocolError("Can't receive data when peer state is ERROR") - try: - event = self._extract_next_receive_event() - if event not in [NEED_DATA, PAUSED]: - self._process_event(self.their_role, cast(Event, event)) - if event is NEED_DATA: - if len(self._receive_buffer) > self._max_incomplete_event_size: - # 431 is "Request header fields too large" which is pretty - # much the only situation where we can get here - raise RemoteProtocolError( - "Receive buffer too long", error_status_hint=431 - ) - if self._receive_buffer_closed: - # We're still trying to complete some event, but that's - # never going to happen because no more data is coming - raise RemoteProtocolError("peer unexpectedly closed connection") - return event - except BaseException as exc: - self._process_error(self.their_role) - if isinstance(exc, LocalProtocolError): - exc._reraise_as_remote_protocol_error() - else: - raise - - @overload - def send(self, event: ConnectionClosed) -> None: - ... - - @overload - def send( - self, event: Union[Request, InformationalResponse, Response, Data, EndOfMessage] - ) -> bytes: - ... - - @overload - def send(self, event: Event) -> Optional[bytes]: - ... - - def send(self, event: Event) -> Optional[bytes]: - """Convert a high-level event into bytes that can be sent to the peer, - while updating our internal state machine. - - Args: - event: The :ref:`event ` to send. - - Returns: - If ``type(event) is ConnectionClosed``, then returns - ``None``. Otherwise, returns a :term:`bytes-like object`. - - Raises: - LocalProtocolError: - Sending this event at this time would violate our - understanding of the HTTP/1.1 protocol. - - If this method raises any exception then it also sets - :attr:`Connection.our_state` to :data:`ERROR` -- see - :ref:`error-handling` for discussion. - - """ - data_list = self.send_with_data_passthrough(event) - if data_list is None: - return None - else: - return b"".join(data_list) - - def send_with_data_passthrough(self, event: Event) -> Optional[List[bytes]]: - """Identical to :meth:`send`, except that in situations where - :meth:`send` returns a single :term:`bytes-like object`, this instead - returns a list of them -- and when sending a :class:`Data` event, this - list is guaranteed to contain the exact object you passed in as - :attr:`Data.data`. See :ref:`sendfile` for discussion. - - """ - if self.our_state is ERROR: - raise LocalProtocolError("Can't send data when our state is ERROR") - try: - if type(event) is Response: - event = self._clean_up_response_headers_for_sending(event) - # We want to call _process_event before calling the writer, - # because if someone tries to do something invalid then this will - # give a sensible error message, while our writers all just assume - # they will only receive valid events. But, _process_event might - # change self._writer. So we have to do a little dance: - writer = self._writer - self._process_event(self.our_role, event) - if type(event) is ConnectionClosed: - return None - else: - # In any situation where writer is None, process_event should - # have raised ProtocolError - assert writer is not None - data_list: List[bytes] = [] - writer(event, data_list.append) - return data_list - except: - self._process_error(self.our_role) - raise - - def send_failed(self) -> None: - """Notify the state machine that we failed to send the data it gave - us. - - This causes :attr:`Connection.our_state` to immediately become - :data:`ERROR` -- see :ref:`error-handling` for discussion. - - """ - self._process_error(self.our_role) - - # When sending a Response, we take responsibility for a few things: - # - # - Sometimes you MUST set Connection: close. We take care of those - # times. (You can also set it yourself if you want, and if you do then - # we'll respect that and close the connection at the right time. But you - # don't have to worry about that unless you want to.) - # - # - The user has to set Content-Length if they want it. Otherwise, for - # responses that have bodies (e.g. not HEAD), then we will automatically - # select the right mechanism for streaming a body of unknown length, - # which depends on depending on the peer's HTTP version. - # - # This function's *only* responsibility is making sure headers are set up - # right -- everything downstream just looks at the headers. There are no - # side channels. - def _clean_up_response_headers_for_sending(self, response: Response) -> Response: - assert type(response) is Response - - headers = response.headers - need_close = False - - # HEAD requests need some special handling: they always act like they - # have Content-Length: 0, and that's how _body_framing treats - # them. But their headers are supposed to match what we would send if - # the request was a GET. (Technically there is one deviation allowed: - # we're allowed to leave out the framing headers -- see - # https://tools.ietf.org/html/rfc7231#section-4.3.2 . But it's just as - # easy to get them right.) - method_for_choosing_headers = cast(bytes, self._request_method) - if method_for_choosing_headers == b"HEAD": - method_for_choosing_headers = b"GET" - framing_type, _ = _body_framing(method_for_choosing_headers, response) - if framing_type in ("chunked", "http/1.0"): - # This response has a body of unknown length. - # If our peer is HTTP/1.1, we use Transfer-Encoding: chunked - # If our peer is HTTP/1.0, we use no framing headers, and close the - # connection afterwards. - # - # Make sure to clear Content-Length (in principle user could have - # set both and then we ignored Content-Length b/c - # Transfer-Encoding overwrote it -- this would be naughty of them, - # but the HTTP spec says that if our peer does this then we have - # to fix it instead of erroring out, so we'll accord the user the - # same respect). - headers = set_comma_header(headers, b"content-length", []) - if self.their_http_version is None or self.their_http_version < b"1.1": - # Either we never got a valid request and are sending back an - # error (their_http_version is None), so we assume the worst; - # or else we did get a valid HTTP/1.0 request, so we know that - # they don't understand chunked encoding. - headers = set_comma_header(headers, b"transfer-encoding", []) - # This is actually redundant ATM, since currently we - # unconditionally disable keep-alive when talking to HTTP/1.0 - # peers. But let's be defensive just in case we add - # Connection: keep-alive support later: - if self._request_method != b"HEAD": - need_close = True - else: - headers = set_comma_header(headers, b"transfer-encoding", [b"chunked"]) - - if not self._cstate.keep_alive or need_close: - # Make sure Connection: close is set - connection = set(get_comma_header(headers, b"connection")) - connection.discard(b"keep-alive") - connection.add(b"close") - headers = set_comma_header(headers, b"connection", sorted(connection)) - - return Response( - headers=headers, - status_code=response.status_code, - http_version=response.http_version, - reason=response.reason, - ) diff --git a/.env/Lib/site-packages/h11/_events.py b/.env/Lib/site-packages/h11/_events.py deleted file mode 100644 index ca1c3ad..0000000 --- a/.env/Lib/site-packages/h11/_events.py +++ /dev/null @@ -1,369 +0,0 @@ -# High level events that make up HTTP/1.1 conversations. Loosely inspired by -# the corresponding events in hyper-h2: -# -# http://python-hyper.org/h2/en/stable/api.html#events -# -# Don't subclass these. Stuff will break. - -import re -from abc import ABC -from dataclasses import dataclass -from typing import List, Tuple, Union - -from ._abnf import method, request_target -from ._headers import Headers, normalize_and_validate -from ._util import bytesify, LocalProtocolError, validate - -# Everything in __all__ gets re-exported as part of the h11 public API. -__all__ = [ - "Event", - "Request", - "InformationalResponse", - "Response", - "Data", - "EndOfMessage", - "ConnectionClosed", -] - -method_re = re.compile(method.encode("ascii")) -request_target_re = re.compile(request_target.encode("ascii")) - - -class Event(ABC): - """ - Base class for h11 events. - """ - - __slots__ = () - - -@dataclass(init=False, frozen=True) -class Request(Event): - """The beginning of an HTTP request. - - Fields: - - .. attribute:: method - - An HTTP method, e.g. ``b"GET"`` or ``b"POST"``. Always a byte - string. :term:`Bytes-like objects ` and native - strings containing only ascii characters will be automatically - converted to byte strings. - - .. attribute:: target - - The target of an HTTP request, e.g. ``b"/index.html"``, or one of the - more exotic formats described in `RFC 7320, section 5.3 - `_. Always a byte - string. :term:`Bytes-like objects ` and native - strings containing only ascii characters will be automatically - converted to byte strings. - - .. attribute:: headers - - Request headers, represented as a list of (name, value) pairs. See - :ref:`the header normalization rules ` for details. - - .. attribute:: http_version - - The HTTP protocol version, represented as a byte string like - ``b"1.1"``. See :ref:`the HTTP version normalization rules - ` for details. - - """ - - __slots__ = ("method", "headers", "target", "http_version") - - method: bytes - headers: Headers - target: bytes - http_version: bytes - - def __init__( - self, - *, - method: Union[bytes, str], - headers: Union[Headers, List[Tuple[bytes, bytes]], List[Tuple[str, str]]], - target: Union[bytes, str], - http_version: Union[bytes, str] = b"1.1", - _parsed: bool = False, - ) -> None: - super().__init__() - if isinstance(headers, Headers): - object.__setattr__(self, "headers", headers) - else: - object.__setattr__( - self, "headers", normalize_and_validate(headers, _parsed=_parsed) - ) - if not _parsed: - object.__setattr__(self, "method", bytesify(method)) - object.__setattr__(self, "target", bytesify(target)) - object.__setattr__(self, "http_version", bytesify(http_version)) - else: - object.__setattr__(self, "method", method) - object.__setattr__(self, "target", target) - object.__setattr__(self, "http_version", http_version) - - # "A server MUST respond with a 400 (Bad Request) status code to any - # HTTP/1.1 request message that lacks a Host header field and to any - # request message that contains more than one Host header field or a - # Host header field with an invalid field-value." - # -- https://tools.ietf.org/html/rfc7230#section-5.4 - host_count = 0 - for name, value in self.headers: - if name == b"host": - host_count += 1 - if self.http_version == b"1.1" and host_count == 0: - raise LocalProtocolError("Missing mandatory Host: header") - if host_count > 1: - raise LocalProtocolError("Found multiple Host: headers") - - validate(method_re, self.method, "Illegal method characters") - validate(request_target_re, self.target, "Illegal target characters") - - # This is an unhashable type. - __hash__ = None # type: ignore - - -@dataclass(init=False, frozen=True) -class _ResponseBase(Event): - __slots__ = ("headers", "http_version", "reason", "status_code") - - headers: Headers - http_version: bytes - reason: bytes - status_code: int - - def __init__( - self, - *, - headers: Union[Headers, List[Tuple[bytes, bytes]], List[Tuple[str, str]]], - status_code: int, - http_version: Union[bytes, str] = b"1.1", - reason: Union[bytes, str] = b"", - _parsed: bool = False, - ) -> None: - super().__init__() - if isinstance(headers, Headers): - object.__setattr__(self, "headers", headers) - else: - object.__setattr__( - self, "headers", normalize_and_validate(headers, _parsed=_parsed) - ) - if not _parsed: - object.__setattr__(self, "reason", bytesify(reason)) - object.__setattr__(self, "http_version", bytesify(http_version)) - if not isinstance(status_code, int): - raise LocalProtocolError("status code must be integer") - # Because IntEnum objects are instances of int, but aren't - # duck-compatible (sigh), see gh-72. - object.__setattr__(self, "status_code", int(status_code)) - else: - object.__setattr__(self, "reason", reason) - object.__setattr__(self, "http_version", http_version) - object.__setattr__(self, "status_code", status_code) - - self.__post_init__() - - def __post_init__(self) -> None: - pass - - # This is an unhashable type. - __hash__ = None # type: ignore - - -@dataclass(init=False, frozen=True) -class InformationalResponse(_ResponseBase): - """An HTTP informational response. - - Fields: - - .. attribute:: status_code - - The status code of this response, as an integer. For an - :class:`InformationalResponse`, this is always in the range [100, - 200). - - .. attribute:: headers - - Request headers, represented as a list of (name, value) pairs. See - :ref:`the header normalization rules ` for - details. - - .. attribute:: http_version - - The HTTP protocol version, represented as a byte string like - ``b"1.1"``. See :ref:`the HTTP version normalization rules - ` for details. - - .. attribute:: reason - - The reason phrase of this response, as a byte string. For example: - ``b"OK"``, or ``b"Not Found"``. - - """ - - def __post_init__(self) -> None: - if not (100 <= self.status_code < 200): - raise LocalProtocolError( - "InformationalResponse status_code should be in range " - "[100, 200), not {}".format(self.status_code) - ) - - # This is an unhashable type. - __hash__ = None # type: ignore - - -@dataclass(init=False, frozen=True) -class Response(_ResponseBase): - """The beginning of an HTTP response. - - Fields: - - .. attribute:: status_code - - The status code of this response, as an integer. For an - :class:`Response`, this is always in the range [200, - 1000). - - .. attribute:: headers - - Request headers, represented as a list of (name, value) pairs. See - :ref:`the header normalization rules ` for details. - - .. attribute:: http_version - - The HTTP protocol version, represented as a byte string like - ``b"1.1"``. See :ref:`the HTTP version normalization rules - ` for details. - - .. attribute:: reason - - The reason phrase of this response, as a byte string. For example: - ``b"OK"``, or ``b"Not Found"``. - - """ - - def __post_init__(self) -> None: - if not (200 <= self.status_code < 1000): - raise LocalProtocolError( - "Response status_code should be in range [200, 1000), not {}".format( - self.status_code - ) - ) - - # This is an unhashable type. - __hash__ = None # type: ignore - - -@dataclass(init=False, frozen=True) -class Data(Event): - """Part of an HTTP message body. - - Fields: - - .. attribute:: data - - A :term:`bytes-like object` containing part of a message body. Or, if - using the ``combine=False`` argument to :meth:`Connection.send`, then - any object that your socket writing code knows what to do with, and for - which calling :func:`len` returns the number of bytes that will be - written -- see :ref:`sendfile` for details. - - .. attribute:: chunk_start - - A marker that indicates whether this data object is from the start of a - chunked transfer encoding chunk. This field is ignored when when a Data - event is provided to :meth:`Connection.send`: it is only valid on - events emitted from :meth:`Connection.next_event`. You probably - shouldn't use this attribute at all; see - :ref:`chunk-delimiters-are-bad` for details. - - .. attribute:: chunk_end - - A marker that indicates whether this data object is the last for a - given chunked transfer encoding chunk. This field is ignored when when - a Data event is provided to :meth:`Connection.send`: it is only valid - on events emitted from :meth:`Connection.next_event`. You probably - shouldn't use this attribute at all; see - :ref:`chunk-delimiters-are-bad` for details. - - """ - - __slots__ = ("data", "chunk_start", "chunk_end") - - data: bytes - chunk_start: bool - chunk_end: bool - - def __init__( - self, data: bytes, chunk_start: bool = False, chunk_end: bool = False - ) -> None: - object.__setattr__(self, "data", data) - object.__setattr__(self, "chunk_start", chunk_start) - object.__setattr__(self, "chunk_end", chunk_end) - - # This is an unhashable type. - __hash__ = None # type: ignore - - -# XX FIXME: "A recipient MUST ignore (or consider as an error) any fields that -# are forbidden to be sent in a trailer, since processing them as if they were -# present in the header section might bypass external security filters." -# https://svn.tools.ietf.org/svn/wg/httpbis/specs/rfc7230.html#chunked.trailer.part -# Unfortunately, the list of forbidden fields is long and vague :-/ -@dataclass(init=False, frozen=True) -class EndOfMessage(Event): - """The end of an HTTP message. - - Fields: - - .. attribute:: headers - - Default value: ``[]`` - - Any trailing headers attached to this message, represented as a list of - (name, value) pairs. See :ref:`the header normalization rules - ` for details. - - Must be empty unless ``Transfer-Encoding: chunked`` is in use. - - """ - - __slots__ = ("headers",) - - headers: Headers - - def __init__( - self, - *, - headers: Union[ - Headers, List[Tuple[bytes, bytes]], List[Tuple[str, str]], None - ] = None, - _parsed: bool = False, - ) -> None: - super().__init__() - if headers is None: - headers = Headers([]) - elif not isinstance(headers, Headers): - headers = normalize_and_validate(headers, _parsed=_parsed) - - object.__setattr__(self, "headers", headers) - - # This is an unhashable type. - __hash__ = None # type: ignore - - -@dataclass(frozen=True) -class ConnectionClosed(Event): - """This event indicates that the sender has closed their outgoing - connection. - - Note that this does not necessarily mean that they can't *receive* further - data, because TCP connections are composed to two one-way channels which - can be closed independently. See :ref:`closing` for details. - - No fields. - """ - - pass diff --git a/.env/Lib/site-packages/h11/_headers.py b/.env/Lib/site-packages/h11/_headers.py deleted file mode 100644 index 31da3e2..0000000 --- a/.env/Lib/site-packages/h11/_headers.py +++ /dev/null @@ -1,282 +0,0 @@ -import re -from typing import AnyStr, cast, List, overload, Sequence, Tuple, TYPE_CHECKING, Union - -from ._abnf import field_name, field_value -from ._util import bytesify, LocalProtocolError, validate - -if TYPE_CHECKING: - from ._events import Request - -try: - from typing import Literal -except ImportError: - from typing_extensions import Literal # type: ignore - -CONTENT_LENGTH_MAX_DIGITS = 20 # allow up to 1 billion TB - 1 - - -# Facts -# ----- -# -# Headers are: -# keys: case-insensitive ascii -# values: mixture of ascii and raw bytes -# -# "Historically, HTTP has allowed field content with text in the ISO-8859-1 -# charset [ISO-8859-1], supporting other charsets only through use of -# [RFC2047] encoding. In practice, most HTTP header field values use only a -# subset of the US-ASCII charset [USASCII]. Newly defined header fields SHOULD -# limit their field values to US-ASCII octets. A recipient SHOULD treat other -# octets in field content (obs-text) as opaque data." -# And it deprecates all non-ascii values -# -# Leading/trailing whitespace in header names is forbidden -# -# Values get leading/trailing whitespace stripped -# -# Content-Disposition actually needs to contain unicode semantically; to -# accomplish this it has a terrifically weird way of encoding the filename -# itself as ascii (and even this still has lots of cross-browser -# incompatibilities) -# -# Order is important: -# "a proxy MUST NOT change the order of these field values when forwarding a -# message" -# (and there are several headers where the order indicates a preference) -# -# Multiple occurences of the same header: -# "A sender MUST NOT generate multiple header fields with the same field name -# in a message unless either the entire field value for that header field is -# defined as a comma-separated list [or the header is Set-Cookie which gets a -# special exception]" - RFC 7230. (cookies are in RFC 6265) -# -# So every header aside from Set-Cookie can be merged by b", ".join if it -# occurs repeatedly. But, of course, they can't necessarily be split by -# .split(b","), because quoting. -# -# Given all this mess (case insensitive, duplicates allowed, order is -# important, ...), there doesn't appear to be any standard way to handle -# headers in Python -- they're almost like dicts, but... actually just -# aren't. For now we punt and just use a super simple representation: headers -# are a list of pairs -# -# [(name1, value1), (name2, value2), ...] -# -# where all entries are bytestrings, names are lowercase and have no -# leading/trailing whitespace, and values are bytestrings with no -# leading/trailing whitespace. Searching and updating are done via naive O(n) -# methods. -# -# Maybe a dict-of-lists would be better? - -_content_length_re = re.compile(rb"[0-9]+") -_field_name_re = re.compile(field_name.encode("ascii")) -_field_value_re = re.compile(field_value.encode("ascii")) - - -class Headers(Sequence[Tuple[bytes, bytes]]): - """ - A list-like interface that allows iterating over headers as byte-pairs - of (lowercased-name, value). - - Internally we actually store the representation as three-tuples, - including both the raw original casing, in order to preserve casing - over-the-wire, and the lowercased name, for case-insensitive comparisions. - - r = Request( - method="GET", - target="/", - headers=[("Host", "example.org"), ("Connection", "keep-alive")], - http_version="1.1", - ) - assert r.headers == [ - (b"host", b"example.org"), - (b"connection", b"keep-alive") - ] - assert r.headers.raw_items() == [ - (b"Host", b"example.org"), - (b"Connection", b"keep-alive") - ] - """ - - __slots__ = "_full_items" - - def __init__(self, full_items: List[Tuple[bytes, bytes, bytes]]) -> None: - self._full_items = full_items - - def __bool__(self) -> bool: - return bool(self._full_items) - - def __eq__(self, other: object) -> bool: - return list(self) == list(other) # type: ignore - - def __len__(self) -> int: - return len(self._full_items) - - def __repr__(self) -> str: - return "" % repr(list(self)) - - def __getitem__(self, idx: int) -> Tuple[bytes, bytes]: # type: ignore[override] - _, name, value = self._full_items[idx] - return (name, value) - - def raw_items(self) -> List[Tuple[bytes, bytes]]: - return [(raw_name, value) for raw_name, _, value in self._full_items] - - -HeaderTypes = Union[ - List[Tuple[bytes, bytes]], - List[Tuple[bytes, str]], - List[Tuple[str, bytes]], - List[Tuple[str, str]], -] - - -@overload -def normalize_and_validate(headers: Headers, _parsed: Literal[True]) -> Headers: - ... - - -@overload -def normalize_and_validate(headers: HeaderTypes, _parsed: Literal[False]) -> Headers: - ... - - -@overload -def normalize_and_validate( - headers: Union[Headers, HeaderTypes], _parsed: bool = False -) -> Headers: - ... - - -def normalize_and_validate( - headers: Union[Headers, HeaderTypes], _parsed: bool = False -) -> Headers: - new_headers = [] - seen_content_length = None - saw_transfer_encoding = False - for name, value in headers: - # For headers coming out of the parser, we can safely skip some steps, - # because it always returns bytes and has already run these regexes - # over the data: - if not _parsed: - name = bytesify(name) - value = bytesify(value) - validate(_field_name_re, name, "Illegal header name {!r}", name) - validate(_field_value_re, value, "Illegal header value {!r}", value) - assert isinstance(name, bytes) - assert isinstance(value, bytes) - - raw_name = name - name = name.lower() - if name == b"content-length": - lengths = {length.strip() for length in value.split(b",")} - if len(lengths) != 1: - raise LocalProtocolError("conflicting Content-Length headers") - value = lengths.pop() - validate(_content_length_re, value, "bad Content-Length") - if len(value) > CONTENT_LENGTH_MAX_DIGITS: - raise LocalProtocolError("bad Content-Length") - if seen_content_length is None: - seen_content_length = value - new_headers.append((raw_name, name, value)) - elif seen_content_length != value: - raise LocalProtocolError("conflicting Content-Length headers") - elif name == b"transfer-encoding": - # "A server that receives a request message with a transfer coding - # it does not understand SHOULD respond with 501 (Not - # Implemented)." - # https://tools.ietf.org/html/rfc7230#section-3.3.1 - if saw_transfer_encoding: - raise LocalProtocolError( - "multiple Transfer-Encoding headers", error_status_hint=501 - ) - # "All transfer-coding names are case-insensitive" - # -- https://tools.ietf.org/html/rfc7230#section-4 - value = value.lower() - if value != b"chunked": - raise LocalProtocolError( - "Only Transfer-Encoding: chunked is supported", - error_status_hint=501, - ) - saw_transfer_encoding = True - new_headers.append((raw_name, name, value)) - else: - new_headers.append((raw_name, name, value)) - return Headers(new_headers) - - -def get_comma_header(headers: Headers, name: bytes) -> List[bytes]: - # Should only be used for headers whose value is a list of - # comma-separated, case-insensitive values. - # - # The header name `name` is expected to be lower-case bytes. - # - # Connection: meets these criteria (including cast insensitivity). - # - # Content-Length: technically is just a single value (1*DIGIT), but the - # standard makes reference to implementations that do multiple values, and - # using this doesn't hurt. Ditto, case insensitivity doesn't things either - # way. - # - # Transfer-Encoding: is more complex (allows for quoted strings), so - # splitting on , is actually wrong. For example, this is legal: - # - # Transfer-Encoding: foo; options="1,2", chunked - # - # and should be parsed as - # - # foo; options="1,2" - # chunked - # - # but this naive function will parse it as - # - # foo; options="1 - # 2" - # chunked - # - # However, this is okay because the only thing we are going to do with - # any Transfer-Encoding is reject ones that aren't just "chunked", so - # both of these will be treated the same anyway. - # - # Expect: the only legal value is the literal string - # "100-continue". Splitting on commas is harmless. Case insensitive. - # - out: List[bytes] = [] - for _, found_name, found_raw_value in headers._full_items: - if found_name == name: - found_raw_value = found_raw_value.lower() - for found_split_value in found_raw_value.split(b","): - found_split_value = found_split_value.strip() - if found_split_value: - out.append(found_split_value) - return out - - -def set_comma_header(headers: Headers, name: bytes, new_values: List[bytes]) -> Headers: - # The header name `name` is expected to be lower-case bytes. - # - # Note that when we store the header we use title casing for the header - # names, in order to match the conventional HTTP header style. - # - # Simply calling `.title()` is a blunt approach, but it's correct - # here given the cases where we're using `set_comma_header`... - # - # Connection, Content-Length, Transfer-Encoding. - new_headers: List[Tuple[bytes, bytes]] = [] - for found_raw_name, found_name, found_raw_value in headers._full_items: - if found_name != name: - new_headers.append((found_raw_name, found_raw_value)) - for new_value in new_values: - new_headers.append((name.title(), new_value)) - return normalize_and_validate(new_headers) - - -def has_expect_100_continue(request: "Request") -> bool: - # https://tools.ietf.org/html/rfc7231#section-5.1.1 - # "A server that receives a 100-continue expectation in an HTTP/1.0 request - # MUST ignore that expectation." - if request.http_version < b"1.1": - return False - expect = get_comma_header(request.headers, b"expect") - return b"100-continue" in expect diff --git a/.env/Lib/site-packages/h11/_readers.py b/.env/Lib/site-packages/h11/_readers.py deleted file mode 100644 index 576804c..0000000 --- a/.env/Lib/site-packages/h11/_readers.py +++ /dev/null @@ -1,250 +0,0 @@ -# Code to read HTTP data -# -# Strategy: each reader is a callable which takes a ReceiveBuffer object, and -# either: -# 1) consumes some of it and returns an Event -# 2) raises a LocalProtocolError (for consistency -- e.g. we call validate() -# and it might raise a LocalProtocolError, so simpler just to always use -# this) -# 3) returns None, meaning "I need more data" -# -# If they have a .read_eof attribute, then this will be called if an EOF is -# received -- but this is optional. Either way, the actual ConnectionClosed -# event will be generated afterwards. -# -# READERS is a dict describing how to pick a reader. It maps states to either: -# - a reader -# - or, for body readers, a dict of per-framing reader factories - -import re -from typing import Any, Callable, Dict, Iterable, NoReturn, Optional, Tuple, Type, Union - -from ._abnf import chunk_header, header_field, request_line, status_line -from ._events import Data, EndOfMessage, InformationalResponse, Request, Response -from ._receivebuffer import ReceiveBuffer -from ._state import ( - CLIENT, - CLOSED, - DONE, - IDLE, - MUST_CLOSE, - SEND_BODY, - SEND_RESPONSE, - SERVER, -) -from ._util import LocalProtocolError, RemoteProtocolError, Sentinel, validate - -__all__ = ["READERS"] - -header_field_re = re.compile(header_field.encode("ascii")) -obs_fold_re = re.compile(rb"[ \t]+") - - -def _obsolete_line_fold(lines: Iterable[bytes]) -> Iterable[bytes]: - it = iter(lines) - last: Optional[bytes] = None - for line in it: - match = obs_fold_re.match(line) - if match: - if last is None: - raise LocalProtocolError("continuation line at start of headers") - if not isinstance(last, bytearray): - # Cast to a mutable type, avoiding copy on append to ensure O(n) time - last = bytearray(last) - last += b" " - last += line[match.end() :] - else: - if last is not None: - yield last - last = line - if last is not None: - yield last - - -def _decode_header_lines( - lines: Iterable[bytes], -) -> Iterable[Tuple[bytes, bytes]]: - for line in _obsolete_line_fold(lines): - matches = validate(header_field_re, line, "illegal header line: {!r}", line) - yield (matches["field_name"], matches["field_value"]) - - -request_line_re = re.compile(request_line.encode("ascii")) - - -def maybe_read_from_IDLE_client(buf: ReceiveBuffer) -> Optional[Request]: - lines = buf.maybe_extract_lines() - if lines is None: - if buf.is_next_line_obviously_invalid_request_line(): - raise LocalProtocolError("illegal request line") - return None - if not lines: - raise LocalProtocolError("no request line received") - matches = validate( - request_line_re, lines[0], "illegal request line: {!r}", lines[0] - ) - return Request( - headers=list(_decode_header_lines(lines[1:])), _parsed=True, **matches - ) - - -status_line_re = re.compile(status_line.encode("ascii")) - - -def maybe_read_from_SEND_RESPONSE_server( - buf: ReceiveBuffer, -) -> Union[InformationalResponse, Response, None]: - lines = buf.maybe_extract_lines() - if lines is None: - if buf.is_next_line_obviously_invalid_request_line(): - raise LocalProtocolError("illegal request line") - return None - if not lines: - raise LocalProtocolError("no response line received") - matches = validate(status_line_re, lines[0], "illegal status line: {!r}", lines[0]) - http_version = ( - b"1.1" if matches["http_version"] is None else matches["http_version"] - ) - reason = b"" if matches["reason"] is None else matches["reason"] - status_code = int(matches["status_code"]) - class_: Union[Type[InformationalResponse], Type[Response]] = ( - InformationalResponse if status_code < 200 else Response - ) - return class_( - headers=list(_decode_header_lines(lines[1:])), - _parsed=True, - status_code=status_code, - reason=reason, - http_version=http_version, - ) - - -class ContentLengthReader: - def __init__(self, length: int) -> None: - self._length = length - self._remaining = length - - def __call__(self, buf: ReceiveBuffer) -> Union[Data, EndOfMessage, None]: - if self._remaining == 0: - return EndOfMessage() - data = buf.maybe_extract_at_most(self._remaining) - if data is None: - return None - self._remaining -= len(data) - return Data(data=data) - - def read_eof(self) -> NoReturn: - raise RemoteProtocolError( - "peer closed connection without sending complete message body " - "(received {} bytes, expected {})".format( - self._length - self._remaining, self._length - ) - ) - - -chunk_header_re = re.compile(chunk_header.encode("ascii")) - - -class ChunkedReader: - def __init__(self) -> None: - self._bytes_in_chunk = 0 - # After reading a chunk, we have to throw away the trailing \r\n. - # This tracks the bytes that we need to match and throw away. - self._bytes_to_discard = b"" - self._reading_trailer = False - - def __call__(self, buf: ReceiveBuffer) -> Union[Data, EndOfMessage, None]: - if self._reading_trailer: - lines = buf.maybe_extract_lines() - if lines is None: - return None - return EndOfMessage(headers=list(_decode_header_lines(lines))) - if self._bytes_to_discard: - data = buf.maybe_extract_at_most(len(self._bytes_to_discard)) - if data is None: - return None - if data != self._bytes_to_discard[: len(data)]: - raise LocalProtocolError( - f"malformed chunk footer: {data!r} (expected {self._bytes_to_discard!r})" - ) - self._bytes_to_discard = self._bytes_to_discard[len(data) :] - if self._bytes_to_discard: - return None - # else, fall through and read some more - assert self._bytes_to_discard == b"" - if self._bytes_in_chunk == 0: - # We need to refill our chunk count - chunk_header = buf.maybe_extract_next_line() - if chunk_header is None: - return None - matches = validate( - chunk_header_re, - chunk_header, - "illegal chunk header: {!r}", - chunk_header, - ) - # XX FIXME: we discard chunk extensions. Does anyone care? - self._bytes_in_chunk = int(matches["chunk_size"], base=16) - if self._bytes_in_chunk == 0: - self._reading_trailer = True - return self(buf) - chunk_start = True - else: - chunk_start = False - assert self._bytes_in_chunk > 0 - data = buf.maybe_extract_at_most(self._bytes_in_chunk) - if data is None: - return None - self._bytes_in_chunk -= len(data) - if self._bytes_in_chunk == 0: - self._bytes_to_discard = b"\r\n" - chunk_end = True - else: - chunk_end = False - return Data(data=data, chunk_start=chunk_start, chunk_end=chunk_end) - - def read_eof(self) -> NoReturn: - raise RemoteProtocolError( - "peer closed connection without sending complete message body " - "(incomplete chunked read)" - ) - - -class Http10Reader: - def __call__(self, buf: ReceiveBuffer) -> Optional[Data]: - data = buf.maybe_extract_at_most(999999999) - if data is None: - return None - return Data(data=data) - - def read_eof(self) -> EndOfMessage: - return EndOfMessage() - - -def expect_nothing(buf: ReceiveBuffer) -> None: - if buf: - raise LocalProtocolError("Got data when expecting EOF") - return None - - -ReadersType = Dict[ - Union[Type[Sentinel], Tuple[Type[Sentinel], Type[Sentinel]]], - Union[Callable[..., Any], Dict[str, Callable[..., Any]]], -] - -READERS: ReadersType = { - (CLIENT, IDLE): maybe_read_from_IDLE_client, - (SERVER, IDLE): maybe_read_from_SEND_RESPONSE_server, - (SERVER, SEND_RESPONSE): maybe_read_from_SEND_RESPONSE_server, - (CLIENT, DONE): expect_nothing, - (CLIENT, MUST_CLOSE): expect_nothing, - (CLIENT, CLOSED): expect_nothing, - (SERVER, DONE): expect_nothing, - (SERVER, MUST_CLOSE): expect_nothing, - (SERVER, CLOSED): expect_nothing, - SEND_BODY: { - "chunked": ChunkedReader, - "content-length": ContentLengthReader, - "http/1.0": Http10Reader, - }, -} diff --git a/.env/Lib/site-packages/h11/_receivebuffer.py b/.env/Lib/site-packages/h11/_receivebuffer.py deleted file mode 100644 index e5c4e08..0000000 --- a/.env/Lib/site-packages/h11/_receivebuffer.py +++ /dev/null @@ -1,153 +0,0 @@ -import re -import sys -from typing import List, Optional, Union - -__all__ = ["ReceiveBuffer"] - - -# Operations we want to support: -# - find next \r\n or \r\n\r\n (\n or \n\n are also acceptable), -# or wait until there is one -# - read at-most-N bytes -# Goals: -# - on average, do this fast -# - worst case, do this in O(n) where n is the number of bytes processed -# Plan: -# - store bytearray, offset, how far we've searched for a separator token -# - use the how-far-we've-searched data to avoid rescanning -# - while doing a stream of uninterrupted processing, advance offset instead -# of constantly copying -# WARNING: -# - I haven't benchmarked or profiled any of this yet. -# -# Note that starting in Python 3.4, deleting the initial n bytes from a -# bytearray is amortized O(n), thanks to some excellent work by Antoine -# Martin: -# -# https://bugs.python.org/issue19087 -# -# This means that if we only supported 3.4+, we could get rid of the code here -# involving self._start and self.compress, because it's doing exactly the same -# thing that bytearray now does internally. -# -# BUT unfortunately, we still support 2.7, and reading short segments out of a -# long buffer MUST be O(bytes read) to avoid DoS issues, so we can't actually -# delete this code. Yet: -# -# https://pythonclock.org/ -# -# (Two things to double-check first though: make sure PyPy also has the -# optimization, and benchmark to make sure it's a win, since we do have a -# slightly clever thing where we delay calling compress() until we've -# processed a whole event, which could in theory be slightly more efficient -# than the internal bytearray support.) -blank_line_regex = re.compile(b"\n\r?\n", re.MULTILINE) - - -class ReceiveBuffer: - def __init__(self) -> None: - self._data = bytearray() - self._next_line_search = 0 - self._multiple_lines_search = 0 - - def __iadd__(self, byteslike: Union[bytes, bytearray]) -> "ReceiveBuffer": - self._data += byteslike - return self - - def __bool__(self) -> bool: - return bool(len(self)) - - def __len__(self) -> int: - return len(self._data) - - # for @property unprocessed_data - def __bytes__(self) -> bytes: - return bytes(self._data) - - def _extract(self, count: int) -> bytearray: - # extracting an initial slice of the data buffer and return it - out = self._data[:count] - del self._data[:count] - - self._next_line_search = 0 - self._multiple_lines_search = 0 - - return out - - def maybe_extract_at_most(self, count: int) -> Optional[bytearray]: - """ - Extract a fixed number of bytes from the buffer. - """ - out = self._data[:count] - if not out: - return None - - return self._extract(count) - - def maybe_extract_next_line(self) -> Optional[bytearray]: - """ - Extract the first line, if it is completed in the buffer. - """ - # Only search in buffer space that we've not already looked at. - search_start_index = max(0, self._next_line_search - 1) - partial_idx = self._data.find(b"\r\n", search_start_index) - - if partial_idx == -1: - self._next_line_search = len(self._data) - return None - - # + 2 is to compensate len(b"\r\n") - idx = partial_idx + 2 - - return self._extract(idx) - - def maybe_extract_lines(self) -> Optional[List[bytearray]]: - """ - Extract everything up to the first blank line, and return a list of lines. - """ - # Handle the case where we have an immediate empty line. - if self._data[:1] == b"\n": - self._extract(1) - return [] - - if self._data[:2] == b"\r\n": - self._extract(2) - return [] - - # Only search in buffer space that we've not already looked at. - match = blank_line_regex.search(self._data, self._multiple_lines_search) - if match is None: - self._multiple_lines_search = max(0, len(self._data) - 2) - return None - - # Truncate the buffer and return it. - idx = match.span(0)[-1] - out = self._extract(idx) - lines = out.split(b"\n") - - for line in lines: - if line.endswith(b"\r"): - del line[-1] - - assert lines[-2] == lines[-1] == b"" - - del lines[-2:] - - return lines - - # In theory we should wait until `\r\n` before starting to validate - # incoming data. However it's interesting to detect (very) invalid data - # early given they might not even contain `\r\n` at all (hence only - # timeout will get rid of them). - # This is not a 100% effective detection but more of a cheap sanity check - # allowing for early abort in some useful cases. - # This is especially interesting when peer is messing up with HTTPS and - # sent us a TLS stream where we were expecting plain HTTP given all - # versions of TLS so far start handshake with a 0x16 message type code. - def is_next_line_obviously_invalid_request_line(self) -> bool: - try: - # HTTP header line must not contain non-printable characters - # and should not start with a space - return self._data[0] < 0x21 - except IndexError: - return False diff --git a/.env/Lib/site-packages/h11/_state.py b/.env/Lib/site-packages/h11/_state.py deleted file mode 100644 index 3ad444b..0000000 --- a/.env/Lib/site-packages/h11/_state.py +++ /dev/null @@ -1,365 +0,0 @@ -################################################################ -# The core state machine -################################################################ -# -# Rule 1: everything that affects the state machine and state transitions must -# live here in this file. As much as possible goes into the table-based -# representation, but for the bits that don't quite fit, the actual code and -# state must nonetheless live here. -# -# Rule 2: this file does not know about what role we're playing; it only knows -# about HTTP request/response cycles in the abstract. This ensures that we -# don't cheat and apply different rules to local and remote parties. -# -# -# Theory of operation -# =================== -# -# Possibly the simplest way to think about this is that we actually have 5 -# different state machines here. Yes, 5. These are: -# -# 1) The client state, with its complicated automaton (see the docs) -# 2) The server state, with its complicated automaton (see the docs) -# 3) The keep-alive state, with possible states {True, False} -# 4) The SWITCH_CONNECT state, with possible states {False, True} -# 5) The SWITCH_UPGRADE state, with possible states {False, True} -# -# For (3)-(5), the first state listed is the initial state. -# -# (1)-(3) are stored explicitly in member variables. The last -# two are stored implicitly in the pending_switch_proposals set as: -# (state of 4) == (_SWITCH_CONNECT in pending_switch_proposals) -# (state of 5) == (_SWITCH_UPGRADE in pending_switch_proposals) -# -# And each of these machines has two different kinds of transitions: -# -# a) Event-triggered -# b) State-triggered -# -# Event triggered is the obvious thing that you'd think it is: some event -# happens, and if it's the right event at the right time then a transition -# happens. But there are somewhat complicated rules for which machines can -# "see" which events. (As a rule of thumb, if a machine "sees" an event, this -# means two things: the event can affect the machine, and if the machine is -# not in a state where it expects that event then it's an error.) These rules -# are: -# -# 1) The client machine sees all h11.events objects emitted by the client. -# -# 2) The server machine sees all h11.events objects emitted by the server. -# -# It also sees the client's Request event. -# -# And sometimes, server events are annotated with a _SWITCH_* event. For -# example, we can have a (Response, _SWITCH_CONNECT) event, which is -# different from a regular Response event. -# -# 3) The keep-alive machine sees the process_keep_alive_disabled() event -# (which is derived from Request/Response events), and this event -# transitions it from True -> False, or from False -> False. There's no way -# to transition back. -# -# 4&5) The _SWITCH_* machines transition from False->True when we get a -# Request that proposes the relevant type of switch (via -# process_client_switch_proposals), and they go from True->False when we -# get a Response that has no _SWITCH_* annotation. -# -# So that's event-triggered transitions. -# -# State-triggered transitions are less standard. What they do here is couple -# the machines together. The way this works is, when certain *joint* -# configurations of states are achieved, then we automatically transition to a -# new *joint* state. So, for example, if we're ever in a joint state with -# -# client: DONE -# keep-alive: False -# -# then the client state immediately transitions to: -# -# client: MUST_CLOSE -# -# This is fundamentally different from an event-based transition, because it -# doesn't matter how we arrived at the {client: DONE, keep-alive: False} state -# -- maybe the client transitioned SEND_BODY -> DONE, or keep-alive -# transitioned True -> False. Either way, once this precondition is satisfied, -# this transition is immediately triggered. -# -# What if two conflicting state-based transitions get enabled at the same -# time? In practice there's only one case where this arises (client DONE -> -# MIGHT_SWITCH_PROTOCOL versus DONE -> MUST_CLOSE), and we resolve it by -# explicitly prioritizing the DONE -> MIGHT_SWITCH_PROTOCOL transition. -# -# Implementation -# -------------- -# -# The event-triggered transitions for the server and client machines are all -# stored explicitly in a table. Ditto for the state-triggered transitions that -# involve just the server and client state. -# -# The transitions for the other machines, and the state-triggered transitions -# that involve the other machines, are written out as explicit Python code. -# -# It'd be nice if there were some cleaner way to do all this. This isn't -# *too* terrible, but I feel like it could probably be better. -# -# WARNING -# ------- -# -# The script that generates the state machine diagrams for the docs knows how -# to read out the EVENT_TRIGGERED_TRANSITIONS and STATE_TRIGGERED_TRANSITIONS -# tables. But it can't automatically read the transitions that are written -# directly in Python code. So if you touch those, you need to also update the -# script to keep it in sync! -from typing import cast, Dict, Optional, Set, Tuple, Type, Union - -from ._events import * -from ._util import LocalProtocolError, Sentinel - -# Everything in __all__ gets re-exported as part of the h11 public API. -__all__ = [ - "CLIENT", - "SERVER", - "IDLE", - "SEND_RESPONSE", - "SEND_BODY", - "DONE", - "MUST_CLOSE", - "CLOSED", - "MIGHT_SWITCH_PROTOCOL", - "SWITCHED_PROTOCOL", - "ERROR", -] - - -class CLIENT(Sentinel, metaclass=Sentinel): - pass - - -class SERVER(Sentinel, metaclass=Sentinel): - pass - - -# States -class IDLE(Sentinel, metaclass=Sentinel): - pass - - -class SEND_RESPONSE(Sentinel, metaclass=Sentinel): - pass - - -class SEND_BODY(Sentinel, metaclass=Sentinel): - pass - - -class DONE(Sentinel, metaclass=Sentinel): - pass - - -class MUST_CLOSE(Sentinel, metaclass=Sentinel): - pass - - -class CLOSED(Sentinel, metaclass=Sentinel): - pass - - -class ERROR(Sentinel, metaclass=Sentinel): - pass - - -# Switch types -class MIGHT_SWITCH_PROTOCOL(Sentinel, metaclass=Sentinel): - pass - - -class SWITCHED_PROTOCOL(Sentinel, metaclass=Sentinel): - pass - - -class _SWITCH_UPGRADE(Sentinel, metaclass=Sentinel): - pass - - -class _SWITCH_CONNECT(Sentinel, metaclass=Sentinel): - pass - - -EventTransitionType = Dict[ - Type[Sentinel], - Dict[ - Type[Sentinel], - Dict[Union[Type[Event], Tuple[Type[Event], Type[Sentinel]]], Type[Sentinel]], - ], -] - -EVENT_TRIGGERED_TRANSITIONS: EventTransitionType = { - CLIENT: { - IDLE: {Request: SEND_BODY, ConnectionClosed: CLOSED}, - SEND_BODY: {Data: SEND_BODY, EndOfMessage: DONE}, - DONE: {ConnectionClosed: CLOSED}, - MUST_CLOSE: {ConnectionClosed: CLOSED}, - CLOSED: {ConnectionClosed: CLOSED}, - MIGHT_SWITCH_PROTOCOL: {}, - SWITCHED_PROTOCOL: {}, - ERROR: {}, - }, - SERVER: { - IDLE: { - ConnectionClosed: CLOSED, - Response: SEND_BODY, - # Special case: server sees client Request events, in this form - (Request, CLIENT): SEND_RESPONSE, - }, - SEND_RESPONSE: { - InformationalResponse: SEND_RESPONSE, - Response: SEND_BODY, - (InformationalResponse, _SWITCH_UPGRADE): SWITCHED_PROTOCOL, - (Response, _SWITCH_CONNECT): SWITCHED_PROTOCOL, - }, - SEND_BODY: {Data: SEND_BODY, EndOfMessage: DONE}, - DONE: {ConnectionClosed: CLOSED}, - MUST_CLOSE: {ConnectionClosed: CLOSED}, - CLOSED: {ConnectionClosed: CLOSED}, - SWITCHED_PROTOCOL: {}, - ERROR: {}, - }, -} - -StateTransitionType = Dict[ - Tuple[Type[Sentinel], Type[Sentinel]], Dict[Type[Sentinel], Type[Sentinel]] -] - -# NB: there are also some special-case state-triggered transitions hard-coded -# into _fire_state_triggered_transitions below. -STATE_TRIGGERED_TRANSITIONS: StateTransitionType = { - # (Client state, Server state) -> new states - # Protocol negotiation - (MIGHT_SWITCH_PROTOCOL, SWITCHED_PROTOCOL): {CLIENT: SWITCHED_PROTOCOL}, - # Socket shutdown - (CLOSED, DONE): {SERVER: MUST_CLOSE}, - (CLOSED, IDLE): {SERVER: MUST_CLOSE}, - (ERROR, DONE): {SERVER: MUST_CLOSE}, - (DONE, CLOSED): {CLIENT: MUST_CLOSE}, - (IDLE, CLOSED): {CLIENT: MUST_CLOSE}, - (DONE, ERROR): {CLIENT: MUST_CLOSE}, -} - - -class ConnectionState: - def __init__(self) -> None: - # Extra bits of state that don't quite fit into the state model. - - # If this is False then it enables the automatic DONE -> MUST_CLOSE - # transition. Don't set this directly; call .keep_alive_disabled() - self.keep_alive = True - - # This is a subset of {UPGRADE, CONNECT}, containing the proposals - # made by the client for switching protocols. - self.pending_switch_proposals: Set[Type[Sentinel]] = set() - - self.states: Dict[Type[Sentinel], Type[Sentinel]] = {CLIENT: IDLE, SERVER: IDLE} - - def process_error(self, role: Type[Sentinel]) -> None: - self.states[role] = ERROR - self._fire_state_triggered_transitions() - - def process_keep_alive_disabled(self) -> None: - self.keep_alive = False - self._fire_state_triggered_transitions() - - def process_client_switch_proposal(self, switch_event: Type[Sentinel]) -> None: - self.pending_switch_proposals.add(switch_event) - self._fire_state_triggered_transitions() - - def process_event( - self, - role: Type[Sentinel], - event_type: Type[Event], - server_switch_event: Optional[Type[Sentinel]] = None, - ) -> None: - _event_type: Union[Type[Event], Tuple[Type[Event], Type[Sentinel]]] = event_type - if server_switch_event is not None: - assert role is SERVER - if server_switch_event not in self.pending_switch_proposals: - raise LocalProtocolError( - "Received server _SWITCH_UPGRADE event without a pending proposal" - ) - _event_type = (event_type, server_switch_event) - if server_switch_event is None and _event_type is Response: - self.pending_switch_proposals = set() - self._fire_event_triggered_transitions(role, _event_type) - # Special case: the server state does get to see Request - # events. - if _event_type is Request: - assert role is CLIENT - self._fire_event_triggered_transitions(SERVER, (Request, CLIENT)) - self._fire_state_triggered_transitions() - - def _fire_event_triggered_transitions( - self, - role: Type[Sentinel], - event_type: Union[Type[Event], Tuple[Type[Event], Type[Sentinel]]], - ) -> None: - state = self.states[role] - try: - new_state = EVENT_TRIGGERED_TRANSITIONS[role][state][event_type] - except KeyError: - event_type = cast(Type[Event], event_type) - raise LocalProtocolError( - "can't handle event type {} when role={} and state={}".format( - event_type.__name__, role, self.states[role] - ) - ) from None - self.states[role] = new_state - - def _fire_state_triggered_transitions(self) -> None: - # We apply these rules repeatedly until converging on a fixed point - while True: - start_states = dict(self.states) - - # It could happen that both these special-case transitions are - # enabled at the same time: - # - # DONE -> MIGHT_SWITCH_PROTOCOL - # DONE -> MUST_CLOSE - # - # For example, this will always be true of a HTTP/1.0 client - # requesting CONNECT. If this happens, the protocol switch takes - # priority. From there the client will either go to - # SWITCHED_PROTOCOL, in which case it's none of our business when - # they close the connection, or else the server will deny the - # request, in which case the client will go back to DONE and then - # from there to MUST_CLOSE. - if self.pending_switch_proposals: - if self.states[CLIENT] is DONE: - self.states[CLIENT] = MIGHT_SWITCH_PROTOCOL - - if not self.pending_switch_proposals: - if self.states[CLIENT] is MIGHT_SWITCH_PROTOCOL: - self.states[CLIENT] = DONE - - if not self.keep_alive: - for role in (CLIENT, SERVER): - if self.states[role] is DONE: - self.states[role] = MUST_CLOSE - - # Tabular state-triggered transitions - joint_state = (self.states[CLIENT], self.states[SERVER]) - changes = STATE_TRIGGERED_TRANSITIONS.get(joint_state, {}) - self.states.update(changes) - - if self.states == start_states: - # Fixed point reached - return - - def start_next_cycle(self) -> None: - if self.states != {CLIENT: DONE, SERVER: DONE}: - raise LocalProtocolError( - f"not in a reusable state. self.states={self.states}" - ) - # Can't reach DONE/DONE with any of these active, but still, let's be - # sure. - assert self.keep_alive - assert not self.pending_switch_proposals - self.states = {CLIENT: IDLE, SERVER: IDLE} diff --git a/.env/Lib/site-packages/h11/_util.py b/.env/Lib/site-packages/h11/_util.py deleted file mode 100644 index 6718445..0000000 --- a/.env/Lib/site-packages/h11/_util.py +++ /dev/null @@ -1,135 +0,0 @@ -from typing import Any, Dict, NoReturn, Pattern, Tuple, Type, TypeVar, Union - -__all__ = [ - "ProtocolError", - "LocalProtocolError", - "RemoteProtocolError", - "validate", - "bytesify", -] - - -class ProtocolError(Exception): - """Exception indicating a violation of the HTTP/1.1 protocol. - - This as an abstract base class, with two concrete base classes: - :exc:`LocalProtocolError`, which indicates that you tried to do something - that HTTP/1.1 says is illegal, and :exc:`RemoteProtocolError`, which - indicates that the remote peer tried to do something that HTTP/1.1 says is - illegal. See :ref:`error-handling` for details. - - In addition to the normal :exc:`Exception` features, it has one attribute: - - .. attribute:: error_status_hint - - This gives a suggestion as to what status code a server might use if - this error occurred as part of a request. - - For a :exc:`RemoteProtocolError`, this is useful as a suggestion for - how you might want to respond to a misbehaving peer, if you're - implementing a server. - - For a :exc:`LocalProtocolError`, this can be taken as a suggestion for - how your peer might have responded to *you* if h11 had allowed you to - continue. - - The default is 400 Bad Request, a generic catch-all for protocol - violations. - - """ - - def __init__(self, msg: str, error_status_hint: int = 400) -> None: - if type(self) is ProtocolError: - raise TypeError("tried to directly instantiate ProtocolError") - Exception.__init__(self, msg) - self.error_status_hint = error_status_hint - - -# Strategy: there are a number of public APIs where a LocalProtocolError can -# be raised (send(), all the different event constructors, ...), and only one -# public API where RemoteProtocolError can be raised -# (receive_data()). Therefore we always raise LocalProtocolError internally, -# and then receive_data will translate this into a RemoteProtocolError. -# -# Internally: -# LocalProtocolError is the generic "ProtocolError". -# Externally: -# LocalProtocolError is for local errors and RemoteProtocolError is for -# remote errors. -class LocalProtocolError(ProtocolError): - def _reraise_as_remote_protocol_error(self) -> NoReturn: - # After catching a LocalProtocolError, use this method to re-raise it - # as a RemoteProtocolError. This method must be called from inside an - # except: block. - # - # An easy way to get an equivalent RemoteProtocolError is just to - # modify 'self' in place. - self.__class__ = RemoteProtocolError # type: ignore - # But the re-raising is somewhat non-trivial -- you might think that - # now that we've modified the in-flight exception object, that just - # doing 'raise' to re-raise it would be enough. But it turns out that - # this doesn't work, because Python tracks the exception type - # (exc_info[0]) separately from the exception object (exc_info[1]), - # and we only modified the latter. So we really do need to re-raise - # the new type explicitly. - # On py3, the traceback is part of the exception object, so our - # in-place modification preserved it and we can just re-raise: - raise self - - -class RemoteProtocolError(ProtocolError): - pass - - -def validate( - regex: Pattern[bytes], data: bytes, msg: str = "malformed data", *format_args: Any -) -> Dict[str, bytes]: - match = regex.fullmatch(data) - if not match: - if format_args: - msg = msg.format(*format_args) - raise LocalProtocolError(msg) - return match.groupdict() - - -# Sentinel values -# -# - Inherit identity-based comparison and hashing from object -# - Have a nice repr -# - Have a *bonus property*: type(sentinel) is sentinel -# -# The bonus property is useful if you want to take the return value from -# next_event() and do some sort of dispatch based on type(event). - -_T_Sentinel = TypeVar("_T_Sentinel", bound="Sentinel") - - -class Sentinel(type): - def __new__( - cls: Type[_T_Sentinel], - name: str, - bases: Tuple[type, ...], - namespace: Dict[str, Any], - **kwds: Any - ) -> _T_Sentinel: - assert bases == (Sentinel,) - v = super().__new__(cls, name, bases, namespace, **kwds) - v.__class__ = v # type: ignore - return v - - def __repr__(self) -> str: - return self.__name__ - - -# Used for methods, request targets, HTTP versions, header names, and header -# values. Accepts ascii-strings, or bytes/bytearray/memoryview/..., and always -# returns bytes. -def bytesify(s: Union[bytes, bytearray, memoryview, int, str]) -> bytes: - # Fast-path: - if type(s) is bytes: - return s - if isinstance(s, str): - s = s.encode("ascii") - if isinstance(s, int): - raise TypeError("expected bytes-like object, not int") - return bytes(s) diff --git a/.env/Lib/site-packages/h11/_version.py b/.env/Lib/site-packages/h11/_version.py deleted file mode 100644 index 76e7327..0000000 --- a/.env/Lib/site-packages/h11/_version.py +++ /dev/null @@ -1,16 +0,0 @@ -# This file must be kept very simple, because it is consumed from several -# places -- it is imported by h11/__init__.py, execfile'd by setup.py, etc. - -# We use a simple scheme: -# 1.0.0 -> 1.0.0+dev -> 1.1.0 -> 1.1.0+dev -# where the +dev versions are never released into the wild, they're just what -# we stick into the VCS in between releases. -# -# This is compatible with PEP 440: -# http://legacy.python.org/dev/peps/pep-0440/ -# via the use of the "local suffix" "+dev", which is disallowed on index -# servers and causes 1.0.0+dev to sort after plain 1.0.0, which is what we -# want. (Contrast with the special suffix 1.0.0.dev, which sorts *before* -# 1.0.0.) - -__version__ = "0.16.0" diff --git a/.env/Lib/site-packages/h11/_writers.py b/.env/Lib/site-packages/h11/_writers.py deleted file mode 100644 index 939cdb9..0000000 --- a/.env/Lib/site-packages/h11/_writers.py +++ /dev/null @@ -1,145 +0,0 @@ -# Code to read HTTP data -# -# Strategy: each writer takes an event + a write-some-bytes function, which is -# calls. -# -# WRITERS is a dict describing how to pick a reader. It maps states to either: -# - a writer -# - or, for body writers, a dict of framin-dependent writer factories - -from typing import Any, Callable, Dict, List, Tuple, Type, Union - -from ._events import Data, EndOfMessage, Event, InformationalResponse, Request, Response -from ._headers import Headers -from ._state import CLIENT, IDLE, SEND_BODY, SEND_RESPONSE, SERVER -from ._util import LocalProtocolError, Sentinel - -__all__ = ["WRITERS"] - -Writer = Callable[[bytes], Any] - - -def write_headers(headers: Headers, write: Writer) -> None: - # "Since the Host field-value is critical information for handling a - # request, a user agent SHOULD generate Host as the first header field - # following the request-line." - RFC 7230 - raw_items = headers._full_items - for raw_name, name, value in raw_items: - if name == b"host": - write(b"%s: %s\r\n" % (raw_name, value)) - for raw_name, name, value in raw_items: - if name != b"host": - write(b"%s: %s\r\n" % (raw_name, value)) - write(b"\r\n") - - -def write_request(request: Request, write: Writer) -> None: - if request.http_version != b"1.1": - raise LocalProtocolError("I only send HTTP/1.1") - write(b"%s %s HTTP/1.1\r\n" % (request.method, request.target)) - write_headers(request.headers, write) - - -# Shared between InformationalResponse and Response -def write_any_response( - response: Union[InformationalResponse, Response], write: Writer -) -> None: - if response.http_version != b"1.1": - raise LocalProtocolError("I only send HTTP/1.1") - status_bytes = str(response.status_code).encode("ascii") - # We don't bother sending ascii status messages like "OK"; they're - # optional and ignored by the protocol. (But the space after the numeric - # status code is mandatory.) - # - # XX FIXME: could at least make an effort to pull out the status message - # from stdlib's http.HTTPStatus table. Or maybe just steal their enums - # (either by import or copy/paste). We already accept them as status codes - # since they're of type IntEnum < int. - write(b"HTTP/1.1 %s %s\r\n" % (status_bytes, response.reason)) - write_headers(response.headers, write) - - -class BodyWriter: - def __call__(self, event: Event, write: Writer) -> None: - if type(event) is Data: - self.send_data(event.data, write) - elif type(event) is EndOfMessage: - self.send_eom(event.headers, write) - else: # pragma: no cover - assert False - - def send_data(self, data: bytes, write: Writer) -> None: - pass - - def send_eom(self, headers: Headers, write: Writer) -> None: - pass - - -# -# These are all careful not to do anything to 'data' except call len(data) and -# write(data). This allows us to transparently pass-through funny objects, -# like placeholder objects referring to files on disk that will be sent via -# sendfile(2). -# -class ContentLengthWriter(BodyWriter): - def __init__(self, length: int) -> None: - self._length = length - - def send_data(self, data: bytes, write: Writer) -> None: - self._length -= len(data) - if self._length < 0: - raise LocalProtocolError("Too much data for declared Content-Length") - write(data) - - def send_eom(self, headers: Headers, write: Writer) -> None: - if self._length != 0: - raise LocalProtocolError("Too little data for declared Content-Length") - if headers: - raise LocalProtocolError("Content-Length and trailers don't mix") - - -class ChunkedWriter(BodyWriter): - def send_data(self, data: bytes, write: Writer) -> None: - # if we encoded 0-length data in the naive way, it would look like an - # end-of-message. - if not data: - return - write(b"%x\r\n" % len(data)) - write(data) - write(b"\r\n") - - def send_eom(self, headers: Headers, write: Writer) -> None: - write(b"0\r\n") - write_headers(headers, write) - - -class Http10Writer(BodyWriter): - def send_data(self, data: bytes, write: Writer) -> None: - write(data) - - def send_eom(self, headers: Headers, write: Writer) -> None: - if headers: - raise LocalProtocolError("can't send trailers to HTTP/1.0 client") - # no need to close the socket ourselves, that will be taken care of by - # Connection: close machinery - - -WritersType = Dict[ - Union[Tuple[Type[Sentinel], Type[Sentinel]], Type[Sentinel]], - Union[ - Dict[str, Type[BodyWriter]], - Callable[[Union[InformationalResponse, Response], Writer], None], - Callable[[Request, Writer], None], - ], -] - -WRITERS: WritersType = { - (CLIENT, IDLE): write_request, - (SERVER, IDLE): write_any_response, - (SERVER, SEND_RESPONSE): write_any_response, - SEND_BODY: { - "chunked": ChunkedWriter, - "content-length": ContentLengthWriter, - "http/1.0": Http10Writer, - }, -} diff --git a/.env/Lib/site-packages/h11/py.typed b/.env/Lib/site-packages/h11/py.typed deleted file mode 100644 index f5642f7..0000000 --- a/.env/Lib/site-packages/h11/py.typed +++ /dev/null @@ -1 +0,0 @@ -Marker diff --git a/.env/Lib/site-packages/idna-3.11.dist-info/INSTALLER b/.env/Lib/site-packages/idna-3.11.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/idna-3.11.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/idna-3.11.dist-info/METADATA b/.env/Lib/site-packages/idna-3.11.dist-info/METADATA deleted file mode 100644 index 7a4a4b7..0000000 --- a/.env/Lib/site-packages/idna-3.11.dist-info/METADATA +++ /dev/null @@ -1,209 +0,0 @@ -Metadata-Version: 2.4 -Name: idna -Version: 3.11 -Summary: Internationalized Domain Names in Applications (IDNA) -Author-email: Kim Davies -Requires-Python: >=3.8 -Description-Content-Type: text/x-rst -License-Expression: BSD-3-Clause -Classifier: Development Status :: 5 - Production/Stable -Classifier: Intended Audience :: Developers -Classifier: Intended Audience :: System Administrators -Classifier: Operating System :: OS Independent -Classifier: Programming Language :: Python -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: 3.13 -Classifier: Programming Language :: Python :: 3.14 -Classifier: Programming Language :: Python :: Implementation :: CPython -Classifier: Programming Language :: Python :: Implementation :: PyPy -Classifier: Topic :: Internet :: Name Service (DNS) -Classifier: Topic :: Software Development :: Libraries :: Python Modules -Classifier: Topic :: Utilities -License-File: LICENSE.md -Requires-Dist: ruff >= 0.6.2 ; extra == "all" -Requires-Dist: mypy >= 1.11.2 ; extra == "all" -Requires-Dist: pytest >= 8.3.2 ; extra == "all" -Requires-Dist: flake8 >= 7.1.1 ; extra == "all" -Project-URL: Changelog, https://github.com/kjd/idna/blob/master/HISTORY.rst -Project-URL: Issue tracker, https://github.com/kjd/idna/issues -Project-URL: Source, https://github.com/kjd/idna -Provides-Extra: all - -Internationalized Domain Names in Applications (IDNA) -===================================================== - -Support for `Internationalized Domain Names in -Applications (IDNA) `_ -and `Unicode IDNA Compatibility Processing -`_. - -The latest versions of these standards supplied here provide -more comprehensive language coverage and reduce the potential of -allowing domains with known security vulnerabilities. This library -is a suitable replacement for the “encodings.idna” -module that comes with the Python standard library, but which -only supports an older superseded IDNA specification from 2003. - -Basic functions are simply executed: - -.. code-block:: pycon - - >>> import idna - >>> idna.encode('ドメイン.テスト') - b'xn--eckwd4c7c.xn--zckzah' - >>> print(idna.decode('xn--eckwd4c7c.xn--zckzah')) - ドメイン.テスト - - -Installation ------------- - -This package is available for installation from PyPI via the -typical mechanisms, such as: - -.. code-block:: bash - - $ python3 -m pip install idna - - -Usage ------ - -For typical usage, the ``encode`` and ``decode`` functions will take a -domain name argument and perform a conversion to ASCII compatible encoding -(known as A-labels), or to Unicode strings (known as U-labels) -respectively. - -.. code-block:: pycon - - >>> import idna - >>> idna.encode('ドメイン.テスト') - b'xn--eckwd4c7c.xn--zckzah' - >>> print(idna.decode('xn--eckwd4c7c.xn--zckzah')) - ドメイン.テスト - -Conversions can be applied at a per-label basis using the ``ulabel`` or -``alabel`` functions if necessary: - -.. code-block:: pycon - - >>> idna.alabel('测试') - b'xn--0zwm56d' - - -Compatibility Mapping (UTS #46) -+++++++++++++++++++++++++++++++ - -This library provides support for `Unicode IDNA Compatibility -Processing `_ which normalizes input from -different potential ways a user may input a domain prior to performing the IDNA -conversion operations. This functionality, known as a -`mapping `_, is considered by the -specification to be a local user-interface issue distinct from IDNA -conversion functionality. - -For example, “Königsgäßchen” is not a permissible label as *LATIN -CAPITAL LETTER K* is not allowed (nor are capital letters in general). -UTS 46 will convert this into lower case prior to applying the IDNA -conversion. - -.. code-block:: pycon - - >>> import idna - >>> idna.encode('Königsgäßchen') - ... - idna.core.InvalidCodepoint: Codepoint U+004B at position 1 of 'Königsgäßchen' not allowed - >>> idna.encode('Königsgäßchen', uts46=True) - b'xn--knigsgchen-b4a3dun' - >>> print(idna.decode('xn--knigsgchen-b4a3dun')) - königsgäßchen - - -Exceptions ----------- - -All errors raised during the conversion following the specification -should raise an exception derived from the ``idna.IDNAError`` base -class. - -More specific exceptions that may be generated as ``idna.IDNABidiError`` -when the error reflects an illegal combination of left-to-right and -right-to-left characters in a label; ``idna.InvalidCodepoint`` when -a specific codepoint is an illegal character in an IDN label (i.e. -INVALID); and ``idna.InvalidCodepointContext`` when the codepoint is -illegal based on its position in the string (i.e. it is CONTEXTO or CONTEXTJ -but the contextual requirements are not satisfied.) - -Building and Diagnostics ------------------------- - -The IDNA and UTS 46 functionality relies upon pre-calculated lookup -tables for performance. These tables are derived from computing against -eligibility criteria in the respective standards using the command-line -script ``tools/idna-data``. - -This tool will fetch relevant codepoint data from the Unicode repository -and perform the required calculations to identify eligibility. There are -three main modes: - -* ``idna-data make-libdata``. Generates ``idnadata.py`` and - ``uts46data.py``, the pre-calculated lookup tables used for IDNA and - UTS 46 conversions. Implementers who wish to track this library against - a different Unicode version may use this tool to manually generate a - different version of the ``idnadata.py`` and ``uts46data.py`` files. - -* ``idna-data make-table``. Generate a table of the IDNA disposition - (e.g. PVALID, CONTEXTJ, CONTEXTO) in the format found in Appendix - B.1 of RFC 5892 and the pre-computed tables published by `IANA - `_. - -* ``idna-data U+0061``. Prints debugging output on the various - properties associated with an individual Unicode codepoint (in this - case, U+0061), that are used to assess the IDNA and UTS 46 status of a - codepoint. This is helpful in debugging or analysis. - -The tool accepts a number of arguments, described using ``idna-data --h``. Most notably, the ``--version`` argument allows the specification -of the version of Unicode to be used in computing the table data. For -example, ``idna-data --version 9.0.0 make-libdata`` will generate -library data against Unicode 9.0.0. - - -Additional Notes ----------------- - -* **Packages**. The latest tagged release version is published in the - `Python Package Index `_. - -* **Version support**. This library supports Python 3.8 and higher. - As this library serves as a low-level toolkit for a variety of - applications, many of which strive for broad compatibility with older - Python versions, there is no rush to remove older interpreter support. - Support for older versions are likely to be removed from new releases - as automated tests can no longer easily be run, i.e. once the Python - version is officially end-of-life. - -* **Testing**. The library has a test suite based on each rule of the - IDNA specification, as well as tests that are provided as part of the - Unicode Technical Standard 46, `Unicode IDNA Compatibility Processing - `_. - -* **Emoji**. It is an occasional request to support emoji domains in - this library. Encoding of symbols like emoji is expressly prohibited by - the technical standard IDNA 2008 and emoji domains are broadly phased - out across the domain industry due to associated security risks. For - now, applications that need to support these non-compliant labels - may wish to consider trying the encode/decode operation in this library - first, and then falling back to using `encodings.idna`. See `the Github - project `_ for more discussion. - -* **Transitional processing**. Unicode 16.0.0 removed transitional - processing so the `transitional` argument for the encode() method - no longer has any effect and will be removed at a later date. - diff --git a/.env/Lib/site-packages/idna-3.11.dist-info/RECORD b/.env/Lib/site-packages/idna-3.11.dist-info/RECORD deleted file mode 100644 index 1203f10..0000000 --- a/.env/Lib/site-packages/idna-3.11.dist-info/RECORD +++ /dev/null @@ -1,22 +0,0 @@ -idna-3.11.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -idna-3.11.dist-info/METADATA,sha256=fCwSww9SuiN8TIHllFSASUQCW55hAs8dzKnr9RaEEbA,8378 -idna-3.11.dist-info/RECORD,, -idna-3.11.dist-info/WHEEL,sha256=G2gURzTEtmeR8nrdXUJfNiB3VYVxigPQ-bEQujpNiNs,82 -idna-3.11.dist-info/licenses/LICENSE.md,sha256=t6M2q_OwThgOwGXN0W5wXQeeHMehT5EKpukYfza5zYc,1541 -idna/__init__.py,sha256=MPqNDLZbXqGaNdXxAFhiqFPKEQXju2jNQhCey6-5eJM,868 -idna/__pycache__/__init__.cpython-312.pyc,, -idna/__pycache__/codec.cpython-312.pyc,, -idna/__pycache__/compat.cpython-312.pyc,, -idna/__pycache__/core.cpython-312.pyc,, -idna/__pycache__/idnadata.cpython-312.pyc,, -idna/__pycache__/intranges.cpython-312.pyc,, -idna/__pycache__/package_data.cpython-312.pyc,, -idna/__pycache__/uts46data.cpython-312.pyc,, -idna/codec.py,sha256=M2SGWN7cs_6B32QmKTyTN6xQGZeYQgQ2wiX3_DR6loE,3438 -idna/compat.py,sha256=RzLy6QQCdl9784aFhb2EX9EKGCJjg0P3PilGdeXXcx8,316 -idna/core.py,sha256=P26_XVycuMTZ1R2mNK1ZREVzM5mvTzdabBXfyZVU1Lc,13246 -idna/idnadata.py,sha256=SG8jhaGE53iiD6B49pt2pwTv_UvClciWE-N54oR2p4U,79623 -idna/intranges.py,sha256=amUtkdhYcQG8Zr-CoMM_kVRacxkivC1WgxN1b63KKdU,1898 -idna/package_data.py,sha256=_CUavOxobnbyNG2FLyHoN8QHP3QM9W1tKuw7eq9QwBk,21 -idna/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -idna/uts46data.py,sha256=H9J35VkD0F9L9mKOqjeNGd2A-Va6FlPoz6Jz4K7h-ps,243725 diff --git a/.env/Lib/site-packages/idna-3.11.dist-info/WHEEL b/.env/Lib/site-packages/idna-3.11.dist-info/WHEEL deleted file mode 100644 index d8b9936..0000000 --- a/.env/Lib/site-packages/idna-3.11.dist-info/WHEEL +++ /dev/null @@ -1,4 +0,0 @@ -Wheel-Version: 1.0 -Generator: flit 3.12.0 -Root-Is-Purelib: true -Tag: py3-none-any diff --git a/.env/Lib/site-packages/idna-3.11.dist-info/licenses/LICENSE.md b/.env/Lib/site-packages/idna-3.11.dist-info/licenses/LICENSE.md deleted file mode 100644 index 256ba90..0000000 --- a/.env/Lib/site-packages/idna-3.11.dist-info/licenses/LICENSE.md +++ /dev/null @@ -1,31 +0,0 @@ -BSD 3-Clause License - -Copyright (c) 2013-2025, Kim Davies and contributors. -All rights reserved. - -Redistribution and use in source and binary forms, with or without -modification, are permitted provided that the following conditions are -met: - -1. Redistributions of source code must retain the above copyright - notice, this list of conditions and the following disclaimer. - -2. Redistributions in binary form must reproduce the above copyright - notice, this list of conditions and the following disclaimer in the - documentation and/or other materials provided with the distribution. - -3. Neither the name of the copyright holder nor the names of its - contributors may be used to endorse or promote products derived from - this software without specific prior written permission. - -THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED -TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR -PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF -LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING -NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS -SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/.env/Lib/site-packages/idna/__init__.py b/.env/Lib/site-packages/idna/__init__.py deleted file mode 100644 index cfdc030..0000000 --- a/.env/Lib/site-packages/idna/__init__.py +++ /dev/null @@ -1,45 +0,0 @@ -from .core import ( - IDNABidiError, - IDNAError, - InvalidCodepoint, - InvalidCodepointContext, - alabel, - check_bidi, - check_hyphen_ok, - check_initial_combiner, - check_label, - check_nfc, - decode, - encode, - ulabel, - uts46_remap, - valid_contextj, - valid_contexto, - valid_label_length, - valid_string_length, -) -from .intranges import intranges_contain -from .package_data import __version__ - -__all__ = [ - "__version__", - "IDNABidiError", - "IDNAError", - "InvalidCodepoint", - "InvalidCodepointContext", - "alabel", - "check_bidi", - "check_hyphen_ok", - "check_initial_combiner", - "check_label", - "check_nfc", - "decode", - "encode", - "intranges_contain", - "ulabel", - "uts46_remap", - "valid_contextj", - "valid_contexto", - "valid_label_length", - "valid_string_length", -] diff --git a/.env/Lib/site-packages/idna/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/idna/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 9d5beeefbd64a6917372f9fd9075ee248d656c55..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 875 zcmbu7zmL-}6vyo}O`E3a@AeoFVnHIq9SdSW2nhRIsU%){QNl#JHV`upWd>x0l-)5{9ex9 zI4_Oy4x}K38Q>5>?jVOZP=mY3w=Iy`=I`sguB*;A#Y z(!O&h{}OD<;_f}HC?~n@RV5m+s*tlQ zzSN_7zdDICDwgF+kLy@bDkhfuUiNR@fO}mLy^uO}_Gllm5v|1IU zyE#eLWJz&KN>VkcTNslp!}tR@H${QV=kxpV%fehm{6ePk<_R%_5^PQ5d>5avL`k!^ zWeF|fBPzD>Gq#8eR?@@0auKsskUJRL@nd|H?=IU9xJ)-0eW-4j1ok-=hX&sp5W)** b44n(`H-6p*F!};+{cxv&^QNV)n-jkPtyk|S diff --git a/.env/Lib/site-packages/idna/__pycache__/codec.cpython-312.pyc b/.env/Lib/site-packages/idna/__pycache__/codec.cpython-312.pyc deleted file mode 100644 index a51979b8ed65f5fea6c64c8b7d611ab231991923..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4965 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 diff --git a/.env/Lib/site-packages/idna/__pycache__/compat.cpython-312.pyc b/.env/Lib/site-packages/idna/__pycache__/compat.cpython-312.pyc deleted file mode 100644 index f85f925983d6593b60e03e242bd7aec0c490c725..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 879 zcmb7CJ!lj`6rS0?`%C;glmv-vW3iaTpaevsa72rP+=3=)jz!q)yu>BDJImWmV8ud2 zY;4lyO50##Z)0zBVz&qu7LxAZtWx=AlM9wQFz@f1dGp@)=F9YSlhOM0`E9tu8T+Z6 z$uiE2@iQ_GnP7rPEarV4YkiF~p$R?GW20}xX5S>=085xq+ct#t#^~3{*PsTCZEcqB z6JilkBcv(nz=bIq1X>KN=Opd7_6|w{bQb1`zQfV{tw;(Dl zgnSQ^K%JT_oeZ_dGb9dKevZ9lAI|J`cEnwWV~tK6ajcWd6l9;ctM|Rl3^4PaN+Io@ zA3TR7+yyT!yj~cfBy2wnAoCU>+4t7MEiVgmSV;X~*WaOVC=&m6Amh}}7t^AwZOG30 zYPX9QD7{5{JVoLa``)^IaP!@Rp)oTipZ$LW*rIFHg35_1=V=2tO=j8Hki?!M3a^l_ z#)}t~xTH{?x&QW_;1j}Lb4PPu%y%ET9sADG(mh*9$m~Ss_Ff)FVFq^M$B-hVc8W5W zfs7npwq2Qb<1~U8k{rZiM2S>#n~M0m+WO=!UL!;il_^0aHRpym*1k1+L!(!f?o5?d zUZi2NgKEO4erIU}65*5wHP2GdaxBfQB8RMEy=*AkNA!z|jp~)7Q!XmHPU?!1!J1s)s*^_r&`E diff --git a/.env/Lib/site-packages/idna/__pycache__/core.cpython-312.pyc b/.env/Lib/site-packages/idna/__pycache__/core.cpython-312.pyc deleted file mode 100644 index 3babbe67aff0a35ac40a893a5201fb59ce2f21f8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16132 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 diff --git a/.env/Lib/site-packages/idna/__pycache__/package_data.cpython-312.pyc b/.env/Lib/site-packages/idna/__pycache__/package_data.cpython-312.pyc deleted file mode 100644 index 47a6eb27ed570d7eaad5c892d1c3a01934ec344e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 206 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 H532tkdX6 Tuple[bytes, int]: - if errors != "strict": - raise IDNAError('Unsupported error handling "{}"'.format(errors)) - - if not data: - return b"", 0 - - return encode(data), len(data) - - def decode(self, data: bytes, errors: str = "strict") -> Tuple[str, int]: - if errors != "strict": - raise IDNAError('Unsupported error handling "{}"'.format(errors)) - - if not data: - return "", 0 - - return decode(data), len(data) - - -class IncrementalEncoder(codecs.BufferedIncrementalEncoder): - def _buffer_encode(self, data: str, errors: str, final: bool) -> Tuple[bytes, int]: - if errors != "strict": - raise IDNAError('Unsupported error handling "{}"'.format(errors)) - - if not data: - return b"", 0 - - labels = _unicode_dots_re.split(data) - trailing_dot = b"" - if labels: - if not labels[-1]: - trailing_dot = b"." - del labels[-1] - elif not final: - # Keep potentially unfinished label until the next call - del labels[-1] - if labels: - trailing_dot = b"." - - result = [] - size = 0 - for label in labels: - result.append(alabel(label)) - if size: - size += 1 - size += len(label) - - # Join with U+002E - result_bytes = b".".join(result) + trailing_dot - size += len(trailing_dot) - return result_bytes, size - - -class IncrementalDecoder(codecs.BufferedIncrementalDecoder): - def _buffer_decode(self, data: Any, errors: str, final: bool) -> Tuple[str, int]: - if errors != "strict": - raise IDNAError('Unsupported error handling "{}"'.format(errors)) - - if not data: - return ("", 0) - - if not isinstance(data, str): - data = str(data, "ascii") - - labels = _unicode_dots_re.split(data) - trailing_dot = "" - if labels: - if not labels[-1]: - trailing_dot = "." - del labels[-1] - elif not final: - # Keep potentially unfinished label until the next call - del labels[-1] - if labels: - trailing_dot = "." - - result = [] - size = 0 - for label in labels: - result.append(ulabel(label)) - if size: - size += 1 - size += len(label) - - result_str = ".".join(result) + trailing_dot - size += len(trailing_dot) - return (result_str, size) - - -class StreamWriter(Codec, codecs.StreamWriter): - pass - - -class StreamReader(Codec, codecs.StreamReader): - pass - - -def search_function(name: str) -> Optional[codecs.CodecInfo]: - if name != "idna2008": - return None - return codecs.CodecInfo( - name=name, - encode=Codec().encode, - decode=Codec().decode, # type: ignore - incrementalencoder=IncrementalEncoder, - incrementaldecoder=IncrementalDecoder, - streamwriter=StreamWriter, - streamreader=StreamReader, - ) - - -codecs.register(search_function) diff --git a/.env/Lib/site-packages/idna/compat.py b/.env/Lib/site-packages/idna/compat.py deleted file mode 100644 index 1df9f2a..0000000 --- a/.env/Lib/site-packages/idna/compat.py +++ /dev/null @@ -1,15 +0,0 @@ -from typing import Any, Union - -from .core import decode, encode - - -def ToASCII(label: str) -> bytes: - return encode(label) - - -def ToUnicode(label: Union[bytes, bytearray]) -> str: - return decode(label) - - -def nameprep(s: Any) -> None: - raise NotImplementedError("IDNA 2008 does not utilise nameprep protocol") diff --git a/.env/Lib/site-packages/idna/core.py b/.env/Lib/site-packages/idna/core.py deleted file mode 100644 index 8177bf7..0000000 --- a/.env/Lib/site-packages/idna/core.py +++ /dev/null @@ -1,437 +0,0 @@ -import bisect -import re -import unicodedata -from typing import Optional, Union - -from . import idnadata -from .intranges import intranges_contain - -_virama_combining_class = 9 -_alabel_prefix = b"xn--" -_unicode_dots_re = re.compile("[\u002e\u3002\uff0e\uff61]") - - -class IDNAError(UnicodeError): - """Base exception for all IDNA-encoding related problems""" - - pass - - -class IDNABidiError(IDNAError): - """Exception when bidirectional requirements are not satisfied""" - - pass - - -class InvalidCodepoint(IDNAError): - """Exception when a disallowed or unallocated codepoint is used""" - - pass - - -class InvalidCodepointContext(IDNAError): - """Exception when the codepoint is not valid in the context it is used""" - - pass - - -def _combining_class(cp: int) -> int: - v = unicodedata.combining(chr(cp)) - if v == 0: - if not unicodedata.name(chr(cp)): - raise ValueError("Unknown character in unicodedata") - return v - - -def _is_script(cp: str, script: str) -> bool: - return intranges_contain(ord(cp), idnadata.scripts[script]) - - -def _punycode(s: str) -> bytes: - return s.encode("punycode") - - -def _unot(s: int) -> str: - return "U+{:04X}".format(s) - - -def valid_label_length(label: Union[bytes, str]) -> bool: - if len(label) > 63: - return False - return True - - -def valid_string_length(label: Union[bytes, str], trailing_dot: bool) -> bool: - if len(label) > (254 if trailing_dot else 253): - return False - return True - - -def check_bidi(label: str, check_ltr: bool = False) -> bool: - # Bidi rules should only be applied if string contains RTL characters - bidi_label = False - for idx, cp in enumerate(label, 1): - direction = unicodedata.bidirectional(cp) - if direction == "": - # String likely comes from a newer version of Unicode - raise IDNABidiError("Unknown directionality in label {} at position {}".format(repr(label), idx)) - if direction in ["R", "AL", "AN"]: - bidi_label = True - if not bidi_label and not check_ltr: - return True - - # Bidi rule 1 - direction = unicodedata.bidirectional(label[0]) - if direction in ["R", "AL"]: - rtl = True - elif direction == "L": - rtl = False - else: - raise IDNABidiError("First codepoint in label {} must be directionality L, R or AL".format(repr(label))) - - valid_ending = False - number_type: Optional[str] = None - for idx, cp in enumerate(label, 1): - direction = unicodedata.bidirectional(cp) - - if rtl: - # Bidi rule 2 - if direction not in [ - "R", - "AL", - "AN", - "EN", - "ES", - "CS", - "ET", - "ON", - "BN", - "NSM", - ]: - raise IDNABidiError("Invalid direction for codepoint at position {} in a right-to-left label".format(idx)) - # Bidi rule 3 - if direction in ["R", "AL", "EN", "AN"]: - valid_ending = True - elif direction != "NSM": - valid_ending = False - # Bidi rule 4 - if direction in ["AN", "EN"]: - if not number_type: - number_type = direction - else: - if number_type != direction: - raise IDNABidiError("Can not mix numeral types in a right-to-left label") - else: - # Bidi rule 5 - if direction not in ["L", "EN", "ES", "CS", "ET", "ON", "BN", "NSM"]: - raise IDNABidiError("Invalid direction for codepoint at position {} in a left-to-right label".format(idx)) - # Bidi rule 6 - if direction in ["L", "EN"]: - valid_ending = True - elif direction != "NSM": - valid_ending = False - - if not valid_ending: - raise IDNABidiError("Label ends with illegal codepoint directionality") - - return True - - -def check_initial_combiner(label: str) -> bool: - if unicodedata.category(label[0])[0] == "M": - raise IDNAError("Label begins with an illegal combining character") - return True - - -def check_hyphen_ok(label: str) -> bool: - if label[2:4] == "--": - raise IDNAError("Label has disallowed hyphens in 3rd and 4th position") - if label[0] == "-" or label[-1] == "-": - raise IDNAError("Label must not start or end with a hyphen") - return True - - -def check_nfc(label: str) -> None: - if unicodedata.normalize("NFC", label) != label: - raise IDNAError("Label must be in Normalization Form C") - - -def valid_contextj(label: str, pos: int) -> bool: - cp_value = ord(label[pos]) - - if cp_value == 0x200C: - if pos > 0: - if _combining_class(ord(label[pos - 1])) == _virama_combining_class: - return True - - ok = False - for i in range(pos - 1, -1, -1): - joining_type = idnadata.joining_types.get(ord(label[i])) - if joining_type == ord("T"): - continue - elif joining_type in [ord("L"), ord("D")]: - ok = True - break - else: - break - - if not ok: - return False - - ok = False - for i in range(pos + 1, len(label)): - joining_type = idnadata.joining_types.get(ord(label[i])) - if joining_type == ord("T"): - continue - elif joining_type in [ord("R"), ord("D")]: - ok = True - break - else: - break - return ok - - if cp_value == 0x200D: - if pos > 0: - if _combining_class(ord(label[pos - 1])) == _virama_combining_class: - return True - return False - - else: - return False - - -def valid_contexto(label: str, pos: int, exception: bool = False) -> bool: - cp_value = ord(label[pos]) - - if cp_value == 0x00B7: - if 0 < pos < len(label) - 1: - if ord(label[pos - 1]) == 0x006C and ord(label[pos + 1]) == 0x006C: - return True - return False - - elif cp_value == 0x0375: - if pos < len(label) - 1 and len(label) > 1: - return _is_script(label[pos + 1], "Greek") - return False - - elif cp_value == 0x05F3 or cp_value == 0x05F4: - if pos > 0: - return _is_script(label[pos - 1], "Hebrew") - return False - - elif cp_value == 0x30FB: - for cp in label: - if cp == "\u30fb": - continue - if _is_script(cp, "Hiragana") or _is_script(cp, "Katakana") or _is_script(cp, "Han"): - return True - return False - - elif 0x660 <= cp_value <= 0x669: - for cp in label: - if 0x6F0 <= ord(cp) <= 0x06F9: - return False - return True - - elif 0x6F0 <= cp_value <= 0x6F9: - for cp in label: - if 0x660 <= ord(cp) <= 0x0669: - return False - return True - - return False - - -def check_label(label: Union[str, bytes, bytearray]) -> None: - if isinstance(label, (bytes, bytearray)): - label = label.decode("utf-8") - if len(label) == 0: - raise IDNAError("Empty Label") - - check_nfc(label) - check_hyphen_ok(label) - check_initial_combiner(label) - - for pos, cp in enumerate(label): - cp_value = ord(cp) - if intranges_contain(cp_value, idnadata.codepoint_classes["PVALID"]): - continue - elif intranges_contain(cp_value, idnadata.codepoint_classes["CONTEXTJ"]): - try: - if not valid_contextj(label, pos): - raise InvalidCodepointContext( - "Joiner {} not allowed at position {} in {}".format(_unot(cp_value), pos + 1, repr(label)) - ) - except ValueError: - raise IDNAError( - "Unknown codepoint adjacent to joiner {} at position {} in {}".format( - _unot(cp_value), pos + 1, repr(label) - ) - ) - elif intranges_contain(cp_value, idnadata.codepoint_classes["CONTEXTO"]): - if not valid_contexto(label, pos): - raise InvalidCodepointContext( - "Codepoint {} not allowed at position {} in {}".format(_unot(cp_value), pos + 1, repr(label)) - ) - else: - raise InvalidCodepoint( - "Codepoint {} at position {} of {} not allowed".format(_unot(cp_value), pos + 1, repr(label)) - ) - - check_bidi(label) - - -def alabel(label: str) -> bytes: - try: - label_bytes = label.encode("ascii") - ulabel(label_bytes) - if not valid_label_length(label_bytes): - raise IDNAError("Label too long") - return label_bytes - except UnicodeEncodeError: - pass - - check_label(label) - label_bytes = _alabel_prefix + _punycode(label) - - if not valid_label_length(label_bytes): - raise IDNAError("Label too long") - - return label_bytes - - -def ulabel(label: Union[str, bytes, bytearray]) -> str: - if not isinstance(label, (bytes, bytearray)): - try: - label_bytes = label.encode("ascii") - except UnicodeEncodeError: - check_label(label) - return label - else: - label_bytes = bytes(label) - - label_bytes = label_bytes.lower() - if label_bytes.startswith(_alabel_prefix): - label_bytes = label_bytes[len(_alabel_prefix) :] - if not label_bytes: - raise IDNAError("Malformed A-label, no Punycode eligible content found") - if label_bytes.decode("ascii")[-1] == "-": - raise IDNAError("A-label must not end with a hyphen") - else: - check_label(label_bytes) - return label_bytes.decode("ascii") - - try: - label = label_bytes.decode("punycode") - except UnicodeError: - raise IDNAError("Invalid A-label") - check_label(label) - return label - - -def uts46_remap(domain: str, std3_rules: bool = True, transitional: bool = False) -> str: - """Re-map the characters in the string according to UTS46 processing.""" - from .uts46data import uts46data - - output = "" - - for pos, char in enumerate(domain): - code_point = ord(char) - try: - uts46row = uts46data[code_point if code_point < 256 else bisect.bisect_left(uts46data, (code_point, "Z")) - 1] - status = uts46row[1] - replacement: Optional[str] = None - if len(uts46row) == 3: - replacement = uts46row[2] - if ( - status == "V" - or (status == "D" and not transitional) - or (status == "3" and not std3_rules and replacement is None) - ): - output += char - elif replacement is not None and ( - status == "M" or (status == "3" and not std3_rules) or (status == "D" and transitional) - ): - output += replacement - elif status != "I": - raise IndexError() - except IndexError: - raise InvalidCodepoint( - "Codepoint {} not allowed at position {} in {}".format(_unot(code_point), pos + 1, repr(domain)) - ) - - return unicodedata.normalize("NFC", output) - - -def encode( - s: Union[str, bytes, bytearray], - strict: bool = False, - uts46: bool = False, - std3_rules: bool = False, - transitional: bool = False, -) -> bytes: - if not isinstance(s, str): - try: - s = str(s, "ascii") - except UnicodeDecodeError: - raise IDNAError("should pass a unicode string to the function rather than a byte string.") - if uts46: - s = uts46_remap(s, std3_rules, transitional) - trailing_dot = False - result = [] - if strict: - labels = s.split(".") - else: - labels = _unicode_dots_re.split(s) - if not labels or labels == [""]: - raise IDNAError("Empty domain") - if labels[-1] == "": - del labels[-1] - trailing_dot = True - for label in labels: - s = alabel(label) - if s: - result.append(s) - else: - raise IDNAError("Empty label") - if trailing_dot: - result.append(b"") - s = b".".join(result) - if not valid_string_length(s, trailing_dot): - raise IDNAError("Domain too long") - return s - - -def decode( - s: Union[str, bytes, bytearray], - strict: bool = False, - uts46: bool = False, - std3_rules: bool = False, -) -> str: - try: - if not isinstance(s, str): - s = str(s, "ascii") - except UnicodeDecodeError: - raise IDNAError("Invalid ASCII in A-label") - if uts46: - s = uts46_remap(s, std3_rules, False) - trailing_dot = False - result = [] - if not strict: - labels = _unicode_dots_re.split(s) - else: - labels = s.split(".") - if not labels or labels == [""]: - raise IDNAError("Empty domain") - if not labels[-1]: - del labels[-1] - trailing_dot = True - for label in labels: - s = ulabel(label) - if s: - result.append(s) - else: - raise IDNAError("Empty label") - if trailing_dot: - result.append("") - return ".".join(result) diff --git a/.env/Lib/site-packages/idna/idnadata.py b/.env/Lib/site-packages/idna/idnadata.py deleted file mode 100644 index ded47ca..0000000 --- a/.env/Lib/site-packages/idna/idnadata.py +++ /dev/null @@ -1,4309 +0,0 @@ -# This file is automatically generated by tools/idna-data - -__version__ = "16.0.0" - -scripts = { - "Greek": ( - 0x37000000374, - 0x37500000378, - 0x37A0000037E, - 0x37F00000380, - 0x38400000385, - 0x38600000387, - 0x3880000038B, - 0x38C0000038D, - 0x38E000003A2, - 0x3A3000003E2, - 0x3F000000400, - 0x1D2600001D2B, - 0x1D5D00001D62, - 0x1D6600001D6B, - 0x1DBF00001DC0, - 0x1F0000001F16, - 0x1F1800001F1E, - 0x1F2000001F46, - 0x1F4800001F4E, - 0x1F5000001F58, - 0x1F5900001F5A, - 0x1F5B00001F5C, - 0x1F5D00001F5E, - 0x1F5F00001F7E, - 0x1F8000001FB5, - 0x1FB600001FC5, - 0x1FC600001FD4, - 0x1FD600001FDC, - 0x1FDD00001FF0, - 0x1FF200001FF5, - 0x1FF600001FFF, - 0x212600002127, - 0xAB650000AB66, - 0x101400001018F, - 0x101A0000101A1, - 0x1D2000001D246, - ), - "Han": ( - 0x2E8000002E9A, - 0x2E9B00002EF4, - 0x2F0000002FD6, - 0x300500003006, - 0x300700003008, - 0x30210000302A, - 0x30380000303C, - 0x340000004DC0, - 0x4E000000A000, - 0xF9000000FA6E, - 0xFA700000FADA, - 0x16FE200016FE4, - 0x16FF000016FF2, - 0x200000002A6E0, - 0x2A7000002B73A, - 0x2B7400002B81E, - 0x2B8200002CEA2, - 0x2CEB00002EBE1, - 0x2EBF00002EE5E, - 0x2F8000002FA1E, - 0x300000003134B, - 0x31350000323B0, - ), - "Hebrew": ( - 0x591000005C8, - 0x5D0000005EB, - 0x5EF000005F5, - 0xFB1D0000FB37, - 0xFB380000FB3D, - 0xFB3E0000FB3F, - 0xFB400000FB42, - 0xFB430000FB45, - 0xFB460000FB50, - ), - "Hiragana": ( - 0x304100003097, - 0x309D000030A0, - 0x1B0010001B120, - 0x1B1320001B133, - 0x1B1500001B153, - 0x1F2000001F201, - ), - "Katakana": ( - 0x30A1000030FB, - 0x30FD00003100, - 0x31F000003200, - 0x32D0000032FF, - 0x330000003358, - 0xFF660000FF70, - 0xFF710000FF9E, - 0x1AFF00001AFF4, - 0x1AFF50001AFFC, - 0x1AFFD0001AFFF, - 0x1B0000001B001, - 0x1B1200001B123, - 0x1B1550001B156, - 0x1B1640001B168, - ), -} -joining_types = { - 0xAD: 84, - 0x300: 84, - 0x301: 84, - 0x302: 84, - 0x303: 84, - 0x304: 84, - 0x305: 84, - 0x306: 84, - 0x307: 84, - 0x308: 84, - 0x309: 84, - 0x30A: 84, - 0x30B: 84, - 0x30C: 84, - 0x30D: 84, - 0x30E: 84, - 0x30F: 84, - 0x310: 84, - 0x311: 84, - 0x312: 84, - 0x313: 84, - 0x314: 84, - 0x315: 84, - 0x316: 84, - 0x317: 84, - 0x318: 84, - 0x319: 84, - 0x31A: 84, - 0x31B: 84, - 0x31C: 84, - 0x31D: 84, - 0x31E: 84, - 0x31F: 84, - 0x320: 84, - 0x321: 84, - 0x322: 84, - 0x323: 84, - 0x324: 84, - 0x325: 84, - 0x326: 84, - 0x327: 84, - 0x328: 84, - 0x329: 84, - 0x32A: 84, - 0x32B: 84, - 0x32C: 84, - 0x32D: 84, - 0x32E: 84, - 0x32F: 84, - 0x330: 84, - 0x331: 84, - 0x332: 84, - 0x333: 84, - 0x334: 84, - 0x335: 84, - 0x336: 84, - 0x337: 84, - 0x338: 84, - 0x339: 84, - 0x33A: 84, - 0x33B: 84, - 0x33C: 84, - 0x33D: 84, - 0x33E: 84, - 0x33F: 84, - 0x340: 84, - 0x341: 84, - 0x342: 84, - 0x343: 84, - 0x344: 84, - 0x345: 84, - 0x346: 84, - 0x347: 84, - 0x348: 84, - 0x349: 84, - 0x34A: 84, - 0x34B: 84, - 0x34C: 84, - 0x34D: 84, - 0x34E: 84, - 0x34F: 84, - 0x350: 84, - 0x351: 84, - 0x352: 84, - 0x353: 84, - 0x354: 84, - 0x355: 84, - 0x356: 84, - 0x357: 84, - 0x358: 84, - 0x359: 84, - 0x35A: 84, - 0x35B: 84, - 0x35C: 84, - 0x35D: 84, - 0x35E: 84, - 0x35F: 84, - 0x360: 84, - 0x361: 84, - 0x362: 84, - 0x363: 84, - 0x364: 84, - 0x365: 84, - 0x366: 84, - 0x367: 84, - 0x368: 84, - 0x369: 84, - 0x36A: 84, - 0x36B: 84, - 0x36C: 84, - 0x36D: 84, - 0x36E: 84, - 0x36F: 84, - 0x483: 84, - 0x484: 84, - 0x485: 84, - 0x486: 84, - 0x487: 84, - 0x488: 84, - 0x489: 84, - 0x591: 84, - 0x592: 84, - 0x593: 84, - 0x594: 84, - 0x595: 84, - 0x596: 84, - 0x597: 84, - 0x598: 84, - 0x599: 84, - 0x59A: 84, - 0x59B: 84, - 0x59C: 84, - 0x59D: 84, - 0x59E: 84, - 0x59F: 84, - 0x5A0: 84, - 0x5A1: 84, - 0x5A2: 84, - 0x5A3: 84, - 0x5A4: 84, - 0x5A5: 84, - 0x5A6: 84, - 0x5A7: 84, - 0x5A8: 84, - 0x5A9: 84, - 0x5AA: 84, - 0x5AB: 84, - 0x5AC: 84, - 0x5AD: 84, - 0x5AE: 84, - 0x5AF: 84, - 0x5B0: 84, - 0x5B1: 84, - 0x5B2: 84, - 0x5B3: 84, - 0x5B4: 84, - 0x5B5: 84, - 0x5B6: 84, - 0x5B7: 84, - 0x5B8: 84, - 0x5B9: 84, - 0x5BA: 84, - 0x5BB: 84, - 0x5BC: 84, - 0x5BD: 84, - 0x5BF: 84, - 0x5C1: 84, - 0x5C2: 84, - 0x5C4: 84, - 0x5C5: 84, - 0x5C7: 84, - 0x610: 84, - 0x611: 84, - 0x612: 84, - 0x613: 84, - 0x614: 84, - 0x615: 84, - 0x616: 84, - 0x617: 84, - 0x618: 84, - 0x619: 84, - 0x61A: 84, - 0x61C: 84, - 0x620: 68, - 0x622: 82, - 0x623: 82, - 0x624: 82, - 0x625: 82, - 0x626: 68, - 0x627: 82, - 0x628: 68, - 0x629: 82, - 0x62A: 68, - 0x62B: 68, - 0x62C: 68, - 0x62D: 68, - 0x62E: 68, - 0x62F: 82, - 0x630: 82, - 0x631: 82, - 0x632: 82, - 0x633: 68, - 0x634: 68, - 0x635: 68, - 0x636: 68, - 0x637: 68, - 0x638: 68, - 0x639: 68, - 0x63A: 68, - 0x63B: 68, - 0x63C: 68, - 0x63D: 68, - 0x63E: 68, - 0x63F: 68, - 0x640: 67, - 0x641: 68, - 0x642: 68, - 0x643: 68, - 0x644: 68, - 0x645: 68, - 0x646: 68, - 0x647: 68, - 0x648: 82, - 0x649: 68, - 0x64A: 68, - 0x64B: 84, - 0x64C: 84, - 0x64D: 84, - 0x64E: 84, - 0x64F: 84, - 0x650: 84, - 0x651: 84, - 0x652: 84, - 0x653: 84, - 0x654: 84, - 0x655: 84, - 0x656: 84, - 0x657: 84, - 0x658: 84, - 0x659: 84, - 0x65A: 84, - 0x65B: 84, - 0x65C: 84, - 0x65D: 84, - 0x65E: 84, - 0x65F: 84, - 0x66E: 68, - 0x66F: 68, - 0x670: 84, - 0x671: 82, - 0x672: 82, - 0x673: 82, - 0x675: 82, - 0x676: 82, - 0x677: 82, - 0x678: 68, - 0x679: 68, - 0x67A: 68, - 0x67B: 68, - 0x67C: 68, - 0x67D: 68, - 0x67E: 68, - 0x67F: 68, - 0x680: 68, - 0x681: 68, - 0x682: 68, - 0x683: 68, - 0x684: 68, - 0x685: 68, - 0x686: 68, - 0x687: 68, - 0x688: 82, - 0x689: 82, - 0x68A: 82, - 0x68B: 82, - 0x68C: 82, - 0x68D: 82, - 0x68E: 82, - 0x68F: 82, - 0x690: 82, - 0x691: 82, - 0x692: 82, - 0x693: 82, - 0x694: 82, - 0x695: 82, - 0x696: 82, - 0x697: 82, - 0x698: 82, - 0x699: 82, - 0x69A: 68, - 0x69B: 68, - 0x69C: 68, - 0x69D: 68, - 0x69E: 68, - 0x69F: 68, - 0x6A0: 68, - 0x6A1: 68, - 0x6A2: 68, - 0x6A3: 68, - 0x6A4: 68, - 0x6A5: 68, - 0x6A6: 68, - 0x6A7: 68, - 0x6A8: 68, - 0x6A9: 68, - 0x6AA: 68, - 0x6AB: 68, - 0x6AC: 68, - 0x6AD: 68, - 0x6AE: 68, - 0x6AF: 68, - 0x6B0: 68, - 0x6B1: 68, - 0x6B2: 68, - 0x6B3: 68, - 0x6B4: 68, - 0x6B5: 68, - 0x6B6: 68, - 0x6B7: 68, - 0x6B8: 68, - 0x6B9: 68, - 0x6BA: 68, - 0x6BB: 68, - 0x6BC: 68, - 0x6BD: 68, - 0x6BE: 68, - 0x6BF: 68, - 0x6C0: 82, - 0x6C1: 68, - 0x6C2: 68, - 0x6C3: 82, - 0x6C4: 82, - 0x6C5: 82, - 0x6C6: 82, - 0x6C7: 82, - 0x6C8: 82, - 0x6C9: 82, - 0x6CA: 82, - 0x6CB: 82, - 0x6CC: 68, - 0x6CD: 82, - 0x6CE: 68, - 0x6CF: 82, - 0x6D0: 68, - 0x6D1: 68, - 0x6D2: 82, - 0x6D3: 82, - 0x6D5: 82, - 0x6D6: 84, - 0x6D7: 84, - 0x6D8: 84, - 0x6D9: 84, - 0x6DA: 84, - 0x6DB: 84, - 0x6DC: 84, - 0x6DF: 84, - 0x6E0: 84, - 0x6E1: 84, - 0x6E2: 84, - 0x6E3: 84, - 0x6E4: 84, - 0x6E7: 84, - 0x6E8: 84, - 0x6EA: 84, - 0x6EB: 84, - 0x6EC: 84, - 0x6ED: 84, - 0x6EE: 82, - 0x6EF: 82, - 0x6FA: 68, - 0x6FB: 68, - 0x6FC: 68, - 0x6FF: 68, - 0x70F: 84, - 0x710: 82, - 0x711: 84, - 0x712: 68, - 0x713: 68, - 0x714: 68, - 0x715: 82, - 0x716: 82, - 0x717: 82, - 0x718: 82, - 0x719: 82, - 0x71A: 68, - 0x71B: 68, - 0x71C: 68, - 0x71D: 68, - 0x71E: 82, - 0x71F: 68, - 0x720: 68, - 0x721: 68, - 0x722: 68, - 0x723: 68, - 0x724: 68, - 0x725: 68, - 0x726: 68, - 0x727: 68, - 0x728: 82, - 0x729: 68, - 0x72A: 82, - 0x72B: 68, - 0x72C: 82, - 0x72D: 68, - 0x72E: 68, - 0x72F: 82, - 0x730: 84, - 0x731: 84, - 0x732: 84, - 0x733: 84, - 0x734: 84, - 0x735: 84, - 0x736: 84, - 0x737: 84, - 0x738: 84, - 0x739: 84, - 0x73A: 84, - 0x73B: 84, - 0x73C: 84, - 0x73D: 84, - 0x73E: 84, - 0x73F: 84, - 0x740: 84, - 0x741: 84, - 0x742: 84, - 0x743: 84, - 0x744: 84, - 0x745: 84, - 0x746: 84, - 0x747: 84, - 0x748: 84, - 0x749: 84, - 0x74A: 84, - 0x74D: 82, - 0x74E: 68, - 0x74F: 68, - 0x750: 68, - 0x751: 68, - 0x752: 68, - 0x753: 68, - 0x754: 68, - 0x755: 68, - 0x756: 68, - 0x757: 68, - 0x758: 68, - 0x759: 82, - 0x75A: 82, - 0x75B: 82, - 0x75C: 68, - 0x75D: 68, - 0x75E: 68, - 0x75F: 68, - 0x760: 68, - 0x761: 68, - 0x762: 68, - 0x763: 68, - 0x764: 68, - 0x765: 68, - 0x766: 68, - 0x767: 68, - 0x768: 68, - 0x769: 68, - 0x76A: 68, - 0x76B: 82, - 0x76C: 82, - 0x76D: 68, - 0x76E: 68, - 0x76F: 68, - 0x770: 68, - 0x771: 82, - 0x772: 68, - 0x773: 82, - 0x774: 82, - 0x775: 68, - 0x776: 68, - 0x777: 68, - 0x778: 82, - 0x779: 82, - 0x77A: 68, - 0x77B: 68, - 0x77C: 68, - 0x77D: 68, - 0x77E: 68, - 0x77F: 68, - 0x7A6: 84, - 0x7A7: 84, - 0x7A8: 84, - 0x7A9: 84, - 0x7AA: 84, - 0x7AB: 84, - 0x7AC: 84, - 0x7AD: 84, - 0x7AE: 84, - 0x7AF: 84, - 0x7B0: 84, - 0x7CA: 68, - 0x7CB: 68, - 0x7CC: 68, - 0x7CD: 68, - 0x7CE: 68, - 0x7CF: 68, - 0x7D0: 68, - 0x7D1: 68, - 0x7D2: 68, - 0x7D3: 68, - 0x7D4: 68, - 0x7D5: 68, - 0x7D6: 68, - 0x7D7: 68, - 0x7D8: 68, - 0x7D9: 68, - 0x7DA: 68, - 0x7DB: 68, - 0x7DC: 68, - 0x7DD: 68, - 0x7DE: 68, - 0x7DF: 68, - 0x7E0: 68, - 0x7E1: 68, - 0x7E2: 68, - 0x7E3: 68, - 0x7E4: 68, - 0x7E5: 68, - 0x7E6: 68, - 0x7E7: 68, - 0x7E8: 68, - 0x7E9: 68, - 0x7EA: 68, - 0x7EB: 84, - 0x7EC: 84, - 0x7ED: 84, - 0x7EE: 84, - 0x7EF: 84, - 0x7F0: 84, - 0x7F1: 84, - 0x7F2: 84, - 0x7F3: 84, - 0x7FA: 67, - 0x7FD: 84, - 0x816: 84, - 0x817: 84, - 0x818: 84, - 0x819: 84, - 0x81B: 84, - 0x81C: 84, - 0x81D: 84, - 0x81E: 84, - 0x81F: 84, - 0x820: 84, - 0x821: 84, - 0x822: 84, - 0x823: 84, - 0x825: 84, - 0x826: 84, - 0x827: 84, - 0x829: 84, - 0x82A: 84, - 0x82B: 84, - 0x82C: 84, - 0x82D: 84, - 0x840: 82, - 0x841: 68, - 0x842: 68, - 0x843: 68, - 0x844: 68, - 0x845: 68, - 0x846: 82, - 0x847: 82, - 0x848: 68, - 0x849: 82, - 0x84A: 68, - 0x84B: 68, - 0x84C: 68, - 0x84D: 68, - 0x84E: 68, - 0x84F: 68, - 0x850: 68, - 0x851: 68, - 0x852: 68, - 0x853: 68, - 0x854: 82, - 0x855: 68, - 0x856: 82, - 0x857: 82, - 0x858: 82, - 0x859: 84, - 0x85A: 84, - 0x85B: 84, - 0x860: 68, - 0x862: 68, - 0x863: 68, - 0x864: 68, - 0x865: 68, - 0x867: 82, - 0x868: 68, - 0x869: 82, - 0x86A: 82, - 0x870: 82, - 0x871: 82, - 0x872: 82, - 0x873: 82, - 0x874: 82, - 0x875: 82, - 0x876: 82, - 0x877: 82, - 0x878: 82, - 0x879: 82, - 0x87A: 82, - 0x87B: 82, - 0x87C: 82, - 0x87D: 82, - 0x87E: 82, - 0x87F: 82, - 0x880: 82, - 0x881: 82, - 0x882: 82, - 0x883: 67, - 0x884: 67, - 0x885: 67, - 0x886: 68, - 0x889: 68, - 0x88A: 68, - 0x88B: 68, - 0x88C: 68, - 0x88D: 68, - 0x88E: 82, - 0x897: 84, - 0x898: 84, - 0x899: 84, - 0x89A: 84, - 0x89B: 84, - 0x89C: 84, - 0x89D: 84, - 0x89E: 84, - 0x89F: 84, - 0x8A0: 68, - 0x8A1: 68, - 0x8A2: 68, - 0x8A3: 68, - 0x8A4: 68, - 0x8A5: 68, - 0x8A6: 68, - 0x8A7: 68, - 0x8A8: 68, - 0x8A9: 68, - 0x8AA: 82, - 0x8AB: 82, - 0x8AC: 82, - 0x8AE: 82, - 0x8AF: 68, - 0x8B0: 68, - 0x8B1: 82, - 0x8B2: 82, - 0x8B3: 68, - 0x8B4: 68, - 0x8B5: 68, - 0x8B6: 68, - 0x8B7: 68, - 0x8B8: 68, - 0x8B9: 82, - 0x8BA: 68, - 0x8BB: 68, - 0x8BC: 68, - 0x8BD: 68, - 0x8BE: 68, - 0x8BF: 68, - 0x8C0: 68, - 0x8C1: 68, - 0x8C2: 68, - 0x8C3: 68, - 0x8C4: 68, - 0x8C5: 68, - 0x8C6: 68, - 0x8C7: 68, - 0x8C8: 68, - 0x8CA: 84, - 0x8CB: 84, - 0x8CC: 84, - 0x8CD: 84, - 0x8CE: 84, - 0x8CF: 84, - 0x8D0: 84, - 0x8D1: 84, - 0x8D2: 84, - 0x8D3: 84, - 0x8D4: 84, - 0x8D5: 84, - 0x8D6: 84, - 0x8D7: 84, - 0x8D8: 84, - 0x8D9: 84, - 0x8DA: 84, - 0x8DB: 84, - 0x8DC: 84, - 0x8DD: 84, - 0x8DE: 84, - 0x8DF: 84, - 0x8E0: 84, - 0x8E1: 84, - 0x8E3: 84, - 0x8E4: 84, - 0x8E5: 84, - 0x8E6: 84, - 0x8E7: 84, - 0x8E8: 84, - 0x8E9: 84, - 0x8EA: 84, - 0x8EB: 84, - 0x8EC: 84, - 0x8ED: 84, - 0x8EE: 84, - 0x8EF: 84, - 0x8F0: 84, - 0x8F1: 84, - 0x8F2: 84, - 0x8F3: 84, - 0x8F4: 84, - 0x8F5: 84, - 0x8F6: 84, - 0x8F7: 84, - 0x8F8: 84, - 0x8F9: 84, - 0x8FA: 84, - 0x8FB: 84, - 0x8FC: 84, - 0x8FD: 84, - 0x8FE: 84, - 0x8FF: 84, - 0x900: 84, - 0x901: 84, - 0x902: 84, - 0x93A: 84, - 0x93C: 84, - 0x941: 84, - 0x942: 84, - 0x943: 84, - 0x944: 84, - 0x945: 84, - 0x946: 84, - 0x947: 84, - 0x948: 84, - 0x94D: 84, - 0x951: 84, - 0x952: 84, - 0x953: 84, - 0x954: 84, - 0x955: 84, - 0x956: 84, - 0x957: 84, - 0x962: 84, - 0x963: 84, - 0x981: 84, - 0x9BC: 84, - 0x9C1: 84, - 0x9C2: 84, - 0x9C3: 84, - 0x9C4: 84, - 0x9CD: 84, - 0x9E2: 84, - 0x9E3: 84, - 0x9FE: 84, - 0xA01: 84, - 0xA02: 84, - 0xA3C: 84, - 0xA41: 84, - 0xA42: 84, - 0xA47: 84, - 0xA48: 84, - 0xA4B: 84, - 0xA4C: 84, - 0xA4D: 84, - 0xA51: 84, - 0xA70: 84, - 0xA71: 84, - 0xA75: 84, - 0xA81: 84, - 0xA82: 84, - 0xABC: 84, - 0xAC1: 84, - 0xAC2: 84, - 0xAC3: 84, - 0xAC4: 84, - 0xAC5: 84, - 0xAC7: 84, - 0xAC8: 84, - 0xACD: 84, - 0xAE2: 84, - 0xAE3: 84, - 0xAFA: 84, - 0xAFB: 84, - 0xAFC: 84, - 0xAFD: 84, - 0xAFE: 84, - 0xAFF: 84, - 0xB01: 84, - 0xB3C: 84, - 0xB3F: 84, - 0xB41: 84, - 0xB42: 84, - 0xB43: 84, - 0xB44: 84, - 0xB4D: 84, - 0xB55: 84, - 0xB56: 84, - 0xB62: 84, - 0xB63: 84, - 0xB82: 84, - 0xBC0: 84, - 0xBCD: 84, - 0xC00: 84, - 0xC04: 84, - 0xC3C: 84, - 0xC3E: 84, - 0xC3F: 84, - 0xC40: 84, - 0xC46: 84, - 0xC47: 84, - 0xC48: 84, - 0xC4A: 84, - 0xC4B: 84, - 0xC4C: 84, - 0xC4D: 84, - 0xC55: 84, - 0xC56: 84, - 0xC62: 84, - 0xC63: 84, - 0xC81: 84, - 0xCBC: 84, - 0xCBF: 84, - 0xCC6: 84, - 0xCCC: 84, - 0xCCD: 84, - 0xCE2: 84, - 0xCE3: 84, - 0xD00: 84, - 0xD01: 84, - 0xD3B: 84, - 0xD3C: 84, - 0xD41: 84, - 0xD42: 84, - 0xD43: 84, - 0xD44: 84, - 0xD4D: 84, - 0xD62: 84, - 0xD63: 84, - 0xD81: 84, - 0xDCA: 84, - 0xDD2: 84, - 0xDD3: 84, - 0xDD4: 84, - 0xDD6: 84, - 0xE31: 84, - 0xE34: 84, - 0xE35: 84, - 0xE36: 84, - 0xE37: 84, - 0xE38: 84, - 0xE39: 84, - 0xE3A: 84, - 0xE47: 84, - 0xE48: 84, - 0xE49: 84, - 0xE4A: 84, - 0xE4B: 84, - 0xE4C: 84, - 0xE4D: 84, - 0xE4E: 84, - 0xEB1: 84, - 0xEB4: 84, - 0xEB5: 84, - 0xEB6: 84, - 0xEB7: 84, - 0xEB8: 84, - 0xEB9: 84, - 0xEBA: 84, - 0xEBB: 84, - 0xEBC: 84, - 0xEC8: 84, - 0xEC9: 84, - 0xECA: 84, - 0xECB: 84, - 0xECC: 84, - 0xECD: 84, - 0xECE: 84, - 0xF18: 84, - 0xF19: 84, - 0xF35: 84, - 0xF37: 84, - 0xF39: 84, - 0xF71: 84, - 0xF72: 84, - 0xF73: 84, - 0xF74: 84, - 0xF75: 84, - 0xF76: 84, - 0xF77: 84, - 0xF78: 84, - 0xF79: 84, - 0xF7A: 84, - 0xF7B: 84, - 0xF7C: 84, - 0xF7D: 84, - 0xF7E: 84, - 0xF80: 84, - 0xF81: 84, - 0xF82: 84, - 0xF83: 84, - 0xF84: 84, - 0xF86: 84, - 0xF87: 84, - 0xF8D: 84, - 0xF8E: 84, - 0xF8F: 84, - 0xF90: 84, - 0xF91: 84, - 0xF92: 84, - 0xF93: 84, - 0xF94: 84, - 0xF95: 84, - 0xF96: 84, - 0xF97: 84, - 0xF99: 84, - 0xF9A: 84, - 0xF9B: 84, - 0xF9C: 84, - 0xF9D: 84, - 0xF9E: 84, - 0xF9F: 84, - 0xFA0: 84, - 0xFA1: 84, - 0xFA2: 84, - 0xFA3: 84, - 0xFA4: 84, - 0xFA5: 84, - 0xFA6: 84, - 0xFA7: 84, - 0xFA8: 84, - 0xFA9: 84, - 0xFAA: 84, - 0xFAB: 84, - 0xFAC: 84, - 0xFAD: 84, - 0xFAE: 84, - 0xFAF: 84, - 0xFB0: 84, - 0xFB1: 84, - 0xFB2: 84, - 0xFB3: 84, - 0xFB4: 84, - 0xFB5: 84, - 0xFB6: 84, - 0xFB7: 84, - 0xFB8: 84, - 0xFB9: 84, - 0xFBA: 84, - 0xFBB: 84, - 0xFBC: 84, - 0xFC6: 84, - 0x102D: 84, - 0x102E: 84, - 0x102F: 84, - 0x1030: 84, - 0x1032: 84, - 0x1033: 84, - 0x1034: 84, - 0x1035: 84, - 0x1036: 84, - 0x1037: 84, - 0x1039: 84, - 0x103A: 84, - 0x103D: 84, - 0x103E: 84, - 0x1058: 84, - 0x1059: 84, - 0x105E: 84, - 0x105F: 84, - 0x1060: 84, - 0x1071: 84, - 0x1072: 84, - 0x1073: 84, - 0x1074: 84, - 0x1082: 84, - 0x1085: 84, - 0x1086: 84, - 0x108D: 84, - 0x109D: 84, - 0x135D: 84, - 0x135E: 84, - 0x135F: 84, - 0x1712: 84, - 0x1713: 84, - 0x1714: 84, - 0x1732: 84, - 0x1733: 84, - 0x1752: 84, - 0x1753: 84, - 0x1772: 84, - 0x1773: 84, - 0x17B4: 84, - 0x17B5: 84, - 0x17B7: 84, - 0x17B8: 84, - 0x17B9: 84, - 0x17BA: 84, - 0x17BB: 84, - 0x17BC: 84, - 0x17BD: 84, - 0x17C6: 84, - 0x17C9: 84, - 0x17CA: 84, - 0x17CB: 84, - 0x17CC: 84, - 0x17CD: 84, - 0x17CE: 84, - 0x17CF: 84, - 0x17D0: 84, - 0x17D1: 84, - 0x17D2: 84, - 0x17D3: 84, - 0x17DD: 84, - 0x1807: 68, - 0x180A: 67, - 0x180B: 84, - 0x180C: 84, - 0x180D: 84, - 0x180F: 84, - 0x1820: 68, - 0x1821: 68, - 0x1822: 68, - 0x1823: 68, - 0x1824: 68, - 0x1825: 68, - 0x1826: 68, - 0x1827: 68, - 0x1828: 68, - 0x1829: 68, - 0x182A: 68, - 0x182B: 68, - 0x182C: 68, - 0x182D: 68, - 0x182E: 68, - 0x182F: 68, - 0x1830: 68, - 0x1831: 68, - 0x1832: 68, - 0x1833: 68, - 0x1834: 68, - 0x1835: 68, - 0x1836: 68, - 0x1837: 68, - 0x1838: 68, - 0x1839: 68, - 0x183A: 68, - 0x183B: 68, - 0x183C: 68, - 0x183D: 68, - 0x183E: 68, - 0x183F: 68, - 0x1840: 68, - 0x1841: 68, - 0x1842: 68, - 0x1843: 68, - 0x1844: 68, - 0x1845: 68, - 0x1846: 68, - 0x1847: 68, - 0x1848: 68, - 0x1849: 68, - 0x184A: 68, - 0x184B: 68, - 0x184C: 68, - 0x184D: 68, - 0x184E: 68, - 0x184F: 68, - 0x1850: 68, - 0x1851: 68, - 0x1852: 68, - 0x1853: 68, - 0x1854: 68, - 0x1855: 68, - 0x1856: 68, - 0x1857: 68, - 0x1858: 68, - 0x1859: 68, - 0x185A: 68, - 0x185B: 68, - 0x185C: 68, - 0x185D: 68, - 0x185E: 68, - 0x185F: 68, - 0x1860: 68, - 0x1861: 68, - 0x1862: 68, - 0x1863: 68, - 0x1864: 68, - 0x1865: 68, - 0x1866: 68, - 0x1867: 68, - 0x1868: 68, - 0x1869: 68, - 0x186A: 68, - 0x186B: 68, - 0x186C: 68, - 0x186D: 68, - 0x186E: 68, - 0x186F: 68, - 0x1870: 68, - 0x1871: 68, - 0x1872: 68, - 0x1873: 68, - 0x1874: 68, - 0x1875: 68, - 0x1876: 68, - 0x1877: 68, - 0x1878: 68, - 0x1885: 84, - 0x1886: 84, - 0x1887: 68, - 0x1888: 68, - 0x1889: 68, - 0x188A: 68, - 0x188B: 68, - 0x188C: 68, - 0x188D: 68, - 0x188E: 68, - 0x188F: 68, - 0x1890: 68, - 0x1891: 68, - 0x1892: 68, - 0x1893: 68, - 0x1894: 68, - 0x1895: 68, - 0x1896: 68, - 0x1897: 68, - 0x1898: 68, - 0x1899: 68, - 0x189A: 68, - 0x189B: 68, - 0x189C: 68, - 0x189D: 68, - 0x189E: 68, - 0x189F: 68, - 0x18A0: 68, - 0x18A1: 68, - 0x18A2: 68, - 0x18A3: 68, - 0x18A4: 68, - 0x18A5: 68, - 0x18A6: 68, - 0x18A7: 68, - 0x18A8: 68, - 0x18A9: 84, - 0x18AA: 68, - 0x1920: 84, - 0x1921: 84, - 0x1922: 84, - 0x1927: 84, - 0x1928: 84, - 0x1932: 84, - 0x1939: 84, - 0x193A: 84, - 0x193B: 84, - 0x1A17: 84, - 0x1A18: 84, - 0x1A1B: 84, - 0x1A56: 84, - 0x1A58: 84, - 0x1A59: 84, - 0x1A5A: 84, - 0x1A5B: 84, - 0x1A5C: 84, - 0x1A5D: 84, - 0x1A5E: 84, - 0x1A60: 84, - 0x1A62: 84, - 0x1A65: 84, - 0x1A66: 84, - 0x1A67: 84, - 0x1A68: 84, - 0x1A69: 84, - 0x1A6A: 84, - 0x1A6B: 84, - 0x1A6C: 84, - 0x1A73: 84, - 0x1A74: 84, - 0x1A75: 84, - 0x1A76: 84, - 0x1A77: 84, - 0x1A78: 84, - 0x1A79: 84, - 0x1A7A: 84, - 0x1A7B: 84, - 0x1A7C: 84, - 0x1A7F: 84, - 0x1AB0: 84, - 0x1AB1: 84, - 0x1AB2: 84, - 0x1AB3: 84, - 0x1AB4: 84, - 0x1AB5: 84, - 0x1AB6: 84, - 0x1AB7: 84, - 0x1AB8: 84, - 0x1AB9: 84, - 0x1ABA: 84, - 0x1ABB: 84, - 0x1ABC: 84, - 0x1ABD: 84, - 0x1ABE: 84, - 0x1ABF: 84, - 0x1AC0: 84, - 0x1AC1: 84, - 0x1AC2: 84, - 0x1AC3: 84, - 0x1AC4: 84, - 0x1AC5: 84, - 0x1AC6: 84, - 0x1AC7: 84, - 0x1AC8: 84, - 0x1AC9: 84, - 0x1ACA: 84, - 0x1ACB: 84, - 0x1ACC: 84, - 0x1ACD: 84, - 0x1ACE: 84, - 0x1B00: 84, - 0x1B01: 84, - 0x1B02: 84, - 0x1B03: 84, - 0x1B34: 84, - 0x1B36: 84, - 0x1B37: 84, - 0x1B38: 84, - 0x1B39: 84, - 0x1B3A: 84, - 0x1B3C: 84, - 0x1B42: 84, - 0x1B6B: 84, - 0x1B6C: 84, - 0x1B6D: 84, - 0x1B6E: 84, - 0x1B6F: 84, - 0x1B70: 84, - 0x1B71: 84, - 0x1B72: 84, - 0x1B73: 84, - 0x1B80: 84, - 0x1B81: 84, - 0x1BA2: 84, - 0x1BA3: 84, - 0x1BA4: 84, - 0x1BA5: 84, - 0x1BA8: 84, - 0x1BA9: 84, - 0x1BAB: 84, - 0x1BAC: 84, - 0x1BAD: 84, - 0x1BE6: 84, - 0x1BE8: 84, - 0x1BE9: 84, - 0x1BED: 84, - 0x1BEF: 84, - 0x1BF0: 84, - 0x1BF1: 84, - 0x1C2C: 84, - 0x1C2D: 84, - 0x1C2E: 84, - 0x1C2F: 84, - 0x1C30: 84, - 0x1C31: 84, - 0x1C32: 84, - 0x1C33: 84, - 0x1C36: 84, - 0x1C37: 84, - 0x1CD0: 84, - 0x1CD1: 84, - 0x1CD2: 84, - 0x1CD4: 84, - 0x1CD5: 84, - 0x1CD6: 84, - 0x1CD7: 84, - 0x1CD8: 84, - 0x1CD9: 84, - 0x1CDA: 84, - 0x1CDB: 84, - 0x1CDC: 84, - 0x1CDD: 84, - 0x1CDE: 84, - 0x1CDF: 84, - 0x1CE0: 84, - 0x1CE2: 84, - 0x1CE3: 84, - 0x1CE4: 84, - 0x1CE5: 84, - 0x1CE6: 84, - 0x1CE7: 84, - 0x1CE8: 84, - 0x1CED: 84, - 0x1CF4: 84, - 0x1CF8: 84, - 0x1CF9: 84, - 0x1DC0: 84, - 0x1DC1: 84, - 0x1DC2: 84, - 0x1DC3: 84, - 0x1DC4: 84, - 0x1DC5: 84, - 0x1DC6: 84, - 0x1DC7: 84, - 0x1DC8: 84, - 0x1DC9: 84, - 0x1DCA: 84, - 0x1DCB: 84, - 0x1DCC: 84, - 0x1DCD: 84, - 0x1DCE: 84, - 0x1DCF: 84, - 0x1DD0: 84, - 0x1DD1: 84, - 0x1DD2: 84, - 0x1DD3: 84, - 0x1DD4: 84, - 0x1DD5: 84, - 0x1DD6: 84, - 0x1DD7: 84, - 0x1DD8: 84, - 0x1DD9: 84, - 0x1DDA: 84, - 0x1DDB: 84, - 0x1DDC: 84, - 0x1DDD: 84, - 0x1DDE: 84, - 0x1DDF: 84, - 0x1DE0: 84, - 0x1DE1: 84, - 0x1DE2: 84, - 0x1DE3: 84, - 0x1DE4: 84, - 0x1DE5: 84, - 0x1DE6: 84, - 0x1DE7: 84, - 0x1DE8: 84, - 0x1DE9: 84, - 0x1DEA: 84, - 0x1DEB: 84, - 0x1DEC: 84, - 0x1DED: 84, - 0x1DEE: 84, - 0x1DEF: 84, - 0x1DF0: 84, - 0x1DF1: 84, - 0x1DF2: 84, - 0x1DF3: 84, - 0x1DF4: 84, - 0x1DF5: 84, - 0x1DF6: 84, - 0x1DF7: 84, - 0x1DF8: 84, - 0x1DF9: 84, - 0x1DFA: 84, - 0x1DFB: 84, - 0x1DFC: 84, - 0x1DFD: 84, - 0x1DFE: 84, - 0x1DFF: 84, - 0x200B: 84, - 0x200D: 67, - 0x200E: 84, - 0x200F: 84, - 0x202A: 84, - 0x202B: 84, - 0x202C: 84, - 0x202D: 84, - 0x202E: 84, - 0x2060: 84, - 0x2061: 84, - 0x2062: 84, - 0x2063: 84, - 0x2064: 84, - 0x206A: 84, - 0x206B: 84, - 0x206C: 84, - 0x206D: 84, - 0x206E: 84, - 0x206F: 84, - 0x20D0: 84, - 0x20D1: 84, - 0x20D2: 84, - 0x20D3: 84, - 0x20D4: 84, - 0x20D5: 84, - 0x20D6: 84, - 0x20D7: 84, - 0x20D8: 84, - 0x20D9: 84, - 0x20DA: 84, - 0x20DB: 84, - 0x20DC: 84, - 0x20DD: 84, - 0x20DE: 84, - 0x20DF: 84, - 0x20E0: 84, - 0x20E1: 84, - 0x20E2: 84, - 0x20E3: 84, - 0x20E4: 84, - 0x20E5: 84, - 0x20E6: 84, - 0x20E7: 84, - 0x20E8: 84, - 0x20E9: 84, - 0x20EA: 84, - 0x20EB: 84, - 0x20EC: 84, - 0x20ED: 84, - 0x20EE: 84, - 0x20EF: 84, - 0x20F0: 84, - 0x2CEF: 84, - 0x2CF0: 84, - 0x2CF1: 84, - 0x2D7F: 84, - 0x2DE0: 84, - 0x2DE1: 84, - 0x2DE2: 84, - 0x2DE3: 84, - 0x2DE4: 84, - 0x2DE5: 84, - 0x2DE6: 84, - 0x2DE7: 84, - 0x2DE8: 84, - 0x2DE9: 84, - 0x2DEA: 84, - 0x2DEB: 84, - 0x2DEC: 84, - 0x2DED: 84, - 0x2DEE: 84, - 0x2DEF: 84, - 0x2DF0: 84, - 0x2DF1: 84, - 0x2DF2: 84, - 0x2DF3: 84, - 0x2DF4: 84, - 0x2DF5: 84, - 0x2DF6: 84, - 0x2DF7: 84, - 0x2DF8: 84, - 0x2DF9: 84, - 0x2DFA: 84, - 0x2DFB: 84, - 0x2DFC: 84, - 0x2DFD: 84, - 0x2DFE: 84, - 0x2DFF: 84, - 0x302A: 84, - 0x302B: 84, - 0x302C: 84, - 0x302D: 84, - 0x3099: 84, - 0x309A: 84, - 0xA66F: 84, - 0xA670: 84, - 0xA671: 84, - 0xA672: 84, - 0xA674: 84, - 0xA675: 84, - 0xA676: 84, - 0xA677: 84, - 0xA678: 84, - 0xA679: 84, - 0xA67A: 84, - 0xA67B: 84, - 0xA67C: 84, - 0xA67D: 84, - 0xA69E: 84, - 0xA69F: 84, - 0xA6F0: 84, - 0xA6F1: 84, - 0xA802: 84, - 0xA806: 84, - 0xA80B: 84, - 0xA825: 84, - 0xA826: 84, - 0xA82C: 84, - 0xA840: 68, - 0xA841: 68, - 0xA842: 68, - 0xA843: 68, - 0xA844: 68, - 0xA845: 68, - 0xA846: 68, - 0xA847: 68, - 0xA848: 68, - 0xA849: 68, - 0xA84A: 68, - 0xA84B: 68, - 0xA84C: 68, - 0xA84D: 68, - 0xA84E: 68, - 0xA84F: 68, - 0xA850: 68, - 0xA851: 68, - 0xA852: 68, - 0xA853: 68, - 0xA854: 68, - 0xA855: 68, - 0xA856: 68, - 0xA857: 68, - 0xA858: 68, - 0xA859: 68, - 0xA85A: 68, - 0xA85B: 68, - 0xA85C: 68, - 0xA85D: 68, - 0xA85E: 68, - 0xA85F: 68, - 0xA860: 68, - 0xA861: 68, - 0xA862: 68, - 0xA863: 68, - 0xA864: 68, - 0xA865: 68, - 0xA866: 68, - 0xA867: 68, - 0xA868: 68, - 0xA869: 68, - 0xA86A: 68, - 0xA86B: 68, - 0xA86C: 68, - 0xA86D: 68, - 0xA86E: 68, - 0xA86F: 68, - 0xA870: 68, - 0xA871: 68, - 0xA872: 76, - 0xA8C4: 84, - 0xA8C5: 84, - 0xA8E0: 84, - 0xA8E1: 84, - 0xA8E2: 84, - 0xA8E3: 84, - 0xA8E4: 84, - 0xA8E5: 84, - 0xA8E6: 84, - 0xA8E7: 84, - 0xA8E8: 84, - 0xA8E9: 84, - 0xA8EA: 84, - 0xA8EB: 84, - 0xA8EC: 84, - 0xA8ED: 84, - 0xA8EE: 84, - 0xA8EF: 84, - 0xA8F0: 84, - 0xA8F1: 84, - 0xA8FF: 84, - 0xA926: 84, - 0xA927: 84, - 0xA928: 84, - 0xA929: 84, - 0xA92A: 84, - 0xA92B: 84, - 0xA92C: 84, - 0xA92D: 84, - 0xA947: 84, - 0xA948: 84, - 0xA949: 84, - 0xA94A: 84, - 0xA94B: 84, - 0xA94C: 84, - 0xA94D: 84, - 0xA94E: 84, - 0xA94F: 84, - 0xA950: 84, - 0xA951: 84, - 0xA980: 84, - 0xA981: 84, - 0xA982: 84, - 0xA9B3: 84, - 0xA9B6: 84, - 0xA9B7: 84, - 0xA9B8: 84, - 0xA9B9: 84, - 0xA9BC: 84, - 0xA9BD: 84, - 0xA9E5: 84, - 0xAA29: 84, - 0xAA2A: 84, - 0xAA2B: 84, - 0xAA2C: 84, - 0xAA2D: 84, - 0xAA2E: 84, - 0xAA31: 84, - 0xAA32: 84, - 0xAA35: 84, - 0xAA36: 84, - 0xAA43: 84, - 0xAA4C: 84, - 0xAA7C: 84, - 0xAAB0: 84, - 0xAAB2: 84, - 0xAAB3: 84, - 0xAAB4: 84, - 0xAAB7: 84, - 0xAAB8: 84, - 0xAABE: 84, - 0xAABF: 84, - 0xAAC1: 84, - 0xAAEC: 84, - 0xAAED: 84, - 0xAAF6: 84, - 0xABE5: 84, - 0xABE8: 84, - 0xABED: 84, - 0xFB1E: 84, - 0xFE00: 84, - 0xFE01: 84, - 0xFE02: 84, - 0xFE03: 84, - 0xFE04: 84, - 0xFE05: 84, - 0xFE06: 84, - 0xFE07: 84, - 0xFE08: 84, - 0xFE09: 84, - 0xFE0A: 84, - 0xFE0B: 84, - 0xFE0C: 84, - 0xFE0D: 84, - 0xFE0E: 84, - 0xFE0F: 84, - 0xFE20: 84, - 0xFE21: 84, - 0xFE22: 84, - 0xFE23: 84, - 0xFE24: 84, - 0xFE25: 84, - 0xFE26: 84, - 0xFE27: 84, - 0xFE28: 84, - 0xFE29: 84, - 0xFE2A: 84, - 0xFE2B: 84, - 0xFE2C: 84, - 0xFE2D: 84, - 0xFE2E: 84, - 0xFE2F: 84, - 0xFEFF: 84, - 0xFFF9: 84, - 0xFFFA: 84, - 0xFFFB: 84, - 0x101FD: 84, - 0x102E0: 84, - 0x10376: 84, - 0x10377: 84, - 0x10378: 84, - 0x10379: 84, - 0x1037A: 84, - 0x10A01: 84, - 0x10A02: 84, - 0x10A03: 84, - 0x10A05: 84, - 0x10A06: 84, - 0x10A0C: 84, - 0x10A0D: 84, - 0x10A0E: 84, - 0x10A0F: 84, - 0x10A38: 84, - 0x10A39: 84, - 0x10A3A: 84, - 0x10A3F: 84, - 0x10AC0: 68, - 0x10AC1: 68, - 0x10AC2: 68, - 0x10AC3: 68, - 0x10AC4: 68, - 0x10AC5: 82, - 0x10AC7: 82, - 0x10AC9: 82, - 0x10ACA: 82, - 0x10ACD: 76, - 0x10ACE: 82, - 0x10ACF: 82, - 0x10AD0: 82, - 0x10AD1: 82, - 0x10AD2: 82, - 0x10AD3: 68, - 0x10AD4: 68, - 0x10AD5: 68, - 0x10AD6: 68, - 0x10AD7: 76, - 0x10AD8: 68, - 0x10AD9: 68, - 0x10ADA: 68, - 0x10ADB: 68, - 0x10ADC: 68, - 0x10ADD: 82, - 0x10ADE: 68, - 0x10ADF: 68, - 0x10AE0: 68, - 0x10AE1: 82, - 0x10AE4: 82, - 0x10AE5: 84, - 0x10AE6: 84, - 0x10AEB: 68, - 0x10AEC: 68, - 0x10AED: 68, - 0x10AEE: 68, - 0x10AEF: 82, - 0x10B80: 68, - 0x10B81: 82, - 0x10B82: 68, - 0x10B83: 82, - 0x10B84: 82, - 0x10B85: 82, - 0x10B86: 68, - 0x10B87: 68, - 0x10B88: 68, - 0x10B89: 82, - 0x10B8A: 68, - 0x10B8B: 68, - 0x10B8C: 82, - 0x10B8D: 68, - 0x10B8E: 82, - 0x10B8F: 82, - 0x10B90: 68, - 0x10B91: 82, - 0x10BA9: 82, - 0x10BAA: 82, - 0x10BAB: 82, - 0x10BAC: 82, - 0x10BAD: 68, - 0x10BAE: 68, - 0x10D00: 76, - 0x10D01: 68, - 0x10D02: 68, - 0x10D03: 68, - 0x10D04: 68, - 0x10D05: 68, - 0x10D06: 68, - 0x10D07: 68, - 0x10D08: 68, - 0x10D09: 68, - 0x10D0A: 68, - 0x10D0B: 68, - 0x10D0C: 68, - 0x10D0D: 68, - 0x10D0E: 68, - 0x10D0F: 68, - 0x10D10: 68, - 0x10D11: 68, - 0x10D12: 68, - 0x10D13: 68, - 0x10D14: 68, - 0x10D15: 68, - 0x10D16: 68, - 0x10D17: 68, - 0x10D18: 68, - 0x10D19: 68, - 0x10D1A: 68, - 0x10D1B: 68, - 0x10D1C: 68, - 0x10D1D: 68, - 0x10D1E: 68, - 0x10D1F: 68, - 0x10D20: 68, - 0x10D21: 68, - 0x10D22: 82, - 0x10D23: 68, - 0x10D24: 84, - 0x10D25: 84, - 0x10D26: 84, - 0x10D27: 84, - 0x10D69: 84, - 0x10D6A: 84, - 0x10D6B: 84, - 0x10D6C: 84, - 0x10D6D: 84, - 0x10EAB: 84, - 0x10EAC: 84, - 0x10EC2: 82, - 0x10EC3: 68, - 0x10EC4: 68, - 0x10EFC: 84, - 0x10EFD: 84, - 0x10EFE: 84, - 0x10EFF: 84, - 0x10F30: 68, - 0x10F31: 68, - 0x10F32: 68, - 0x10F33: 82, - 0x10F34: 68, - 0x10F35: 68, - 0x10F36: 68, - 0x10F37: 68, - 0x10F38: 68, - 0x10F39: 68, - 0x10F3A: 68, - 0x10F3B: 68, - 0x10F3C: 68, - 0x10F3D: 68, - 0x10F3E: 68, - 0x10F3F: 68, - 0x10F40: 68, - 0x10F41: 68, - 0x10F42: 68, - 0x10F43: 68, - 0x10F44: 68, - 0x10F46: 84, - 0x10F47: 84, - 0x10F48: 84, - 0x10F49: 84, - 0x10F4A: 84, - 0x10F4B: 84, - 0x10F4C: 84, - 0x10F4D: 84, - 0x10F4E: 84, - 0x10F4F: 84, - 0x10F50: 84, - 0x10F51: 68, - 0x10F52: 68, - 0x10F53: 68, - 0x10F54: 82, - 0x10F70: 68, - 0x10F71: 68, - 0x10F72: 68, - 0x10F73: 68, - 0x10F74: 82, - 0x10F75: 82, - 0x10F76: 68, - 0x10F77: 68, - 0x10F78: 68, - 0x10F79: 68, - 0x10F7A: 68, - 0x10F7B: 68, - 0x10F7C: 68, - 0x10F7D: 68, - 0x10F7E: 68, - 0x10F7F: 68, - 0x10F80: 68, - 0x10F81: 68, - 0x10F82: 84, - 0x10F83: 84, - 0x10F84: 84, - 0x10F85: 84, - 0x10FB0: 68, - 0x10FB2: 68, - 0x10FB3: 68, - 0x10FB4: 82, - 0x10FB5: 82, - 0x10FB6: 82, - 0x10FB8: 68, - 0x10FB9: 82, - 0x10FBA: 82, - 0x10FBB: 68, - 0x10FBC: 68, - 0x10FBD: 82, - 0x10FBE: 68, - 0x10FBF: 68, - 0x10FC1: 68, - 0x10FC2: 82, - 0x10FC3: 82, - 0x10FC4: 68, - 0x10FC9: 82, - 0x10FCA: 68, - 0x10FCB: 76, - 0x11001: 84, - 0x11038: 84, - 0x11039: 84, - 0x1103A: 84, - 0x1103B: 84, - 0x1103C: 84, - 0x1103D: 84, - 0x1103E: 84, - 0x1103F: 84, - 0x11040: 84, - 0x11041: 84, - 0x11042: 84, - 0x11043: 84, - 0x11044: 84, - 0x11045: 84, - 0x11046: 84, - 0x11070: 84, - 0x11073: 84, - 0x11074: 84, - 0x1107F: 84, - 0x11080: 84, - 0x11081: 84, - 0x110B3: 84, - 0x110B4: 84, - 0x110B5: 84, - 0x110B6: 84, - 0x110B9: 84, - 0x110BA: 84, - 0x110C2: 84, - 0x11100: 84, - 0x11101: 84, - 0x11102: 84, - 0x11127: 84, - 0x11128: 84, - 0x11129: 84, - 0x1112A: 84, - 0x1112B: 84, - 0x1112D: 84, - 0x1112E: 84, - 0x1112F: 84, - 0x11130: 84, - 0x11131: 84, - 0x11132: 84, - 0x11133: 84, - 0x11134: 84, - 0x11173: 84, - 0x11180: 84, - 0x11181: 84, - 0x111B6: 84, - 0x111B7: 84, - 0x111B8: 84, - 0x111B9: 84, - 0x111BA: 84, - 0x111BB: 84, - 0x111BC: 84, - 0x111BD: 84, - 0x111BE: 84, - 0x111C9: 84, - 0x111CA: 84, - 0x111CB: 84, - 0x111CC: 84, - 0x111CF: 84, - 0x1122F: 84, - 0x11230: 84, - 0x11231: 84, - 0x11234: 84, - 0x11236: 84, - 0x11237: 84, - 0x1123E: 84, - 0x11241: 84, - 0x112DF: 84, - 0x112E3: 84, - 0x112E4: 84, - 0x112E5: 84, - 0x112E6: 84, - 0x112E7: 84, - 0x112E8: 84, - 0x112E9: 84, - 0x112EA: 84, - 0x11300: 84, - 0x11301: 84, - 0x1133B: 84, - 0x1133C: 84, - 0x11340: 84, - 0x11366: 84, - 0x11367: 84, - 0x11368: 84, - 0x11369: 84, - 0x1136A: 84, - 0x1136B: 84, - 0x1136C: 84, - 0x11370: 84, - 0x11371: 84, - 0x11372: 84, - 0x11373: 84, - 0x11374: 84, - 0x113BB: 84, - 0x113BC: 84, - 0x113BD: 84, - 0x113BE: 84, - 0x113BF: 84, - 0x113C0: 84, - 0x113CE: 84, - 0x113D0: 84, - 0x113D2: 84, - 0x113E1: 84, - 0x113E2: 84, - 0x11438: 84, - 0x11439: 84, - 0x1143A: 84, - 0x1143B: 84, - 0x1143C: 84, - 0x1143D: 84, - 0x1143E: 84, - 0x1143F: 84, - 0x11442: 84, - 0x11443: 84, - 0x11444: 84, - 0x11446: 84, - 0x1145E: 84, - 0x114B3: 84, - 0x114B4: 84, - 0x114B5: 84, - 0x114B6: 84, - 0x114B7: 84, - 0x114B8: 84, - 0x114BA: 84, - 0x114BF: 84, - 0x114C0: 84, - 0x114C2: 84, - 0x114C3: 84, - 0x115B2: 84, - 0x115B3: 84, - 0x115B4: 84, - 0x115B5: 84, - 0x115BC: 84, - 0x115BD: 84, - 0x115BF: 84, - 0x115C0: 84, - 0x115DC: 84, - 0x115DD: 84, - 0x11633: 84, - 0x11634: 84, - 0x11635: 84, - 0x11636: 84, - 0x11637: 84, - 0x11638: 84, - 0x11639: 84, - 0x1163A: 84, - 0x1163D: 84, - 0x1163F: 84, - 0x11640: 84, - 0x116AB: 84, - 0x116AD: 84, - 0x116B0: 84, - 0x116B1: 84, - 0x116B2: 84, - 0x116B3: 84, - 0x116B4: 84, - 0x116B5: 84, - 0x116B7: 84, - 0x1171D: 84, - 0x1171F: 84, - 0x11722: 84, - 0x11723: 84, - 0x11724: 84, - 0x11725: 84, - 0x11727: 84, - 0x11728: 84, - 0x11729: 84, - 0x1172A: 84, - 0x1172B: 84, - 0x1182F: 84, - 0x11830: 84, - 0x11831: 84, - 0x11832: 84, - 0x11833: 84, - 0x11834: 84, - 0x11835: 84, - 0x11836: 84, - 0x11837: 84, - 0x11839: 84, - 0x1183A: 84, - 0x1193B: 84, - 0x1193C: 84, - 0x1193E: 84, - 0x11943: 84, - 0x119D4: 84, - 0x119D5: 84, - 0x119D6: 84, - 0x119D7: 84, - 0x119DA: 84, - 0x119DB: 84, - 0x119E0: 84, - 0x11A01: 84, - 0x11A02: 84, - 0x11A03: 84, - 0x11A04: 84, - 0x11A05: 84, - 0x11A06: 84, - 0x11A07: 84, - 0x11A08: 84, - 0x11A09: 84, - 0x11A0A: 84, - 0x11A33: 84, - 0x11A34: 84, - 0x11A35: 84, - 0x11A36: 84, - 0x11A37: 84, - 0x11A38: 84, - 0x11A3B: 84, - 0x11A3C: 84, - 0x11A3D: 84, - 0x11A3E: 84, - 0x11A47: 84, - 0x11A51: 84, - 0x11A52: 84, - 0x11A53: 84, - 0x11A54: 84, - 0x11A55: 84, - 0x11A56: 84, - 0x11A59: 84, - 0x11A5A: 84, - 0x11A5B: 84, - 0x11A8A: 84, - 0x11A8B: 84, - 0x11A8C: 84, - 0x11A8D: 84, - 0x11A8E: 84, - 0x11A8F: 84, - 0x11A90: 84, - 0x11A91: 84, - 0x11A92: 84, - 0x11A93: 84, - 0x11A94: 84, - 0x11A95: 84, - 0x11A96: 84, - 0x11A98: 84, - 0x11A99: 84, - 0x11C30: 84, - 0x11C31: 84, - 0x11C32: 84, - 0x11C33: 84, - 0x11C34: 84, - 0x11C35: 84, - 0x11C36: 84, - 0x11C38: 84, - 0x11C39: 84, - 0x11C3A: 84, - 0x11C3B: 84, - 0x11C3C: 84, - 0x11C3D: 84, - 0x11C3F: 84, - 0x11C92: 84, - 0x11C93: 84, - 0x11C94: 84, - 0x11C95: 84, - 0x11C96: 84, - 0x11C97: 84, - 0x11C98: 84, - 0x11C99: 84, - 0x11C9A: 84, - 0x11C9B: 84, - 0x11C9C: 84, - 0x11C9D: 84, - 0x11C9E: 84, - 0x11C9F: 84, - 0x11CA0: 84, - 0x11CA1: 84, - 0x11CA2: 84, - 0x11CA3: 84, - 0x11CA4: 84, - 0x11CA5: 84, - 0x11CA6: 84, - 0x11CA7: 84, - 0x11CAA: 84, - 0x11CAB: 84, - 0x11CAC: 84, - 0x11CAD: 84, - 0x11CAE: 84, - 0x11CAF: 84, - 0x11CB0: 84, - 0x11CB2: 84, - 0x11CB3: 84, - 0x11CB5: 84, - 0x11CB6: 84, - 0x11D31: 84, - 0x11D32: 84, - 0x11D33: 84, - 0x11D34: 84, - 0x11D35: 84, - 0x11D36: 84, - 0x11D3A: 84, - 0x11D3C: 84, - 0x11D3D: 84, - 0x11D3F: 84, - 0x11D40: 84, - 0x11D41: 84, - 0x11D42: 84, - 0x11D43: 84, - 0x11D44: 84, - 0x11D45: 84, - 0x11D47: 84, - 0x11D90: 84, - 0x11D91: 84, - 0x11D95: 84, - 0x11D97: 84, - 0x11EF3: 84, - 0x11EF4: 84, - 0x11F00: 84, - 0x11F01: 84, - 0x11F36: 84, - 0x11F37: 84, - 0x11F38: 84, - 0x11F39: 84, - 0x11F3A: 84, - 0x11F40: 84, - 0x11F42: 84, - 0x11F5A: 84, - 0x13430: 84, - 0x13431: 84, - 0x13432: 84, - 0x13433: 84, - 0x13434: 84, - 0x13435: 84, - 0x13436: 84, - 0x13437: 84, - 0x13438: 84, - 0x13439: 84, - 0x1343A: 84, - 0x1343B: 84, - 0x1343C: 84, - 0x1343D: 84, - 0x1343E: 84, - 0x1343F: 84, - 0x13440: 84, - 0x13447: 84, - 0x13448: 84, - 0x13449: 84, - 0x1344A: 84, - 0x1344B: 84, - 0x1344C: 84, - 0x1344D: 84, - 0x1344E: 84, - 0x1344F: 84, - 0x13450: 84, - 0x13451: 84, - 0x13452: 84, - 0x13453: 84, - 0x13454: 84, - 0x13455: 84, - 0x1611E: 84, - 0x1611F: 84, - 0x16120: 84, - 0x16121: 84, - 0x16122: 84, - 0x16123: 84, - 0x16124: 84, - 0x16125: 84, - 0x16126: 84, - 0x16127: 84, - 0x16128: 84, - 0x16129: 84, - 0x1612D: 84, - 0x1612E: 84, - 0x1612F: 84, - 0x16AF0: 84, - 0x16AF1: 84, - 0x16AF2: 84, - 0x16AF3: 84, - 0x16AF4: 84, - 0x16B30: 84, - 0x16B31: 84, - 0x16B32: 84, - 0x16B33: 84, - 0x16B34: 84, - 0x16B35: 84, - 0x16B36: 84, - 0x16F4F: 84, - 0x16F8F: 84, - 0x16F90: 84, - 0x16F91: 84, - 0x16F92: 84, - 0x16FE4: 84, - 0x1BC9D: 84, - 0x1BC9E: 84, - 0x1BCA0: 84, - 0x1BCA1: 84, - 0x1BCA2: 84, - 0x1BCA3: 84, - 0x1CF00: 84, - 0x1CF01: 84, - 0x1CF02: 84, - 0x1CF03: 84, - 0x1CF04: 84, - 0x1CF05: 84, - 0x1CF06: 84, - 0x1CF07: 84, - 0x1CF08: 84, - 0x1CF09: 84, - 0x1CF0A: 84, - 0x1CF0B: 84, - 0x1CF0C: 84, - 0x1CF0D: 84, - 0x1CF0E: 84, - 0x1CF0F: 84, - 0x1CF10: 84, - 0x1CF11: 84, - 0x1CF12: 84, - 0x1CF13: 84, - 0x1CF14: 84, - 0x1CF15: 84, - 0x1CF16: 84, - 0x1CF17: 84, - 0x1CF18: 84, - 0x1CF19: 84, - 0x1CF1A: 84, - 0x1CF1B: 84, - 0x1CF1C: 84, - 0x1CF1D: 84, - 0x1CF1E: 84, - 0x1CF1F: 84, - 0x1CF20: 84, - 0x1CF21: 84, - 0x1CF22: 84, - 0x1CF23: 84, - 0x1CF24: 84, - 0x1CF25: 84, - 0x1CF26: 84, - 0x1CF27: 84, - 0x1CF28: 84, - 0x1CF29: 84, - 0x1CF2A: 84, - 0x1CF2B: 84, - 0x1CF2C: 84, - 0x1CF2D: 84, - 0x1CF30: 84, - 0x1CF31: 84, - 0x1CF32: 84, - 0x1CF33: 84, - 0x1CF34: 84, - 0x1CF35: 84, - 0x1CF36: 84, - 0x1CF37: 84, - 0x1CF38: 84, - 0x1CF39: 84, - 0x1CF3A: 84, - 0x1CF3B: 84, - 0x1CF3C: 84, - 0x1CF3D: 84, - 0x1CF3E: 84, - 0x1CF3F: 84, - 0x1CF40: 84, - 0x1CF41: 84, - 0x1CF42: 84, - 0x1CF43: 84, - 0x1CF44: 84, - 0x1CF45: 84, - 0x1CF46: 84, - 0x1D167: 84, - 0x1D168: 84, - 0x1D169: 84, - 0x1D173: 84, - 0x1D174: 84, - 0x1D175: 84, - 0x1D176: 84, - 0x1D177: 84, - 0x1D178: 84, - 0x1D179: 84, - 0x1D17A: 84, - 0x1D17B: 84, - 0x1D17C: 84, - 0x1D17D: 84, - 0x1D17E: 84, - 0x1D17F: 84, - 0x1D180: 84, - 0x1D181: 84, - 0x1D182: 84, - 0x1D185: 84, - 0x1D186: 84, - 0x1D187: 84, - 0x1D188: 84, - 0x1D189: 84, - 0x1D18A: 84, - 0x1D18B: 84, - 0x1D1AA: 84, - 0x1D1AB: 84, - 0x1D1AC: 84, - 0x1D1AD: 84, - 0x1D242: 84, - 0x1D243: 84, - 0x1D244: 84, - 0x1DA00: 84, - 0x1DA01: 84, - 0x1DA02: 84, - 0x1DA03: 84, - 0x1DA04: 84, - 0x1DA05: 84, - 0x1DA06: 84, - 0x1DA07: 84, - 0x1DA08: 84, - 0x1DA09: 84, - 0x1DA0A: 84, - 0x1DA0B: 84, - 0x1DA0C: 84, - 0x1DA0D: 84, - 0x1DA0E: 84, - 0x1DA0F: 84, - 0x1DA10: 84, - 0x1DA11: 84, - 0x1DA12: 84, - 0x1DA13: 84, - 0x1DA14: 84, - 0x1DA15: 84, - 0x1DA16: 84, - 0x1DA17: 84, - 0x1DA18: 84, - 0x1DA19: 84, - 0x1DA1A: 84, - 0x1DA1B: 84, - 0x1DA1C: 84, - 0x1DA1D: 84, - 0x1DA1E: 84, - 0x1DA1F: 84, - 0x1DA20: 84, - 0x1DA21: 84, - 0x1DA22: 84, - 0x1DA23: 84, - 0x1DA24: 84, - 0x1DA25: 84, - 0x1DA26: 84, - 0x1DA27: 84, - 0x1DA28: 84, - 0x1DA29: 84, - 0x1DA2A: 84, - 0x1DA2B: 84, - 0x1DA2C: 84, - 0x1DA2D: 84, - 0x1DA2E: 84, - 0x1DA2F: 84, - 0x1DA30: 84, - 0x1DA31: 84, - 0x1DA32: 84, - 0x1DA33: 84, - 0x1DA34: 84, - 0x1DA35: 84, - 0x1DA36: 84, - 0x1DA3B: 84, - 0x1DA3C: 84, - 0x1DA3D: 84, - 0x1DA3E: 84, - 0x1DA3F: 84, - 0x1DA40: 84, - 0x1DA41: 84, - 0x1DA42: 84, - 0x1DA43: 84, - 0x1DA44: 84, - 0x1DA45: 84, - 0x1DA46: 84, - 0x1DA47: 84, - 0x1DA48: 84, - 0x1DA49: 84, - 0x1DA4A: 84, - 0x1DA4B: 84, - 0x1DA4C: 84, - 0x1DA4D: 84, - 0x1DA4E: 84, - 0x1DA4F: 84, - 0x1DA50: 84, - 0x1DA51: 84, - 0x1DA52: 84, - 0x1DA53: 84, - 0x1DA54: 84, - 0x1DA55: 84, - 0x1DA56: 84, - 0x1DA57: 84, - 0x1DA58: 84, - 0x1DA59: 84, - 0x1DA5A: 84, - 0x1DA5B: 84, - 0x1DA5C: 84, - 0x1DA5D: 84, - 0x1DA5E: 84, - 0x1DA5F: 84, - 0x1DA60: 84, - 0x1DA61: 84, - 0x1DA62: 84, - 0x1DA63: 84, - 0x1DA64: 84, - 0x1DA65: 84, - 0x1DA66: 84, - 0x1DA67: 84, - 0x1DA68: 84, - 0x1DA69: 84, - 0x1DA6A: 84, - 0x1DA6B: 84, - 0x1DA6C: 84, - 0x1DA75: 84, - 0x1DA84: 84, - 0x1DA9B: 84, - 0x1DA9C: 84, - 0x1DA9D: 84, - 0x1DA9E: 84, - 0x1DA9F: 84, - 0x1DAA1: 84, - 0x1DAA2: 84, - 0x1DAA3: 84, - 0x1DAA4: 84, - 0x1DAA5: 84, - 0x1DAA6: 84, - 0x1DAA7: 84, - 0x1DAA8: 84, - 0x1DAA9: 84, - 0x1DAAA: 84, - 0x1DAAB: 84, - 0x1DAAC: 84, - 0x1DAAD: 84, - 0x1DAAE: 84, - 0x1DAAF: 84, - 0x1E000: 84, - 0x1E001: 84, - 0x1E002: 84, - 0x1E003: 84, - 0x1E004: 84, - 0x1E005: 84, - 0x1E006: 84, - 0x1E008: 84, - 0x1E009: 84, - 0x1E00A: 84, - 0x1E00B: 84, - 0x1E00C: 84, - 0x1E00D: 84, - 0x1E00E: 84, - 0x1E00F: 84, - 0x1E010: 84, - 0x1E011: 84, - 0x1E012: 84, - 0x1E013: 84, - 0x1E014: 84, - 0x1E015: 84, - 0x1E016: 84, - 0x1E017: 84, - 0x1E018: 84, - 0x1E01B: 84, - 0x1E01C: 84, - 0x1E01D: 84, - 0x1E01E: 84, - 0x1E01F: 84, - 0x1E020: 84, - 0x1E021: 84, - 0x1E023: 84, - 0x1E024: 84, - 0x1E026: 84, - 0x1E027: 84, - 0x1E028: 84, - 0x1E029: 84, - 0x1E02A: 84, - 0x1E08F: 84, - 0x1E130: 84, - 0x1E131: 84, - 0x1E132: 84, - 0x1E133: 84, - 0x1E134: 84, - 0x1E135: 84, - 0x1E136: 84, - 0x1E2AE: 84, - 0x1E2EC: 84, - 0x1E2ED: 84, - 0x1E2EE: 84, - 0x1E2EF: 84, - 0x1E4EC: 84, - 0x1E4ED: 84, - 0x1E4EE: 84, - 0x1E4EF: 84, - 0x1E5EE: 84, - 0x1E5EF: 84, - 0x1E8D0: 84, - 0x1E8D1: 84, - 0x1E8D2: 84, - 0x1E8D3: 84, - 0x1E8D4: 84, - 0x1E8D5: 84, - 0x1E8D6: 84, - 0x1E900: 68, - 0x1E901: 68, - 0x1E902: 68, - 0x1E903: 68, - 0x1E904: 68, - 0x1E905: 68, - 0x1E906: 68, - 0x1E907: 68, - 0x1E908: 68, - 0x1E909: 68, - 0x1E90A: 68, - 0x1E90B: 68, - 0x1E90C: 68, - 0x1E90D: 68, - 0x1E90E: 68, - 0x1E90F: 68, - 0x1E910: 68, - 0x1E911: 68, - 0x1E912: 68, - 0x1E913: 68, - 0x1E914: 68, - 0x1E915: 68, - 0x1E916: 68, - 0x1E917: 68, - 0x1E918: 68, - 0x1E919: 68, - 0x1E91A: 68, - 0x1E91B: 68, - 0x1E91C: 68, - 0x1E91D: 68, - 0x1E91E: 68, - 0x1E91F: 68, - 0x1E920: 68, - 0x1E921: 68, - 0x1E922: 68, - 0x1E923: 68, - 0x1E924: 68, - 0x1E925: 68, - 0x1E926: 68, - 0x1E927: 68, - 0x1E928: 68, - 0x1E929: 68, - 0x1E92A: 68, - 0x1E92B: 68, - 0x1E92C: 68, - 0x1E92D: 68, - 0x1E92E: 68, - 0x1E92F: 68, - 0x1E930: 68, - 0x1E931: 68, - 0x1E932: 68, - 0x1E933: 68, - 0x1E934: 68, - 0x1E935: 68, - 0x1E936: 68, - 0x1E937: 68, - 0x1E938: 68, - 0x1E939: 68, - 0x1E93A: 68, - 0x1E93B: 68, - 0x1E93C: 68, - 0x1E93D: 68, - 0x1E93E: 68, - 0x1E93F: 68, - 0x1E940: 68, - 0x1E941: 68, - 0x1E942: 68, - 0x1E943: 68, - 0x1E944: 84, - 0x1E945: 84, - 0x1E946: 84, - 0x1E947: 84, - 0x1E948: 84, - 0x1E949: 84, - 0x1E94A: 84, - 0x1E94B: 84, - 0xE0001: 84, - 0xE0020: 84, - 0xE0021: 84, - 0xE0022: 84, - 0xE0023: 84, - 0xE0024: 84, - 0xE0025: 84, - 0xE0026: 84, - 0xE0027: 84, - 0xE0028: 84, - 0xE0029: 84, - 0xE002A: 84, - 0xE002B: 84, - 0xE002C: 84, - 0xE002D: 84, - 0xE002E: 84, - 0xE002F: 84, - 0xE0030: 84, - 0xE0031: 84, - 0xE0032: 84, - 0xE0033: 84, - 0xE0034: 84, - 0xE0035: 84, - 0xE0036: 84, - 0xE0037: 84, - 0xE0038: 84, - 0xE0039: 84, - 0xE003A: 84, - 0xE003B: 84, - 0xE003C: 84, - 0xE003D: 84, - 0xE003E: 84, - 0xE003F: 84, - 0xE0040: 84, - 0xE0041: 84, - 0xE0042: 84, - 0xE0043: 84, - 0xE0044: 84, - 0xE0045: 84, - 0xE0046: 84, - 0xE0047: 84, - 0xE0048: 84, - 0xE0049: 84, - 0xE004A: 84, - 0xE004B: 84, - 0xE004C: 84, - 0xE004D: 84, - 0xE004E: 84, - 0xE004F: 84, - 0xE0050: 84, - 0xE0051: 84, - 0xE0052: 84, - 0xE0053: 84, - 0xE0054: 84, - 0xE0055: 84, - 0xE0056: 84, - 0xE0057: 84, - 0xE0058: 84, - 0xE0059: 84, - 0xE005A: 84, - 0xE005B: 84, - 0xE005C: 84, - 0xE005D: 84, - 0xE005E: 84, - 0xE005F: 84, - 0xE0060: 84, - 0xE0061: 84, - 0xE0062: 84, - 0xE0063: 84, - 0xE0064: 84, - 0xE0065: 84, - 0xE0066: 84, - 0xE0067: 84, - 0xE0068: 84, - 0xE0069: 84, - 0xE006A: 84, - 0xE006B: 84, - 0xE006C: 84, - 0xE006D: 84, - 0xE006E: 84, - 0xE006F: 84, - 0xE0070: 84, - 0xE0071: 84, - 0xE0072: 84, - 0xE0073: 84, - 0xE0074: 84, - 0xE0075: 84, - 0xE0076: 84, - 0xE0077: 84, - 0xE0078: 84, - 0xE0079: 84, - 0xE007A: 84, - 0xE007B: 84, - 0xE007C: 84, - 0xE007D: 84, - 0xE007E: 84, - 0xE007F: 84, - 0xE0100: 84, - 0xE0101: 84, - 0xE0102: 84, - 0xE0103: 84, - 0xE0104: 84, - 0xE0105: 84, - 0xE0106: 84, - 0xE0107: 84, - 0xE0108: 84, - 0xE0109: 84, - 0xE010A: 84, - 0xE010B: 84, - 0xE010C: 84, - 0xE010D: 84, - 0xE010E: 84, - 0xE010F: 84, - 0xE0110: 84, - 0xE0111: 84, - 0xE0112: 84, - 0xE0113: 84, - 0xE0114: 84, - 0xE0115: 84, - 0xE0116: 84, - 0xE0117: 84, - 0xE0118: 84, - 0xE0119: 84, - 0xE011A: 84, - 0xE011B: 84, - 0xE011C: 84, - 0xE011D: 84, - 0xE011E: 84, - 0xE011F: 84, - 0xE0120: 84, - 0xE0121: 84, - 0xE0122: 84, - 0xE0123: 84, - 0xE0124: 84, - 0xE0125: 84, - 0xE0126: 84, - 0xE0127: 84, - 0xE0128: 84, - 0xE0129: 84, - 0xE012A: 84, - 0xE012B: 84, - 0xE012C: 84, - 0xE012D: 84, - 0xE012E: 84, - 0xE012F: 84, - 0xE0130: 84, - 0xE0131: 84, - 0xE0132: 84, - 0xE0133: 84, - 0xE0134: 84, - 0xE0135: 84, - 0xE0136: 84, - 0xE0137: 84, - 0xE0138: 84, - 0xE0139: 84, - 0xE013A: 84, - 0xE013B: 84, - 0xE013C: 84, - 0xE013D: 84, - 0xE013E: 84, - 0xE013F: 84, - 0xE0140: 84, - 0xE0141: 84, - 0xE0142: 84, - 0xE0143: 84, - 0xE0144: 84, - 0xE0145: 84, - 0xE0146: 84, - 0xE0147: 84, - 0xE0148: 84, - 0xE0149: 84, - 0xE014A: 84, - 0xE014B: 84, - 0xE014C: 84, - 0xE014D: 84, - 0xE014E: 84, - 0xE014F: 84, - 0xE0150: 84, - 0xE0151: 84, - 0xE0152: 84, - 0xE0153: 84, - 0xE0154: 84, - 0xE0155: 84, - 0xE0156: 84, - 0xE0157: 84, - 0xE0158: 84, - 0xE0159: 84, - 0xE015A: 84, - 0xE015B: 84, - 0xE015C: 84, - 0xE015D: 84, - 0xE015E: 84, - 0xE015F: 84, - 0xE0160: 84, - 0xE0161: 84, - 0xE0162: 84, - 0xE0163: 84, - 0xE0164: 84, - 0xE0165: 84, - 0xE0166: 84, - 0xE0167: 84, - 0xE0168: 84, - 0xE0169: 84, - 0xE016A: 84, - 0xE016B: 84, - 0xE016C: 84, - 0xE016D: 84, - 0xE016E: 84, - 0xE016F: 84, - 0xE0170: 84, - 0xE0171: 84, - 0xE0172: 84, - 0xE0173: 84, - 0xE0174: 84, - 0xE0175: 84, - 0xE0176: 84, - 0xE0177: 84, - 0xE0178: 84, - 0xE0179: 84, - 0xE017A: 84, - 0xE017B: 84, - 0xE017C: 84, - 0xE017D: 84, - 0xE017E: 84, - 0xE017F: 84, - 0xE0180: 84, - 0xE0181: 84, - 0xE0182: 84, - 0xE0183: 84, - 0xE0184: 84, - 0xE0185: 84, - 0xE0186: 84, - 0xE0187: 84, - 0xE0188: 84, - 0xE0189: 84, - 0xE018A: 84, - 0xE018B: 84, - 0xE018C: 84, - 0xE018D: 84, - 0xE018E: 84, - 0xE018F: 84, - 0xE0190: 84, - 0xE0191: 84, - 0xE0192: 84, - 0xE0193: 84, - 0xE0194: 84, - 0xE0195: 84, - 0xE0196: 84, - 0xE0197: 84, - 0xE0198: 84, - 0xE0199: 84, - 0xE019A: 84, - 0xE019B: 84, - 0xE019C: 84, - 0xE019D: 84, - 0xE019E: 84, - 0xE019F: 84, - 0xE01A0: 84, - 0xE01A1: 84, - 0xE01A2: 84, - 0xE01A3: 84, - 0xE01A4: 84, - 0xE01A5: 84, - 0xE01A6: 84, - 0xE01A7: 84, - 0xE01A8: 84, - 0xE01A9: 84, - 0xE01AA: 84, - 0xE01AB: 84, - 0xE01AC: 84, - 0xE01AD: 84, - 0xE01AE: 84, - 0xE01AF: 84, - 0xE01B0: 84, - 0xE01B1: 84, - 0xE01B2: 84, - 0xE01B3: 84, - 0xE01B4: 84, - 0xE01B5: 84, - 0xE01B6: 84, - 0xE01B7: 84, - 0xE01B8: 84, - 0xE01B9: 84, - 0xE01BA: 84, - 0xE01BB: 84, - 0xE01BC: 84, - 0xE01BD: 84, - 0xE01BE: 84, - 0xE01BF: 84, - 0xE01C0: 84, - 0xE01C1: 84, - 0xE01C2: 84, - 0xE01C3: 84, - 0xE01C4: 84, - 0xE01C5: 84, - 0xE01C6: 84, - 0xE01C7: 84, - 0xE01C8: 84, - 0xE01C9: 84, - 0xE01CA: 84, - 0xE01CB: 84, - 0xE01CC: 84, - 0xE01CD: 84, - 0xE01CE: 84, - 0xE01CF: 84, - 0xE01D0: 84, - 0xE01D1: 84, - 0xE01D2: 84, - 0xE01D3: 84, - 0xE01D4: 84, - 0xE01D5: 84, - 0xE01D6: 84, - 0xE01D7: 84, - 0xE01D8: 84, - 0xE01D9: 84, - 0xE01DA: 84, - 0xE01DB: 84, - 0xE01DC: 84, - 0xE01DD: 84, - 0xE01DE: 84, - 0xE01DF: 84, - 0xE01E0: 84, - 0xE01E1: 84, - 0xE01E2: 84, - 0xE01E3: 84, - 0xE01E4: 84, - 0xE01E5: 84, - 0xE01E6: 84, - 0xE01E7: 84, - 0xE01E8: 84, - 0xE01E9: 84, - 0xE01EA: 84, - 0xE01EB: 84, - 0xE01EC: 84, - 0xE01ED: 84, - 0xE01EE: 84, - 0xE01EF: 84, -} -codepoint_classes = { - "PVALID": ( - 0x2D0000002E, - 0x300000003A, - 0x610000007B, - 0xDF000000F7, - 0xF800000100, - 0x10100000102, - 0x10300000104, - 0x10500000106, - 0x10700000108, - 0x1090000010A, - 0x10B0000010C, - 0x10D0000010E, - 0x10F00000110, - 0x11100000112, - 0x11300000114, - 0x11500000116, - 0x11700000118, - 0x1190000011A, - 0x11B0000011C, - 0x11D0000011E, - 0x11F00000120, - 0x12100000122, - 0x12300000124, - 0x12500000126, - 0x12700000128, - 0x1290000012A, - 0x12B0000012C, - 0x12D0000012E, - 0x12F00000130, - 0x13100000132, - 0x13500000136, - 0x13700000139, - 0x13A0000013B, - 0x13C0000013D, - 0x13E0000013F, - 0x14200000143, - 0x14400000145, - 0x14600000147, - 0x14800000149, - 0x14B0000014C, - 0x14D0000014E, - 0x14F00000150, - 0x15100000152, - 0x15300000154, - 0x15500000156, - 0x15700000158, - 0x1590000015A, - 0x15B0000015C, - 0x15D0000015E, - 0x15F00000160, - 0x16100000162, - 0x16300000164, - 0x16500000166, - 0x16700000168, - 0x1690000016A, - 0x16B0000016C, - 0x16D0000016E, - 0x16F00000170, - 0x17100000172, - 0x17300000174, - 0x17500000176, - 0x17700000178, - 0x17A0000017B, - 0x17C0000017D, - 0x17E0000017F, - 0x18000000181, - 0x18300000184, - 0x18500000186, - 0x18800000189, - 0x18C0000018E, - 0x19200000193, - 0x19500000196, - 0x1990000019C, - 0x19E0000019F, - 0x1A1000001A2, - 0x1A3000001A4, - 0x1A5000001A6, - 0x1A8000001A9, - 0x1AA000001AC, - 0x1AD000001AE, - 0x1B0000001B1, - 0x1B4000001B5, - 0x1B6000001B7, - 0x1B9000001BC, - 0x1BD000001C4, - 0x1CE000001CF, - 0x1D0000001D1, - 0x1D2000001D3, - 0x1D4000001D5, - 0x1D6000001D7, - 0x1D8000001D9, - 0x1DA000001DB, - 0x1DC000001DE, - 0x1DF000001E0, - 0x1E1000001E2, - 0x1E3000001E4, - 0x1E5000001E6, - 0x1E7000001E8, - 0x1E9000001EA, - 0x1EB000001EC, - 0x1ED000001EE, - 0x1EF000001F1, - 0x1F5000001F6, - 0x1F9000001FA, - 0x1FB000001FC, - 0x1FD000001FE, - 0x1FF00000200, - 0x20100000202, - 0x20300000204, - 0x20500000206, - 0x20700000208, - 0x2090000020A, - 0x20B0000020C, - 0x20D0000020E, - 0x20F00000210, - 0x21100000212, - 0x21300000214, - 0x21500000216, - 0x21700000218, - 0x2190000021A, - 0x21B0000021C, - 0x21D0000021E, - 0x21F00000220, - 0x22100000222, - 0x22300000224, - 0x22500000226, - 0x22700000228, - 0x2290000022A, - 0x22B0000022C, - 0x22D0000022E, - 0x22F00000230, - 0x23100000232, - 0x2330000023A, - 0x23C0000023D, - 0x23F00000241, - 0x24200000243, - 0x24700000248, - 0x2490000024A, - 0x24B0000024C, - 0x24D0000024E, - 0x24F000002B0, - 0x2B9000002C2, - 0x2C6000002D2, - 0x2EC000002ED, - 0x2EE000002EF, - 0x30000000340, - 0x34200000343, - 0x3460000034F, - 0x35000000370, - 0x37100000372, - 0x37300000374, - 0x37700000378, - 0x37B0000037E, - 0x39000000391, - 0x3AC000003CF, - 0x3D7000003D8, - 0x3D9000003DA, - 0x3DB000003DC, - 0x3DD000003DE, - 0x3DF000003E0, - 0x3E1000003E2, - 0x3E3000003E4, - 0x3E5000003E6, - 0x3E7000003E8, - 0x3E9000003EA, - 0x3EB000003EC, - 0x3ED000003EE, - 0x3EF000003F0, - 0x3F3000003F4, - 0x3F8000003F9, - 0x3FB000003FD, - 0x43000000460, - 0x46100000462, - 0x46300000464, - 0x46500000466, - 0x46700000468, - 0x4690000046A, - 0x46B0000046C, - 0x46D0000046E, - 0x46F00000470, - 0x47100000472, - 0x47300000474, - 0x47500000476, - 0x47700000478, - 0x4790000047A, - 0x47B0000047C, - 0x47D0000047E, - 0x47F00000480, - 0x48100000482, - 0x48300000488, - 0x48B0000048C, - 0x48D0000048E, - 0x48F00000490, - 0x49100000492, - 0x49300000494, - 0x49500000496, - 0x49700000498, - 0x4990000049A, - 0x49B0000049C, - 0x49D0000049E, - 0x49F000004A0, - 0x4A1000004A2, - 0x4A3000004A4, - 0x4A5000004A6, - 0x4A7000004A8, - 0x4A9000004AA, - 0x4AB000004AC, - 0x4AD000004AE, - 0x4AF000004B0, - 0x4B1000004B2, - 0x4B3000004B4, - 0x4B5000004B6, - 0x4B7000004B8, - 0x4B9000004BA, - 0x4BB000004BC, - 0x4BD000004BE, - 0x4BF000004C0, - 0x4C2000004C3, - 0x4C4000004C5, - 0x4C6000004C7, - 0x4C8000004C9, - 0x4CA000004CB, - 0x4CC000004CD, - 0x4CE000004D0, - 0x4D1000004D2, - 0x4D3000004D4, - 0x4D5000004D6, - 0x4D7000004D8, - 0x4D9000004DA, - 0x4DB000004DC, - 0x4DD000004DE, - 0x4DF000004E0, - 0x4E1000004E2, - 0x4E3000004E4, - 0x4E5000004E6, - 0x4E7000004E8, - 0x4E9000004EA, - 0x4EB000004EC, - 0x4ED000004EE, - 0x4EF000004F0, - 0x4F1000004F2, - 0x4F3000004F4, - 0x4F5000004F6, - 0x4F7000004F8, - 0x4F9000004FA, - 0x4FB000004FC, - 0x4FD000004FE, - 0x4FF00000500, - 0x50100000502, - 0x50300000504, - 0x50500000506, - 0x50700000508, - 0x5090000050A, - 0x50B0000050C, - 0x50D0000050E, - 0x50F00000510, - 0x51100000512, - 0x51300000514, - 0x51500000516, - 0x51700000518, - 0x5190000051A, - 0x51B0000051C, - 0x51D0000051E, - 0x51F00000520, - 0x52100000522, - 0x52300000524, - 0x52500000526, - 0x52700000528, - 0x5290000052A, - 0x52B0000052C, - 0x52D0000052E, - 0x52F00000530, - 0x5590000055A, - 0x56000000587, - 0x58800000589, - 0x591000005BE, - 0x5BF000005C0, - 0x5C1000005C3, - 0x5C4000005C6, - 0x5C7000005C8, - 0x5D0000005EB, - 0x5EF000005F3, - 0x6100000061B, - 0x62000000640, - 0x64100000660, - 0x66E00000675, - 0x679000006D4, - 0x6D5000006DD, - 0x6DF000006E9, - 0x6EA000006F0, - 0x6FA00000700, - 0x7100000074B, - 0x74D000007B2, - 0x7C0000007F6, - 0x7FD000007FE, - 0x8000000082E, - 0x8400000085C, - 0x8600000086B, - 0x87000000888, - 0x8890000088F, - 0x897000008E2, - 0x8E300000958, - 0x96000000964, - 0x96600000970, - 0x97100000984, - 0x9850000098D, - 0x98F00000991, - 0x993000009A9, - 0x9AA000009B1, - 0x9B2000009B3, - 0x9B6000009BA, - 0x9BC000009C5, - 0x9C7000009C9, - 0x9CB000009CF, - 0x9D7000009D8, - 0x9E0000009E4, - 0x9E6000009F2, - 0x9FC000009FD, - 0x9FE000009FF, - 0xA0100000A04, - 0xA0500000A0B, - 0xA0F00000A11, - 0xA1300000A29, - 0xA2A00000A31, - 0xA3200000A33, - 0xA3500000A36, - 0xA3800000A3A, - 0xA3C00000A3D, - 0xA3E00000A43, - 0xA4700000A49, - 0xA4B00000A4E, - 0xA5100000A52, - 0xA5C00000A5D, - 0xA6600000A76, - 0xA8100000A84, - 0xA8500000A8E, - 0xA8F00000A92, - 0xA9300000AA9, - 0xAAA00000AB1, - 0xAB200000AB4, - 0xAB500000ABA, - 0xABC00000AC6, - 0xAC700000ACA, - 0xACB00000ACE, - 0xAD000000AD1, - 0xAE000000AE4, - 0xAE600000AF0, - 0xAF900000B00, - 0xB0100000B04, - 0xB0500000B0D, - 0xB0F00000B11, - 0xB1300000B29, - 0xB2A00000B31, - 0xB3200000B34, - 0xB3500000B3A, - 0xB3C00000B45, - 0xB4700000B49, - 0xB4B00000B4E, - 0xB5500000B58, - 0xB5F00000B64, - 0xB6600000B70, - 0xB7100000B72, - 0xB8200000B84, - 0xB8500000B8B, - 0xB8E00000B91, - 0xB9200000B96, - 0xB9900000B9B, - 0xB9C00000B9D, - 0xB9E00000BA0, - 0xBA300000BA5, - 0xBA800000BAB, - 0xBAE00000BBA, - 0xBBE00000BC3, - 0xBC600000BC9, - 0xBCA00000BCE, - 0xBD000000BD1, - 0xBD700000BD8, - 0xBE600000BF0, - 0xC0000000C0D, - 0xC0E00000C11, - 0xC1200000C29, - 0xC2A00000C3A, - 0xC3C00000C45, - 0xC4600000C49, - 0xC4A00000C4E, - 0xC5500000C57, - 0xC5800000C5B, - 0xC5D00000C5E, - 0xC6000000C64, - 0xC6600000C70, - 0xC8000000C84, - 0xC8500000C8D, - 0xC8E00000C91, - 0xC9200000CA9, - 0xCAA00000CB4, - 0xCB500000CBA, - 0xCBC00000CC5, - 0xCC600000CC9, - 0xCCA00000CCE, - 0xCD500000CD7, - 0xCDD00000CDF, - 0xCE000000CE4, - 0xCE600000CF0, - 0xCF100000CF4, - 0xD0000000D0D, - 0xD0E00000D11, - 0xD1200000D45, - 0xD4600000D49, - 0xD4A00000D4F, - 0xD5400000D58, - 0xD5F00000D64, - 0xD6600000D70, - 0xD7A00000D80, - 0xD8100000D84, - 0xD8500000D97, - 0xD9A00000DB2, - 0xDB300000DBC, - 0xDBD00000DBE, - 0xDC000000DC7, - 0xDCA00000DCB, - 0xDCF00000DD5, - 0xDD600000DD7, - 0xDD800000DE0, - 0xDE600000DF0, - 0xDF200000DF4, - 0xE0100000E33, - 0xE3400000E3B, - 0xE4000000E4F, - 0xE5000000E5A, - 0xE8100000E83, - 0xE8400000E85, - 0xE8600000E8B, - 0xE8C00000EA4, - 0xEA500000EA6, - 0xEA700000EB3, - 0xEB400000EBE, - 0xEC000000EC5, - 0xEC600000EC7, - 0xEC800000ECF, - 0xED000000EDA, - 0xEDE00000EE0, - 0xF0000000F01, - 0xF0B00000F0C, - 0xF1800000F1A, - 0xF2000000F2A, - 0xF3500000F36, - 0xF3700000F38, - 0xF3900000F3A, - 0xF3E00000F43, - 0xF4400000F48, - 0xF4900000F4D, - 0xF4E00000F52, - 0xF5300000F57, - 0xF5800000F5C, - 0xF5D00000F69, - 0xF6A00000F6D, - 0xF7100000F73, - 0xF7400000F75, - 0xF7A00000F81, - 0xF8200000F85, - 0xF8600000F93, - 0xF9400000F98, - 0xF9900000F9D, - 0xF9E00000FA2, - 0xFA300000FA7, - 0xFA800000FAC, - 0xFAD00000FB9, - 0xFBA00000FBD, - 0xFC600000FC7, - 0x10000000104A, - 0x10500000109E, - 0x10D0000010FB, - 0x10FD00001100, - 0x120000001249, - 0x124A0000124E, - 0x125000001257, - 0x125800001259, - 0x125A0000125E, - 0x126000001289, - 0x128A0000128E, - 0x1290000012B1, - 0x12B2000012B6, - 0x12B8000012BF, - 0x12C0000012C1, - 0x12C2000012C6, - 0x12C8000012D7, - 0x12D800001311, - 0x131200001316, - 0x13180000135B, - 0x135D00001360, - 0x138000001390, - 0x13A0000013F6, - 0x14010000166D, - 0x166F00001680, - 0x16810000169B, - 0x16A0000016EB, - 0x16F1000016F9, - 0x170000001716, - 0x171F00001735, - 0x174000001754, - 0x17600000176D, - 0x176E00001771, - 0x177200001774, - 0x1780000017B4, - 0x17B6000017D4, - 0x17D7000017D8, - 0x17DC000017DE, - 0x17E0000017EA, - 0x18100000181A, - 0x182000001879, - 0x1880000018AB, - 0x18B0000018F6, - 0x19000000191F, - 0x19200000192C, - 0x19300000193C, - 0x19460000196E, - 0x197000001975, - 0x1980000019AC, - 0x19B0000019CA, - 0x19D0000019DA, - 0x1A0000001A1C, - 0x1A2000001A5F, - 0x1A6000001A7D, - 0x1A7F00001A8A, - 0x1A9000001A9A, - 0x1AA700001AA8, - 0x1AB000001ABE, - 0x1ABF00001ACF, - 0x1B0000001B4D, - 0x1B5000001B5A, - 0x1B6B00001B74, - 0x1B8000001BF4, - 0x1C0000001C38, - 0x1C4000001C4A, - 0x1C4D00001C7E, - 0x1C8A00001C8B, - 0x1CD000001CD3, - 0x1CD400001CFB, - 0x1D0000001D2C, - 0x1D2F00001D30, - 0x1D3B00001D3C, - 0x1D4E00001D4F, - 0x1D6B00001D78, - 0x1D7900001D9B, - 0x1DC000001E00, - 0x1E0100001E02, - 0x1E0300001E04, - 0x1E0500001E06, - 0x1E0700001E08, - 0x1E0900001E0A, - 0x1E0B00001E0C, - 0x1E0D00001E0E, - 0x1E0F00001E10, - 0x1E1100001E12, - 0x1E1300001E14, - 0x1E1500001E16, - 0x1E1700001E18, - 0x1E1900001E1A, - 0x1E1B00001E1C, - 0x1E1D00001E1E, - 0x1E1F00001E20, - 0x1E2100001E22, - 0x1E2300001E24, - 0x1E2500001E26, - 0x1E2700001E28, - 0x1E2900001E2A, - 0x1E2B00001E2C, - 0x1E2D00001E2E, - 0x1E2F00001E30, - 0x1E3100001E32, - 0x1E3300001E34, - 0x1E3500001E36, - 0x1E3700001E38, - 0x1E3900001E3A, - 0x1E3B00001E3C, - 0x1E3D00001E3E, - 0x1E3F00001E40, - 0x1E4100001E42, - 0x1E4300001E44, - 0x1E4500001E46, - 0x1E4700001E48, - 0x1E4900001E4A, - 0x1E4B00001E4C, - 0x1E4D00001E4E, - 0x1E4F00001E50, - 0x1E5100001E52, - 0x1E5300001E54, - 0x1E5500001E56, - 0x1E5700001E58, - 0x1E5900001E5A, - 0x1E5B00001E5C, - 0x1E5D00001E5E, - 0x1E5F00001E60, - 0x1E6100001E62, - 0x1E6300001E64, - 0x1E6500001E66, - 0x1E6700001E68, - 0x1E6900001E6A, - 0x1E6B00001E6C, - 0x1E6D00001E6E, - 0x1E6F00001E70, - 0x1E7100001E72, - 0x1E7300001E74, - 0x1E7500001E76, - 0x1E7700001E78, - 0x1E7900001E7A, - 0x1E7B00001E7C, - 0x1E7D00001E7E, - 0x1E7F00001E80, - 0x1E8100001E82, - 0x1E8300001E84, - 0x1E8500001E86, - 0x1E8700001E88, - 0x1E8900001E8A, - 0x1E8B00001E8C, - 0x1E8D00001E8E, - 0x1E8F00001E90, - 0x1E9100001E92, - 0x1E9300001E94, - 0x1E9500001E9A, - 0x1E9C00001E9E, - 0x1E9F00001EA0, - 0x1EA100001EA2, - 0x1EA300001EA4, - 0x1EA500001EA6, - 0x1EA700001EA8, - 0x1EA900001EAA, - 0x1EAB00001EAC, - 0x1EAD00001EAE, - 0x1EAF00001EB0, - 0x1EB100001EB2, - 0x1EB300001EB4, - 0x1EB500001EB6, - 0x1EB700001EB8, - 0x1EB900001EBA, - 0x1EBB00001EBC, - 0x1EBD00001EBE, - 0x1EBF00001EC0, - 0x1EC100001EC2, - 0x1EC300001EC4, - 0x1EC500001EC6, - 0x1EC700001EC8, - 0x1EC900001ECA, - 0x1ECB00001ECC, - 0x1ECD00001ECE, - 0x1ECF00001ED0, - 0x1ED100001ED2, - 0x1ED300001ED4, - 0x1ED500001ED6, - 0x1ED700001ED8, - 0x1ED900001EDA, - 0x1EDB00001EDC, - 0x1EDD00001EDE, - 0x1EDF00001EE0, - 0x1EE100001EE2, - 0x1EE300001EE4, - 0x1EE500001EE6, - 0x1EE700001EE8, - 0x1EE900001EEA, - 0x1EEB00001EEC, - 0x1EED00001EEE, - 0x1EEF00001EF0, - 0x1EF100001EF2, - 0x1EF300001EF4, - 0x1EF500001EF6, - 0x1EF700001EF8, - 0x1EF900001EFA, - 0x1EFB00001EFC, - 0x1EFD00001EFE, - 0x1EFF00001F08, - 0x1F1000001F16, - 0x1F2000001F28, - 0x1F3000001F38, - 0x1F4000001F46, - 0x1F5000001F58, - 0x1F6000001F68, - 0x1F7000001F71, - 0x1F7200001F73, - 0x1F7400001F75, - 0x1F7600001F77, - 0x1F7800001F79, - 0x1F7A00001F7B, - 0x1F7C00001F7D, - 0x1FB000001FB2, - 0x1FB600001FB7, - 0x1FC600001FC7, - 0x1FD000001FD3, - 0x1FD600001FD8, - 0x1FE000001FE3, - 0x1FE400001FE8, - 0x1FF600001FF7, - 0x214E0000214F, - 0x218400002185, - 0x2C3000002C60, - 0x2C6100002C62, - 0x2C6500002C67, - 0x2C6800002C69, - 0x2C6A00002C6B, - 0x2C6C00002C6D, - 0x2C7100002C72, - 0x2C7300002C75, - 0x2C7600002C7C, - 0x2C8100002C82, - 0x2C8300002C84, - 0x2C8500002C86, - 0x2C8700002C88, - 0x2C8900002C8A, - 0x2C8B00002C8C, - 0x2C8D00002C8E, - 0x2C8F00002C90, - 0x2C9100002C92, - 0x2C9300002C94, - 0x2C9500002C96, - 0x2C9700002C98, - 0x2C9900002C9A, - 0x2C9B00002C9C, - 0x2C9D00002C9E, - 0x2C9F00002CA0, - 0x2CA100002CA2, - 0x2CA300002CA4, - 0x2CA500002CA6, - 0x2CA700002CA8, - 0x2CA900002CAA, - 0x2CAB00002CAC, - 0x2CAD00002CAE, - 0x2CAF00002CB0, - 0x2CB100002CB2, - 0x2CB300002CB4, - 0x2CB500002CB6, - 0x2CB700002CB8, - 0x2CB900002CBA, - 0x2CBB00002CBC, - 0x2CBD00002CBE, - 0x2CBF00002CC0, - 0x2CC100002CC2, - 0x2CC300002CC4, - 0x2CC500002CC6, - 0x2CC700002CC8, - 0x2CC900002CCA, - 0x2CCB00002CCC, - 0x2CCD00002CCE, - 0x2CCF00002CD0, - 0x2CD100002CD2, - 0x2CD300002CD4, - 0x2CD500002CD6, - 0x2CD700002CD8, - 0x2CD900002CDA, - 0x2CDB00002CDC, - 0x2CDD00002CDE, - 0x2CDF00002CE0, - 0x2CE100002CE2, - 0x2CE300002CE5, - 0x2CEC00002CED, - 0x2CEE00002CF2, - 0x2CF300002CF4, - 0x2D0000002D26, - 0x2D2700002D28, - 0x2D2D00002D2E, - 0x2D3000002D68, - 0x2D7F00002D97, - 0x2DA000002DA7, - 0x2DA800002DAF, - 0x2DB000002DB7, - 0x2DB800002DBF, - 0x2DC000002DC7, - 0x2DC800002DCF, - 0x2DD000002DD7, - 0x2DD800002DDF, - 0x2DE000002E00, - 0x2E2F00002E30, - 0x300500003008, - 0x302A0000302E, - 0x303C0000303D, - 0x304100003097, - 0x30990000309B, - 0x309D0000309F, - 0x30A1000030FB, - 0x30FC000030FF, - 0x310500003130, - 0x31A0000031C0, - 0x31F000003200, - 0x340000004DC0, - 0x4E000000A48D, - 0xA4D00000A4FE, - 0xA5000000A60D, - 0xA6100000A62C, - 0xA6410000A642, - 0xA6430000A644, - 0xA6450000A646, - 0xA6470000A648, - 0xA6490000A64A, - 0xA64B0000A64C, - 0xA64D0000A64E, - 0xA64F0000A650, - 0xA6510000A652, - 0xA6530000A654, - 0xA6550000A656, - 0xA6570000A658, - 0xA6590000A65A, - 0xA65B0000A65C, - 0xA65D0000A65E, - 0xA65F0000A660, - 0xA6610000A662, - 0xA6630000A664, - 0xA6650000A666, - 0xA6670000A668, - 0xA6690000A66A, - 0xA66B0000A66C, - 0xA66D0000A670, - 0xA6740000A67E, - 0xA67F0000A680, - 0xA6810000A682, - 0xA6830000A684, - 0xA6850000A686, - 0xA6870000A688, - 0xA6890000A68A, - 0xA68B0000A68C, - 0xA68D0000A68E, - 0xA68F0000A690, - 0xA6910000A692, - 0xA6930000A694, - 0xA6950000A696, - 0xA6970000A698, - 0xA6990000A69A, - 0xA69B0000A69C, - 0xA69E0000A6E6, - 0xA6F00000A6F2, - 0xA7170000A720, - 0xA7230000A724, - 0xA7250000A726, - 0xA7270000A728, - 0xA7290000A72A, - 0xA72B0000A72C, - 0xA72D0000A72E, - 0xA72F0000A732, - 0xA7330000A734, - 0xA7350000A736, - 0xA7370000A738, - 0xA7390000A73A, - 0xA73B0000A73C, - 0xA73D0000A73E, - 0xA73F0000A740, - 0xA7410000A742, - 0xA7430000A744, - 0xA7450000A746, - 0xA7470000A748, - 0xA7490000A74A, - 0xA74B0000A74C, - 0xA74D0000A74E, - 0xA74F0000A750, - 0xA7510000A752, - 0xA7530000A754, - 0xA7550000A756, - 0xA7570000A758, - 0xA7590000A75A, - 0xA75B0000A75C, - 0xA75D0000A75E, - 0xA75F0000A760, - 0xA7610000A762, - 0xA7630000A764, - 0xA7650000A766, - 0xA7670000A768, - 0xA7690000A76A, - 0xA76B0000A76C, - 0xA76D0000A76E, - 0xA76F0000A770, - 0xA7710000A779, - 0xA77A0000A77B, - 0xA77C0000A77D, - 0xA77F0000A780, - 0xA7810000A782, - 0xA7830000A784, - 0xA7850000A786, - 0xA7870000A789, - 0xA78C0000A78D, - 0xA78E0000A790, - 0xA7910000A792, - 0xA7930000A796, - 0xA7970000A798, - 0xA7990000A79A, - 0xA79B0000A79C, - 0xA79D0000A79E, - 0xA79F0000A7A0, - 0xA7A10000A7A2, - 0xA7A30000A7A4, - 0xA7A50000A7A6, - 0xA7A70000A7A8, - 0xA7A90000A7AA, - 0xA7AF0000A7B0, - 0xA7B50000A7B6, - 0xA7B70000A7B8, - 0xA7B90000A7BA, - 0xA7BB0000A7BC, - 0xA7BD0000A7BE, - 0xA7BF0000A7C0, - 0xA7C10000A7C2, - 0xA7C30000A7C4, - 0xA7C80000A7C9, - 0xA7CA0000A7CB, - 0xA7CD0000A7CE, - 0xA7D10000A7D2, - 0xA7D30000A7D4, - 0xA7D50000A7D6, - 0xA7D70000A7D8, - 0xA7D90000A7DA, - 0xA7DB0000A7DC, - 0xA7F60000A7F8, - 0xA7FA0000A828, - 0xA82C0000A82D, - 0xA8400000A874, - 0xA8800000A8C6, - 0xA8D00000A8DA, - 0xA8E00000A8F8, - 0xA8FB0000A8FC, - 0xA8FD0000A92E, - 0xA9300000A954, - 0xA9800000A9C1, - 0xA9CF0000A9DA, - 0xA9E00000A9FF, - 0xAA000000AA37, - 0xAA400000AA4E, - 0xAA500000AA5A, - 0xAA600000AA77, - 0xAA7A0000AAC3, - 0xAADB0000AADE, - 0xAAE00000AAF0, - 0xAAF20000AAF7, - 0xAB010000AB07, - 0xAB090000AB0F, - 0xAB110000AB17, - 0xAB200000AB27, - 0xAB280000AB2F, - 0xAB300000AB5B, - 0xAB600000AB69, - 0xABC00000ABEB, - 0xABEC0000ABEE, - 0xABF00000ABFA, - 0xAC000000D7A4, - 0xFA0E0000FA10, - 0xFA110000FA12, - 0xFA130000FA15, - 0xFA1F0000FA20, - 0xFA210000FA22, - 0xFA230000FA25, - 0xFA270000FA2A, - 0xFB1E0000FB1F, - 0xFE200000FE30, - 0xFE730000FE74, - 0x100000001000C, - 0x1000D00010027, - 0x100280001003B, - 0x1003C0001003E, - 0x1003F0001004E, - 0x100500001005E, - 0x10080000100FB, - 0x101FD000101FE, - 0x102800001029D, - 0x102A0000102D1, - 0x102E0000102E1, - 0x1030000010320, - 0x1032D00010341, - 0x103420001034A, - 0x103500001037B, - 0x103800001039E, - 0x103A0000103C4, - 0x103C8000103D0, - 0x104280001049E, - 0x104A0000104AA, - 0x104D8000104FC, - 0x1050000010528, - 0x1053000010564, - 0x10597000105A2, - 0x105A3000105B2, - 0x105B3000105BA, - 0x105BB000105BD, - 0x105C0000105F4, - 0x1060000010737, - 0x1074000010756, - 0x1076000010768, - 0x1078000010781, - 0x1080000010806, - 0x1080800010809, - 0x1080A00010836, - 0x1083700010839, - 0x1083C0001083D, - 0x1083F00010856, - 0x1086000010877, - 0x108800001089F, - 0x108E0000108F3, - 0x108F4000108F6, - 0x1090000010916, - 0x109200001093A, - 0x10980000109B8, - 0x109BE000109C0, - 0x10A0000010A04, - 0x10A0500010A07, - 0x10A0C00010A14, - 0x10A1500010A18, - 0x10A1900010A36, - 0x10A3800010A3B, - 0x10A3F00010A40, - 0x10A6000010A7D, - 0x10A8000010A9D, - 0x10AC000010AC8, - 0x10AC900010AE7, - 0x10B0000010B36, - 0x10B4000010B56, - 0x10B6000010B73, - 0x10B8000010B92, - 0x10C0000010C49, - 0x10CC000010CF3, - 0x10D0000010D28, - 0x10D3000010D3A, - 0x10D4000010D50, - 0x10D6900010D6E, - 0x10D6F00010D86, - 0x10E8000010EAA, - 0x10EAB00010EAD, - 0x10EB000010EB2, - 0x10EC200010EC5, - 0x10EFC00010F1D, - 0x10F2700010F28, - 0x10F3000010F51, - 0x10F7000010F86, - 0x10FB000010FC5, - 0x10FE000010FF7, - 0x1100000011047, - 0x1106600011076, - 0x1107F000110BB, - 0x110C2000110C3, - 0x110D0000110E9, - 0x110F0000110FA, - 0x1110000011135, - 0x1113600011140, - 0x1114400011148, - 0x1115000011174, - 0x1117600011177, - 0x11180000111C5, - 0x111C9000111CD, - 0x111CE000111DB, - 0x111DC000111DD, - 0x1120000011212, - 0x1121300011238, - 0x1123E00011242, - 0x1128000011287, - 0x1128800011289, - 0x1128A0001128E, - 0x1128F0001129E, - 0x1129F000112A9, - 0x112B0000112EB, - 0x112F0000112FA, - 0x1130000011304, - 0x113050001130D, - 0x1130F00011311, - 0x1131300011329, - 0x1132A00011331, - 0x1133200011334, - 0x113350001133A, - 0x1133B00011345, - 0x1134700011349, - 0x1134B0001134E, - 0x1135000011351, - 0x1135700011358, - 0x1135D00011364, - 0x113660001136D, - 0x1137000011375, - 0x113800001138A, - 0x1138B0001138C, - 0x1138E0001138F, - 0x11390000113B6, - 0x113B7000113C1, - 0x113C2000113C3, - 0x113C5000113C6, - 0x113C7000113CB, - 0x113CC000113D4, - 0x113E1000113E3, - 0x114000001144B, - 0x114500001145A, - 0x1145E00011462, - 0x11480000114C6, - 0x114C7000114C8, - 0x114D0000114DA, - 0x11580000115B6, - 0x115B8000115C1, - 0x115D8000115DE, - 0x1160000011641, - 0x1164400011645, - 0x116500001165A, - 0x11680000116B9, - 0x116C0000116CA, - 0x116D0000116E4, - 0x117000001171B, - 0x1171D0001172C, - 0x117300001173A, - 0x1174000011747, - 0x118000001183B, - 0x118C0000118EA, - 0x118FF00011907, - 0x119090001190A, - 0x1190C00011914, - 0x1191500011917, - 0x1191800011936, - 0x1193700011939, - 0x1193B00011944, - 0x119500001195A, - 0x119A0000119A8, - 0x119AA000119D8, - 0x119DA000119E2, - 0x119E3000119E5, - 0x11A0000011A3F, - 0x11A4700011A48, - 0x11A5000011A9A, - 0x11A9D00011A9E, - 0x11AB000011AF9, - 0x11BC000011BE1, - 0x11BF000011BFA, - 0x11C0000011C09, - 0x11C0A00011C37, - 0x11C3800011C41, - 0x11C5000011C5A, - 0x11C7200011C90, - 0x11C9200011CA8, - 0x11CA900011CB7, - 0x11D0000011D07, - 0x11D0800011D0A, - 0x11D0B00011D37, - 0x11D3A00011D3B, - 0x11D3C00011D3E, - 0x11D3F00011D48, - 0x11D5000011D5A, - 0x11D6000011D66, - 0x11D6700011D69, - 0x11D6A00011D8F, - 0x11D9000011D92, - 0x11D9300011D99, - 0x11DA000011DAA, - 0x11EE000011EF7, - 0x11F0000011F11, - 0x11F1200011F3B, - 0x11F3E00011F43, - 0x11F5000011F5B, - 0x11FB000011FB1, - 0x120000001239A, - 0x1248000012544, - 0x12F9000012FF1, - 0x1300000013430, - 0x1344000013456, - 0x13460000143FB, - 0x1440000014647, - 0x161000001613A, - 0x1680000016A39, - 0x16A4000016A5F, - 0x16A6000016A6A, - 0x16A7000016ABF, - 0x16AC000016ACA, - 0x16AD000016AEE, - 0x16AF000016AF5, - 0x16B0000016B37, - 0x16B4000016B44, - 0x16B5000016B5A, - 0x16B6300016B78, - 0x16B7D00016B90, - 0x16D4000016D6D, - 0x16D7000016D7A, - 0x16E6000016E80, - 0x16F0000016F4B, - 0x16F4F00016F88, - 0x16F8F00016FA0, - 0x16FE000016FE2, - 0x16FE300016FE5, - 0x16FF000016FF2, - 0x17000000187F8, - 0x1880000018CD6, - 0x18CFF00018D09, - 0x1AFF00001AFF4, - 0x1AFF50001AFFC, - 0x1AFFD0001AFFF, - 0x1B0000001B123, - 0x1B1320001B133, - 0x1B1500001B153, - 0x1B1550001B156, - 0x1B1640001B168, - 0x1B1700001B2FC, - 0x1BC000001BC6B, - 0x1BC700001BC7D, - 0x1BC800001BC89, - 0x1BC900001BC9A, - 0x1BC9D0001BC9F, - 0x1CCF00001CCFA, - 0x1CF000001CF2E, - 0x1CF300001CF47, - 0x1DA000001DA37, - 0x1DA3B0001DA6D, - 0x1DA750001DA76, - 0x1DA840001DA85, - 0x1DA9B0001DAA0, - 0x1DAA10001DAB0, - 0x1DF000001DF1F, - 0x1DF250001DF2B, - 0x1E0000001E007, - 0x1E0080001E019, - 0x1E01B0001E022, - 0x1E0230001E025, - 0x1E0260001E02B, - 0x1E08F0001E090, - 0x1E1000001E12D, - 0x1E1300001E13E, - 0x1E1400001E14A, - 0x1E14E0001E14F, - 0x1E2900001E2AF, - 0x1E2C00001E2FA, - 0x1E4D00001E4FA, - 0x1E5D00001E5FB, - 0x1E7E00001E7E7, - 0x1E7E80001E7EC, - 0x1E7ED0001E7EF, - 0x1E7F00001E7FF, - 0x1E8000001E8C5, - 0x1E8D00001E8D7, - 0x1E9220001E94C, - 0x1E9500001E95A, - 0x200000002A6E0, - 0x2A7000002B73A, - 0x2B7400002B81E, - 0x2B8200002CEA2, - 0x2CEB00002EBE1, - 0x2EBF00002EE5E, - 0x300000003134B, - 0x31350000323B0, - ), - "CONTEXTJ": (0x200C0000200E,), - "CONTEXTO": ( - 0xB7000000B8, - 0x37500000376, - 0x5F3000005F5, - 0x6600000066A, - 0x6F0000006FA, - 0x30FB000030FC, - ), -} diff --git a/.env/Lib/site-packages/idna/intranges.py b/.env/Lib/site-packages/idna/intranges.py deleted file mode 100644 index 7bfaa8d..0000000 --- a/.env/Lib/site-packages/idna/intranges.py +++ /dev/null @@ -1,57 +0,0 @@ -""" -Given a list of integers, made up of (hopefully) a small number of long runs -of consecutive integers, compute a representation of the form -((start1, end1), (start2, end2) ...). Then answer the question "was x present -in the original list?" in time O(log(# runs)). -""" - -import bisect -from typing import List, Tuple - - -def intranges_from_list(list_: List[int]) -> Tuple[int, ...]: - """Represent a list of integers as a sequence of ranges: - ((start_0, end_0), (start_1, end_1), ...), such that the original - integers are exactly those x such that start_i <= x < end_i for some i. - - Ranges are encoded as single integers (start << 32 | end), not as tuples. - """ - - sorted_list = sorted(list_) - ranges = [] - last_write = -1 - for i in range(len(sorted_list)): - if i + 1 < len(sorted_list): - if sorted_list[i] == sorted_list[i + 1] - 1: - continue - current_range = sorted_list[last_write + 1 : i + 1] - ranges.append(_encode_range(current_range[0], current_range[-1] + 1)) - last_write = i - - return tuple(ranges) - - -def _encode_range(start: int, end: int) -> int: - return (start << 32) | end - - -def _decode_range(r: int) -> Tuple[int, int]: - return (r >> 32), (r & ((1 << 32) - 1)) - - -def intranges_contain(int_: int, ranges: Tuple[int, ...]) -> bool: - """Determine if `int_` falls into one of the ranges in `ranges`.""" - tuple_ = _encode_range(int_, 0) - pos = bisect.bisect_left(ranges, tuple_) - # we could be immediately ahead of a tuple (start, end) - # with start < int_ <= end - if pos > 0: - left, right = _decode_range(ranges[pos - 1]) - if left <= int_ < right: - return True - # or we could be immediately behind a tuple (int_, end) - if pos < len(ranges): - left, _ = _decode_range(ranges[pos]) - if left == int_: - return True - return False diff --git a/.env/Lib/site-packages/idna/package_data.py b/.env/Lib/site-packages/idna/package_data.py deleted file mode 100644 index 7272c8d..0000000 --- a/.env/Lib/site-packages/idna/package_data.py +++ /dev/null @@ -1 +0,0 @@ -__version__ = "3.11" diff --git a/.env/Lib/site-packages/idna/py.typed b/.env/Lib/site-packages/idna/py.typed deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/idna/uts46data.py b/.env/Lib/site-packages/idna/uts46data.py deleted file mode 100644 index 4610b71..0000000 --- a/.env/Lib/site-packages/idna/uts46data.py +++ /dev/null @@ -1,8841 +0,0 @@ -# This file is automatically generated by tools/idna-data -# vim: set fileencoding=utf-8 : - -from typing import List, Tuple, Union - -"""IDNA Mapping Table from UTS46.""" - - -__version__ = "16.0.0" - - -def _seg_0() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x0, "V"), - (0x1, "V"), - (0x2, "V"), - (0x3, "V"), - (0x4, "V"), - (0x5, "V"), - (0x6, "V"), - (0x7, "V"), - (0x8, "V"), - (0x9, "V"), - (0xA, "V"), - (0xB, "V"), - (0xC, "V"), - (0xD, "V"), - (0xE, "V"), - (0xF, "V"), - (0x10, "V"), - (0x11, "V"), - (0x12, "V"), - (0x13, "V"), - (0x14, "V"), - (0x15, "V"), - (0x16, "V"), - (0x17, "V"), - (0x18, "V"), - (0x19, "V"), - (0x1A, "V"), - (0x1B, "V"), - (0x1C, "V"), - (0x1D, "V"), - (0x1E, "V"), - (0x1F, "V"), - (0x20, "V"), - (0x21, "V"), - (0x22, "V"), - (0x23, "V"), - (0x24, "V"), - (0x25, "V"), - (0x26, "V"), - (0x27, "V"), - (0x28, "V"), - (0x29, "V"), - (0x2A, "V"), - (0x2B, "V"), - (0x2C, "V"), - (0x2D, "V"), - (0x2E, "V"), - (0x2F, "V"), - (0x30, "V"), - (0x31, "V"), - (0x32, "V"), - (0x33, "V"), - (0x34, "V"), - (0x35, "V"), - (0x36, "V"), - (0x37, "V"), - (0x38, "V"), - (0x39, "V"), - (0x3A, "V"), - (0x3B, "V"), - (0x3C, "V"), - (0x3D, "V"), - (0x3E, "V"), - (0x3F, "V"), - (0x40, "V"), - (0x41, "M", "a"), - (0x42, "M", "b"), - (0x43, "M", "c"), - (0x44, "M", "d"), - (0x45, "M", "e"), - (0x46, "M", "f"), - (0x47, "M", "g"), - (0x48, "M", "h"), - (0x49, "M", "i"), - (0x4A, "M", "j"), - (0x4B, "M", "k"), - (0x4C, "M", "l"), - (0x4D, "M", "m"), - (0x4E, "M", "n"), - (0x4F, "M", "o"), - (0x50, "M", "p"), - (0x51, "M", "q"), - (0x52, "M", "r"), - (0x53, "M", "s"), - (0x54, "M", "t"), - (0x55, "M", "u"), - (0x56, "M", "v"), - (0x57, "M", "w"), - (0x58, "M", "x"), - (0x59, "M", "y"), - (0x5A, "M", "z"), - (0x5B, "V"), - (0x5C, "V"), - (0x5D, "V"), - (0x5E, "V"), - (0x5F, "V"), - (0x60, "V"), - (0x61, "V"), - (0x62, "V"), - (0x63, "V"), - ] - - -def _seg_1() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x64, "V"), - (0x65, "V"), - (0x66, "V"), - (0x67, "V"), - (0x68, "V"), - (0x69, "V"), - (0x6A, "V"), - (0x6B, "V"), - (0x6C, "V"), - (0x6D, "V"), - (0x6E, "V"), - (0x6F, "V"), - (0x70, "V"), - (0x71, "V"), - (0x72, "V"), - (0x73, "V"), - (0x74, "V"), - (0x75, "V"), - (0x76, "V"), - (0x77, "V"), - (0x78, "V"), - (0x79, "V"), - (0x7A, "V"), - (0x7B, "V"), - (0x7C, "V"), - (0x7D, "V"), - (0x7E, "V"), - (0x7F, "V"), - (0x80, "X"), - (0x81, "X"), - (0x82, "X"), - (0x83, "X"), - (0x84, "X"), - (0x85, "X"), - (0x86, "X"), - (0x87, "X"), - (0x88, "X"), - (0x89, "X"), - (0x8A, "X"), - (0x8B, "X"), - (0x8C, "X"), - (0x8D, "X"), - (0x8E, "X"), - (0x8F, "X"), - (0x90, "X"), - (0x91, "X"), - (0x92, "X"), - (0x93, "X"), - (0x94, "X"), - (0x95, "X"), - (0x96, "X"), - (0x97, "X"), - (0x98, "X"), - (0x99, "X"), - (0x9A, "X"), - (0x9B, "X"), - (0x9C, "X"), - (0x9D, "X"), - (0x9E, "X"), - (0x9F, "X"), - (0xA0, "M", " "), - (0xA1, "V"), - (0xA2, "V"), - (0xA3, "V"), - (0xA4, "V"), - (0xA5, "V"), - (0xA6, "V"), - (0xA7, "V"), - (0xA8, "M", " ̈"), - (0xA9, "V"), - (0xAA, "M", "a"), - (0xAB, "V"), - (0xAC, "V"), - (0xAD, "I"), - (0xAE, "V"), - (0xAF, "M", " ̄"), - (0xB0, "V"), - (0xB1, "V"), - (0xB2, "M", "2"), - (0xB3, "M", "3"), - (0xB4, "M", " ́"), - (0xB5, "M", "μ"), - (0xB6, "V"), - (0xB7, "V"), - (0xB8, "M", " ̧"), - (0xB9, "M", "1"), - (0xBA, "M", "o"), - (0xBB, "V"), - (0xBC, "M", "1⁄4"), - (0xBD, "M", "1⁄2"), - (0xBE, "M", "3⁄4"), - (0xBF, "V"), - (0xC0, "M", "à"), - (0xC1, "M", "á"), - (0xC2, "M", "â"), - (0xC3, "M", "ã"), - (0xC4, "M", "ä"), - (0xC5, "M", "å"), - (0xC6, "M", "æ"), - (0xC7, "M", "ç"), - ] - - -def _seg_2() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xC8, "M", "è"), - (0xC9, "M", "é"), - (0xCA, "M", "ê"), - (0xCB, "M", "ë"), - (0xCC, "M", "ì"), - (0xCD, "M", "í"), - (0xCE, "M", "î"), - (0xCF, "M", "ï"), - (0xD0, "M", "ð"), - (0xD1, "M", "ñ"), - (0xD2, "M", "ò"), - (0xD3, "M", "ó"), - (0xD4, "M", "ô"), - (0xD5, "M", "õ"), - (0xD6, "M", "ö"), - (0xD7, "V"), - (0xD8, "M", "ø"), - (0xD9, "M", "ù"), - (0xDA, "M", "ú"), - (0xDB, "M", "û"), - (0xDC, "M", "ü"), - (0xDD, "M", "ý"), - (0xDE, "M", "þ"), - (0xDF, "D", "ss"), - (0xE0, "V"), - (0xE1, "V"), - (0xE2, "V"), - (0xE3, "V"), - (0xE4, "V"), - (0xE5, "V"), - (0xE6, "V"), - (0xE7, "V"), - (0xE8, "V"), - (0xE9, "V"), - (0xEA, "V"), - (0xEB, "V"), - (0xEC, "V"), - (0xED, "V"), - (0xEE, "V"), - (0xEF, "V"), - (0xF0, "V"), - (0xF1, "V"), - (0xF2, "V"), - (0xF3, "V"), - (0xF4, "V"), - (0xF5, "V"), - (0xF6, "V"), - (0xF7, "V"), - (0xF8, "V"), - (0xF9, "V"), - (0xFA, "V"), - (0xFB, "V"), - (0xFC, "V"), - (0xFD, "V"), - (0xFE, "V"), - (0xFF, "V"), - (0x100, "M", "ā"), - (0x101, "V"), - (0x102, "M", "ă"), - (0x103, "V"), - (0x104, "M", "ą"), - (0x105, "V"), - (0x106, "M", "ć"), - (0x107, "V"), - (0x108, "M", "ĉ"), - (0x109, "V"), - (0x10A, "M", "ċ"), - (0x10B, "V"), - (0x10C, "M", "č"), - (0x10D, "V"), - (0x10E, "M", "ď"), - (0x10F, "V"), - (0x110, "M", "đ"), - (0x111, "V"), - (0x112, "M", "ē"), - (0x113, "V"), - (0x114, "M", "ĕ"), - (0x115, "V"), - (0x116, "M", "ė"), - (0x117, "V"), - (0x118, "M", "ę"), - (0x119, "V"), - (0x11A, "M", "ě"), - (0x11B, "V"), - (0x11C, "M", "ĝ"), - (0x11D, "V"), - (0x11E, "M", "ğ"), - (0x11F, "V"), - (0x120, "M", "ġ"), - (0x121, "V"), - (0x122, "M", "ģ"), - (0x123, "V"), - (0x124, "M", "ĥ"), - (0x125, "V"), - (0x126, "M", "ħ"), - (0x127, "V"), - (0x128, "M", "ĩ"), - (0x129, "V"), - (0x12A, "M", "ī"), - (0x12B, "V"), - ] - - -def _seg_3() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x12C, "M", "ĭ"), - (0x12D, "V"), - (0x12E, "M", "į"), - (0x12F, "V"), - (0x130, "M", "i̇"), - (0x131, "V"), - (0x132, "M", "ij"), - (0x134, "M", "ĵ"), - (0x135, "V"), - (0x136, "M", "ķ"), - (0x137, "V"), - (0x139, "M", "ĺ"), - (0x13A, "V"), - (0x13B, "M", "ļ"), - (0x13C, "V"), - (0x13D, "M", "ľ"), - (0x13E, "V"), - (0x13F, "M", "l·"), - (0x141, "M", "ł"), - (0x142, "V"), - (0x143, "M", "ń"), - (0x144, "V"), - (0x145, "M", "ņ"), - (0x146, "V"), - (0x147, "M", "ň"), - (0x148, "V"), - (0x149, "M", "ʼn"), - (0x14A, "M", "ŋ"), - (0x14B, "V"), - (0x14C, "M", "ō"), - (0x14D, "V"), - (0x14E, "M", "ŏ"), - (0x14F, "V"), - (0x150, "M", "ő"), - (0x151, "V"), - (0x152, "M", "œ"), - (0x153, "V"), - (0x154, "M", "ŕ"), - (0x155, "V"), - (0x156, "M", "ŗ"), - (0x157, "V"), - (0x158, "M", "ř"), - (0x159, "V"), - (0x15A, "M", "ś"), - (0x15B, "V"), - (0x15C, "M", "ŝ"), - (0x15D, "V"), - (0x15E, "M", "ş"), - (0x15F, "V"), - (0x160, "M", "š"), - (0x161, "V"), - (0x162, "M", "ţ"), - (0x163, "V"), - (0x164, "M", "ť"), - (0x165, "V"), - (0x166, "M", "ŧ"), - (0x167, "V"), - (0x168, "M", "ũ"), - (0x169, "V"), - (0x16A, "M", "ū"), - (0x16B, "V"), - (0x16C, "M", "ŭ"), - (0x16D, "V"), - (0x16E, "M", "ů"), - (0x16F, "V"), - (0x170, "M", "ű"), - (0x171, "V"), - (0x172, "M", "ų"), - (0x173, "V"), - (0x174, "M", "ŵ"), - (0x175, "V"), - (0x176, "M", "ŷ"), - (0x177, "V"), - (0x178, "M", "ÿ"), - (0x179, "M", "ź"), - (0x17A, "V"), - (0x17B, "M", "ż"), - (0x17C, "V"), - (0x17D, "M", "ž"), - (0x17E, "V"), - (0x17F, "M", "s"), - (0x180, "V"), - (0x181, "M", "ɓ"), - (0x182, "M", "ƃ"), - (0x183, "V"), - (0x184, "M", "ƅ"), - (0x185, "V"), - (0x186, "M", "ɔ"), - (0x187, "M", "ƈ"), - (0x188, "V"), - (0x189, "M", "ɖ"), - (0x18A, "M", "ɗ"), - (0x18B, "M", "ƌ"), - (0x18C, "V"), - (0x18E, "M", "ǝ"), - (0x18F, "M", "ə"), - (0x190, "M", "ɛ"), - (0x191, "M", "ƒ"), - (0x192, "V"), - (0x193, "M", "ɠ"), - ] - - -def _seg_4() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x194, "M", "ɣ"), - (0x195, "V"), - (0x196, "M", "ɩ"), - (0x197, "M", "ɨ"), - (0x198, "M", "ƙ"), - (0x199, "V"), - (0x19C, "M", "ɯ"), - (0x19D, "M", "ɲ"), - (0x19E, "V"), - (0x19F, "M", "ɵ"), - (0x1A0, "M", "ơ"), - (0x1A1, "V"), - (0x1A2, "M", "ƣ"), - (0x1A3, "V"), - (0x1A4, "M", "ƥ"), - (0x1A5, "V"), - (0x1A6, "M", "ʀ"), - (0x1A7, "M", "ƨ"), - (0x1A8, "V"), - (0x1A9, "M", "ʃ"), - (0x1AA, "V"), - (0x1AC, "M", "ƭ"), - (0x1AD, "V"), - (0x1AE, "M", "ʈ"), - (0x1AF, "M", "ư"), - (0x1B0, "V"), - (0x1B1, "M", "ʊ"), - (0x1B2, "M", "ʋ"), - (0x1B3, "M", "ƴ"), - (0x1B4, "V"), - (0x1B5, "M", "ƶ"), - (0x1B6, "V"), - (0x1B7, "M", "ʒ"), - (0x1B8, "M", "ƹ"), - (0x1B9, "V"), - (0x1BC, "M", "ƽ"), - (0x1BD, "V"), - (0x1C4, "M", "dž"), - (0x1C7, "M", "lj"), - (0x1CA, "M", "nj"), - (0x1CD, "M", "ǎ"), - (0x1CE, "V"), - (0x1CF, "M", "ǐ"), - (0x1D0, "V"), - (0x1D1, "M", "ǒ"), - (0x1D2, "V"), - (0x1D3, "M", "ǔ"), - (0x1D4, "V"), - (0x1D5, "M", "ǖ"), - (0x1D6, "V"), - (0x1D7, "M", "ǘ"), - (0x1D8, "V"), - (0x1D9, "M", "ǚ"), - (0x1DA, "V"), - (0x1DB, "M", "ǜ"), - (0x1DC, "V"), - (0x1DE, "M", "ǟ"), - (0x1DF, "V"), - (0x1E0, "M", "ǡ"), - (0x1E1, "V"), - (0x1E2, "M", "ǣ"), - (0x1E3, "V"), - (0x1E4, "M", "ǥ"), - (0x1E5, "V"), - (0x1E6, "M", "ǧ"), - (0x1E7, "V"), - (0x1E8, "M", "ǩ"), - (0x1E9, "V"), - (0x1EA, "M", "ǫ"), - (0x1EB, "V"), - (0x1EC, "M", "ǭ"), - (0x1ED, "V"), - (0x1EE, "M", "ǯ"), - (0x1EF, "V"), - (0x1F1, "M", "dz"), - (0x1F4, "M", "ǵ"), - (0x1F5, "V"), - (0x1F6, "M", "ƕ"), - (0x1F7, "M", "ƿ"), - (0x1F8, "M", "ǹ"), - (0x1F9, "V"), - (0x1FA, "M", "ǻ"), - (0x1FB, "V"), - (0x1FC, "M", "ǽ"), - (0x1FD, "V"), - (0x1FE, "M", "ǿ"), - (0x1FF, "V"), - (0x200, "M", "ȁ"), - (0x201, "V"), - (0x202, "M", "ȃ"), - (0x203, "V"), - (0x204, "M", "ȅ"), - (0x205, "V"), - (0x206, "M", "ȇ"), - (0x207, "V"), - (0x208, "M", "ȉ"), - (0x209, "V"), - (0x20A, "M", "ȋ"), - (0x20B, "V"), - (0x20C, "M", "ȍ"), - ] - - -def _seg_5() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x20D, "V"), - (0x20E, "M", "ȏ"), - (0x20F, "V"), - (0x210, "M", "ȑ"), - (0x211, "V"), - (0x212, "M", "ȓ"), - (0x213, "V"), - (0x214, "M", "ȕ"), - (0x215, "V"), - (0x216, "M", "ȗ"), - (0x217, "V"), - (0x218, "M", "ș"), - (0x219, "V"), - (0x21A, "M", "ț"), - (0x21B, "V"), - (0x21C, "M", "ȝ"), - (0x21D, "V"), - (0x21E, "M", "ȟ"), - (0x21F, "V"), - (0x220, "M", "ƞ"), - (0x221, "V"), - (0x222, "M", "ȣ"), - (0x223, "V"), - (0x224, "M", "ȥ"), - (0x225, "V"), - (0x226, "M", "ȧ"), - (0x227, "V"), - (0x228, "M", "ȩ"), - (0x229, "V"), - (0x22A, "M", "ȫ"), - (0x22B, "V"), - (0x22C, "M", "ȭ"), - (0x22D, "V"), - (0x22E, "M", "ȯ"), - (0x22F, "V"), - (0x230, "M", "ȱ"), - (0x231, "V"), - (0x232, "M", "ȳ"), - (0x233, "V"), - (0x23A, "M", "ⱥ"), - (0x23B, "M", "ȼ"), - (0x23C, "V"), - (0x23D, "M", "ƚ"), - (0x23E, "M", "ⱦ"), - (0x23F, "V"), - (0x241, "M", "ɂ"), - (0x242, "V"), - (0x243, "M", "ƀ"), - (0x244, "M", "ʉ"), - (0x245, "M", "ʌ"), - (0x246, "M", "ɇ"), - (0x247, "V"), - (0x248, "M", "ɉ"), - (0x249, "V"), - (0x24A, "M", "ɋ"), - (0x24B, "V"), - (0x24C, "M", "ɍ"), - (0x24D, "V"), - (0x24E, "M", "ɏ"), - (0x24F, "V"), - (0x2B0, "M", "h"), - (0x2B1, "M", "ɦ"), - (0x2B2, "M", "j"), - (0x2B3, "M", "r"), - (0x2B4, "M", "ɹ"), - (0x2B5, "M", "ɻ"), - (0x2B6, "M", "ʁ"), - (0x2B7, "M", "w"), - (0x2B8, "M", "y"), - (0x2B9, "V"), - (0x2D8, "M", " ̆"), - (0x2D9, "M", " ̇"), - (0x2DA, "M", " ̊"), - (0x2DB, "M", " ̨"), - (0x2DC, "M", " ̃"), - (0x2DD, "M", " ̋"), - (0x2DE, "V"), - (0x2E0, "M", "ɣ"), - (0x2E1, "M", "l"), - (0x2E2, "M", "s"), - (0x2E3, "M", "x"), - (0x2E4, "M", "ʕ"), - (0x2E5, "V"), - (0x340, "M", "̀"), - (0x341, "M", "́"), - (0x342, "V"), - (0x343, "M", "̓"), - (0x344, "M", "̈́"), - (0x345, "M", "ι"), - (0x346, "V"), - (0x34F, "I"), - (0x350, "V"), - (0x370, "M", "ͱ"), - (0x371, "V"), - (0x372, "M", "ͳ"), - (0x373, "V"), - (0x374, "M", "ʹ"), - (0x375, "V"), - (0x376, "M", "ͷ"), - (0x377, "V"), - ] - - -def _seg_6() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x378, "X"), - (0x37A, "M", " ι"), - (0x37B, "V"), - (0x37E, "M", ";"), - (0x37F, "M", "ϳ"), - (0x380, "X"), - (0x384, "M", " ́"), - (0x385, "M", " ̈́"), - (0x386, "M", "ά"), - (0x387, "M", "·"), - (0x388, "M", "έ"), - (0x389, "M", "ή"), - (0x38A, "M", "ί"), - (0x38B, "X"), - (0x38C, "M", "ό"), - (0x38D, "X"), - (0x38E, "M", "ύ"), - (0x38F, "M", "ώ"), - (0x390, "V"), - (0x391, "M", "α"), - (0x392, "M", "β"), - (0x393, "M", "γ"), - (0x394, "M", "δ"), - (0x395, "M", "ε"), - (0x396, "M", "ζ"), - (0x397, "M", "η"), - (0x398, "M", "θ"), - (0x399, "M", "ι"), - (0x39A, "M", "κ"), - (0x39B, "M", "λ"), - (0x39C, "M", "μ"), - (0x39D, "M", "ν"), - (0x39E, "M", "ξ"), - (0x39F, "M", "ο"), - (0x3A0, "M", "π"), - (0x3A1, "M", "ρ"), - (0x3A2, "X"), - (0x3A3, "M", "σ"), - (0x3A4, "M", "τ"), - (0x3A5, "M", "υ"), - (0x3A6, "M", "φ"), - (0x3A7, "M", "χ"), - (0x3A8, "M", "ψ"), - (0x3A9, "M", "ω"), - (0x3AA, "M", "ϊ"), - (0x3AB, "M", "ϋ"), - (0x3AC, "V"), - (0x3C2, "D", "σ"), - (0x3C3, "V"), - (0x3CF, "M", "ϗ"), - (0x3D0, "M", "β"), - (0x3D1, "M", "θ"), - (0x3D2, "M", "υ"), - (0x3D3, "M", "ύ"), - (0x3D4, "M", "ϋ"), - (0x3D5, "M", "φ"), - (0x3D6, "M", "π"), - (0x3D7, "V"), - (0x3D8, "M", "ϙ"), - (0x3D9, "V"), - (0x3DA, "M", "ϛ"), - (0x3DB, "V"), - (0x3DC, "M", "ϝ"), - (0x3DD, "V"), - (0x3DE, "M", "ϟ"), - (0x3DF, "V"), - (0x3E0, "M", "ϡ"), - (0x3E1, "V"), - (0x3E2, "M", "ϣ"), - (0x3E3, "V"), - (0x3E4, "M", "ϥ"), - (0x3E5, "V"), - (0x3E6, "M", "ϧ"), - (0x3E7, "V"), - (0x3E8, "M", "ϩ"), - (0x3E9, "V"), - (0x3EA, "M", "ϫ"), - (0x3EB, "V"), - (0x3EC, "M", "ϭ"), - (0x3ED, "V"), - (0x3EE, "M", "ϯ"), - (0x3EF, "V"), - (0x3F0, "M", "κ"), - (0x3F1, "M", "ρ"), - (0x3F2, "M", "σ"), - (0x3F3, "V"), - (0x3F4, "M", "θ"), - (0x3F5, "M", "ε"), - (0x3F6, "V"), - (0x3F7, "M", "ϸ"), - (0x3F8, "V"), - (0x3F9, "M", "σ"), - (0x3FA, "M", "ϻ"), - (0x3FB, "V"), - (0x3FD, "M", "ͻ"), - (0x3FE, "M", "ͼ"), - (0x3FF, "M", "ͽ"), - (0x400, "M", "ѐ"), - (0x401, "M", "ё"), - (0x402, "M", "ђ"), - ] - - -def _seg_7() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x403, "M", "ѓ"), - (0x404, "M", "є"), - (0x405, "M", "ѕ"), - (0x406, "M", "і"), - (0x407, "M", "ї"), - (0x408, "M", "ј"), - (0x409, "M", "љ"), - (0x40A, "M", "њ"), - (0x40B, "M", "ћ"), - (0x40C, "M", "ќ"), - (0x40D, "M", "ѝ"), - (0x40E, "M", "ў"), - (0x40F, "M", "џ"), - (0x410, "M", "а"), - (0x411, "M", "б"), - (0x412, "M", "в"), - (0x413, "M", "г"), - (0x414, "M", "д"), - (0x415, "M", "е"), - (0x416, "M", "ж"), - (0x417, "M", "з"), - (0x418, "M", "и"), - (0x419, "M", "й"), - (0x41A, "M", "к"), - (0x41B, "M", "л"), - (0x41C, "M", "м"), - (0x41D, "M", "н"), - (0x41E, "M", "о"), - (0x41F, "M", "п"), - (0x420, "M", "р"), - (0x421, "M", "с"), - (0x422, "M", "т"), - (0x423, "M", "у"), - (0x424, "M", "ф"), - (0x425, "M", "х"), - (0x426, "M", "ц"), - (0x427, "M", "ч"), - (0x428, "M", "ш"), - (0x429, "M", "щ"), - (0x42A, "M", "ъ"), - (0x42B, "M", "ы"), - (0x42C, "M", "ь"), - (0x42D, "M", "э"), - (0x42E, "M", "ю"), - (0x42F, "M", "я"), - (0x430, "V"), - (0x460, "M", "ѡ"), - (0x461, "V"), - (0x462, "M", "ѣ"), - (0x463, "V"), - (0x464, "M", "ѥ"), - (0x465, "V"), - (0x466, "M", "ѧ"), - (0x467, "V"), - (0x468, "M", "ѩ"), - (0x469, "V"), - (0x46A, "M", "ѫ"), - (0x46B, "V"), - (0x46C, "M", "ѭ"), - (0x46D, "V"), - (0x46E, "M", "ѯ"), - (0x46F, "V"), - (0x470, "M", "ѱ"), - (0x471, "V"), - (0x472, "M", "ѳ"), - (0x473, "V"), - (0x474, "M", "ѵ"), - (0x475, "V"), - (0x476, "M", "ѷ"), - (0x477, "V"), - (0x478, "M", "ѹ"), - (0x479, "V"), - (0x47A, "M", "ѻ"), - (0x47B, "V"), - (0x47C, "M", "ѽ"), - (0x47D, "V"), - (0x47E, "M", "ѿ"), - (0x47F, "V"), - (0x480, "M", "ҁ"), - (0x481, "V"), - (0x48A, "M", "ҋ"), - (0x48B, "V"), - (0x48C, "M", "ҍ"), - (0x48D, "V"), - (0x48E, "M", "ҏ"), - (0x48F, "V"), - (0x490, "M", "ґ"), - (0x491, "V"), - (0x492, "M", "ғ"), - (0x493, "V"), - (0x494, "M", "ҕ"), - (0x495, "V"), - (0x496, "M", "җ"), - (0x497, "V"), - (0x498, "M", "ҙ"), - (0x499, "V"), - (0x49A, "M", "қ"), - (0x49B, "V"), - (0x49C, "M", "ҝ"), - (0x49D, "V"), - ] - - -def _seg_8() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x49E, "M", "ҟ"), - (0x49F, "V"), - (0x4A0, "M", "ҡ"), - (0x4A1, "V"), - (0x4A2, "M", "ң"), - (0x4A3, "V"), - (0x4A4, "M", "ҥ"), - (0x4A5, "V"), - (0x4A6, "M", "ҧ"), - (0x4A7, "V"), - (0x4A8, "M", "ҩ"), - (0x4A9, "V"), - (0x4AA, "M", "ҫ"), - (0x4AB, "V"), - (0x4AC, "M", "ҭ"), - (0x4AD, "V"), - (0x4AE, "M", "ү"), - (0x4AF, "V"), - (0x4B0, "M", "ұ"), - (0x4B1, "V"), - (0x4B2, "M", "ҳ"), - (0x4B3, "V"), - (0x4B4, "M", "ҵ"), - (0x4B5, "V"), - (0x4B6, "M", "ҷ"), - (0x4B7, "V"), - (0x4B8, "M", "ҹ"), - (0x4B9, "V"), - (0x4BA, "M", "һ"), - (0x4BB, "V"), - (0x4BC, "M", "ҽ"), - (0x4BD, "V"), - (0x4BE, "M", "ҿ"), - (0x4BF, "V"), - (0x4C0, "M", "ӏ"), - (0x4C1, "M", "ӂ"), - (0x4C2, "V"), - (0x4C3, "M", "ӄ"), - (0x4C4, "V"), - (0x4C5, "M", "ӆ"), - (0x4C6, "V"), - (0x4C7, "M", "ӈ"), - (0x4C8, "V"), - (0x4C9, "M", "ӊ"), - (0x4CA, "V"), - (0x4CB, "M", "ӌ"), - (0x4CC, "V"), - (0x4CD, "M", "ӎ"), - (0x4CE, "V"), - (0x4D0, "M", "ӑ"), - (0x4D1, "V"), - (0x4D2, "M", "ӓ"), - (0x4D3, "V"), - (0x4D4, "M", "ӕ"), - (0x4D5, "V"), - (0x4D6, "M", "ӗ"), - (0x4D7, "V"), - (0x4D8, "M", "ә"), - (0x4D9, "V"), - (0x4DA, "M", "ӛ"), - (0x4DB, "V"), - (0x4DC, "M", "ӝ"), - (0x4DD, "V"), - (0x4DE, "M", "ӟ"), - (0x4DF, "V"), - (0x4E0, "M", "ӡ"), - (0x4E1, "V"), - (0x4E2, "M", "ӣ"), - (0x4E3, "V"), - (0x4E4, "M", "ӥ"), - (0x4E5, "V"), - (0x4E6, "M", "ӧ"), - (0x4E7, "V"), - (0x4E8, "M", "ө"), - (0x4E9, "V"), - (0x4EA, "M", "ӫ"), - (0x4EB, "V"), - (0x4EC, "M", "ӭ"), - (0x4ED, "V"), - (0x4EE, "M", "ӯ"), - (0x4EF, "V"), - (0x4F0, "M", "ӱ"), - (0x4F1, "V"), - (0x4F2, "M", "ӳ"), - (0x4F3, "V"), - (0x4F4, "M", "ӵ"), - (0x4F5, "V"), - (0x4F6, "M", "ӷ"), - (0x4F7, "V"), - (0x4F8, "M", "ӹ"), - (0x4F9, "V"), - (0x4FA, "M", "ӻ"), - (0x4FB, "V"), - (0x4FC, "M", "ӽ"), - (0x4FD, "V"), - (0x4FE, "M", "ӿ"), - (0x4FF, "V"), - (0x500, "M", "ԁ"), - (0x501, "V"), - (0x502, "M", "ԃ"), - ] - - -def _seg_9() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x503, "V"), - (0x504, "M", "ԅ"), - (0x505, "V"), - (0x506, "M", "ԇ"), - (0x507, "V"), - (0x508, "M", "ԉ"), - (0x509, "V"), - (0x50A, "M", "ԋ"), - (0x50B, "V"), - (0x50C, "M", "ԍ"), - (0x50D, "V"), - (0x50E, "M", "ԏ"), - (0x50F, "V"), - (0x510, "M", "ԑ"), - (0x511, "V"), - (0x512, "M", "ԓ"), - (0x513, "V"), - (0x514, "M", "ԕ"), - (0x515, "V"), - (0x516, "M", "ԗ"), - (0x517, "V"), - (0x518, "M", "ԙ"), - (0x519, "V"), - (0x51A, "M", "ԛ"), - (0x51B, "V"), - (0x51C, "M", "ԝ"), - (0x51D, "V"), - (0x51E, "M", "ԟ"), - (0x51F, "V"), - (0x520, "M", "ԡ"), - (0x521, "V"), - (0x522, "M", "ԣ"), - (0x523, "V"), - (0x524, "M", "ԥ"), - (0x525, "V"), - (0x526, "M", "ԧ"), - (0x527, "V"), - (0x528, "M", "ԩ"), - (0x529, "V"), - (0x52A, "M", "ԫ"), - (0x52B, "V"), - (0x52C, "M", "ԭ"), - (0x52D, "V"), - (0x52E, "M", "ԯ"), - (0x52F, "V"), - (0x530, "X"), - (0x531, "M", "ա"), - (0x532, "M", "բ"), - (0x533, "M", "գ"), - (0x534, "M", "դ"), - (0x535, "M", "ե"), - (0x536, "M", "զ"), - (0x537, "M", "է"), - (0x538, "M", "ը"), - (0x539, "M", "թ"), - (0x53A, "M", "ժ"), - (0x53B, "M", "ի"), - (0x53C, "M", "լ"), - (0x53D, "M", "խ"), - (0x53E, "M", "ծ"), - (0x53F, "M", "կ"), - (0x540, "M", "հ"), - (0x541, "M", "ձ"), - (0x542, "M", "ղ"), - (0x543, "M", "ճ"), - (0x544, "M", "մ"), - (0x545, "M", "յ"), - (0x546, "M", "ն"), - (0x547, "M", "շ"), - (0x548, "M", "ո"), - (0x549, "M", "չ"), - (0x54A, "M", "պ"), - (0x54B, "M", "ջ"), - (0x54C, "M", "ռ"), - (0x54D, "M", "ս"), - (0x54E, "M", "վ"), - (0x54F, "M", "տ"), - (0x550, "M", "ր"), - (0x551, "M", "ց"), - (0x552, "M", "ւ"), - (0x553, "M", "փ"), - (0x554, "M", "ք"), - (0x555, "M", "օ"), - (0x556, "M", "ֆ"), - (0x557, "X"), - (0x559, "V"), - (0x587, "M", "եւ"), - (0x588, "V"), - (0x58B, "X"), - (0x58D, "V"), - (0x590, "X"), - (0x591, "V"), - (0x5C8, "X"), - (0x5D0, "V"), - (0x5EB, "X"), - (0x5EF, "V"), - (0x5F5, "X"), - (0x606, "V"), - (0x61C, "X"), - (0x61D, "V"), - ] - - -def _seg_10() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x675, "M", "اٴ"), - (0x676, "M", "وٴ"), - (0x677, "M", "ۇٴ"), - (0x678, "M", "يٴ"), - (0x679, "V"), - (0x6DD, "X"), - (0x6DE, "V"), - (0x70E, "X"), - (0x710, "V"), - (0x74B, "X"), - (0x74D, "V"), - (0x7B2, "X"), - (0x7C0, "V"), - (0x7FB, "X"), - (0x7FD, "V"), - (0x82E, "X"), - (0x830, "V"), - (0x83F, "X"), - (0x840, "V"), - (0x85C, "X"), - (0x85E, "V"), - (0x85F, "X"), - (0x860, "V"), - (0x86B, "X"), - (0x870, "V"), - (0x88F, "X"), - (0x897, "V"), - (0x8E2, "X"), - (0x8E3, "V"), - (0x958, "M", "क़"), - (0x959, "M", "ख़"), - (0x95A, "M", "ग़"), - (0x95B, "M", "ज़"), - (0x95C, "M", "ड़"), - (0x95D, "M", "ढ़"), - (0x95E, "M", "फ़"), - (0x95F, "M", "य़"), - (0x960, "V"), - (0x984, "X"), - (0x985, "V"), - (0x98D, "X"), - (0x98F, "V"), - (0x991, "X"), - (0x993, "V"), - (0x9A9, "X"), - (0x9AA, "V"), - (0x9B1, "X"), - (0x9B2, "V"), - (0x9B3, "X"), - (0x9B6, "V"), - (0x9BA, "X"), - (0x9BC, "V"), - (0x9C5, "X"), - (0x9C7, "V"), - (0x9C9, "X"), - (0x9CB, "V"), - (0x9CF, "X"), - (0x9D7, "V"), - (0x9D8, "X"), - (0x9DC, "M", "ড়"), - (0x9DD, "M", "ঢ়"), - (0x9DE, "X"), - (0x9DF, "M", "য়"), - (0x9E0, "V"), - (0x9E4, "X"), - (0x9E6, "V"), - (0x9FF, "X"), - (0xA01, "V"), - (0xA04, "X"), - (0xA05, "V"), - (0xA0B, "X"), - (0xA0F, "V"), - (0xA11, "X"), - (0xA13, "V"), - (0xA29, "X"), - (0xA2A, "V"), - (0xA31, "X"), - (0xA32, "V"), - (0xA33, "M", "ਲ਼"), - (0xA34, "X"), - (0xA35, "V"), - (0xA36, "M", "ਸ਼"), - (0xA37, "X"), - (0xA38, "V"), - (0xA3A, "X"), - (0xA3C, "V"), - (0xA3D, "X"), - (0xA3E, "V"), - (0xA43, "X"), - (0xA47, "V"), - (0xA49, "X"), - (0xA4B, "V"), - (0xA4E, "X"), - (0xA51, "V"), - (0xA52, "X"), - (0xA59, "M", "ਖ਼"), - (0xA5A, "M", "ਗ਼"), - (0xA5B, "M", "ਜ਼"), - (0xA5C, "V"), - (0xA5D, "X"), - ] - - -def _seg_11() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xA5E, "M", "ਫ਼"), - (0xA5F, "X"), - (0xA66, "V"), - (0xA77, "X"), - (0xA81, "V"), - (0xA84, "X"), - (0xA85, "V"), - (0xA8E, "X"), - (0xA8F, "V"), - (0xA92, "X"), - (0xA93, "V"), - (0xAA9, "X"), - (0xAAA, "V"), - (0xAB1, "X"), - (0xAB2, "V"), - (0xAB4, "X"), - (0xAB5, "V"), - (0xABA, "X"), - (0xABC, "V"), - (0xAC6, "X"), - (0xAC7, "V"), - (0xACA, "X"), - (0xACB, "V"), - (0xACE, "X"), - (0xAD0, "V"), - (0xAD1, "X"), - (0xAE0, "V"), - (0xAE4, "X"), - (0xAE6, "V"), - (0xAF2, "X"), - (0xAF9, "V"), - (0xB00, "X"), - (0xB01, "V"), - (0xB04, "X"), - (0xB05, "V"), - (0xB0D, "X"), - (0xB0F, "V"), - (0xB11, "X"), - (0xB13, "V"), - (0xB29, "X"), - (0xB2A, "V"), - (0xB31, "X"), - (0xB32, "V"), - (0xB34, "X"), - (0xB35, "V"), - (0xB3A, "X"), - (0xB3C, "V"), - (0xB45, "X"), - (0xB47, "V"), - (0xB49, "X"), - (0xB4B, "V"), - (0xB4E, "X"), - (0xB55, "V"), - (0xB58, "X"), - (0xB5C, "M", "ଡ଼"), - (0xB5D, "M", "ଢ଼"), - (0xB5E, "X"), - (0xB5F, "V"), - (0xB64, "X"), - (0xB66, "V"), - (0xB78, "X"), - (0xB82, "V"), - (0xB84, "X"), - (0xB85, "V"), - (0xB8B, "X"), - (0xB8E, "V"), - (0xB91, "X"), - (0xB92, "V"), - (0xB96, "X"), - (0xB99, "V"), - (0xB9B, "X"), - (0xB9C, "V"), - (0xB9D, "X"), - (0xB9E, "V"), - (0xBA0, "X"), - (0xBA3, "V"), - (0xBA5, "X"), - (0xBA8, "V"), - (0xBAB, "X"), - (0xBAE, "V"), - (0xBBA, "X"), - (0xBBE, "V"), - (0xBC3, "X"), - (0xBC6, "V"), - (0xBC9, "X"), - (0xBCA, "V"), - (0xBCE, "X"), - (0xBD0, "V"), - (0xBD1, "X"), - (0xBD7, "V"), - (0xBD8, "X"), - (0xBE6, "V"), - (0xBFB, "X"), - (0xC00, "V"), - (0xC0D, "X"), - (0xC0E, "V"), - (0xC11, "X"), - (0xC12, "V"), - (0xC29, "X"), - (0xC2A, "V"), - ] - - -def _seg_12() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xC3A, "X"), - (0xC3C, "V"), - (0xC45, "X"), - (0xC46, "V"), - (0xC49, "X"), - (0xC4A, "V"), - (0xC4E, "X"), - (0xC55, "V"), - (0xC57, "X"), - (0xC58, "V"), - (0xC5B, "X"), - (0xC5D, "V"), - (0xC5E, "X"), - (0xC60, "V"), - (0xC64, "X"), - (0xC66, "V"), - (0xC70, "X"), - (0xC77, "V"), - (0xC8D, "X"), - (0xC8E, "V"), - (0xC91, "X"), - (0xC92, "V"), - (0xCA9, "X"), - (0xCAA, "V"), - (0xCB4, "X"), - (0xCB5, "V"), - (0xCBA, "X"), - (0xCBC, "V"), - (0xCC5, "X"), - (0xCC6, "V"), - (0xCC9, "X"), - (0xCCA, "V"), - (0xCCE, "X"), - (0xCD5, "V"), - (0xCD7, "X"), - (0xCDD, "V"), - (0xCDF, "X"), - (0xCE0, "V"), - (0xCE4, "X"), - (0xCE6, "V"), - (0xCF0, "X"), - (0xCF1, "V"), - (0xCF4, "X"), - (0xD00, "V"), - (0xD0D, "X"), - (0xD0E, "V"), - (0xD11, "X"), - (0xD12, "V"), - (0xD45, "X"), - (0xD46, "V"), - (0xD49, "X"), - (0xD4A, "V"), - (0xD50, "X"), - (0xD54, "V"), - (0xD64, "X"), - (0xD66, "V"), - (0xD80, "X"), - (0xD81, "V"), - (0xD84, "X"), - (0xD85, "V"), - (0xD97, "X"), - (0xD9A, "V"), - (0xDB2, "X"), - (0xDB3, "V"), - (0xDBC, "X"), - (0xDBD, "V"), - (0xDBE, "X"), - (0xDC0, "V"), - (0xDC7, "X"), - (0xDCA, "V"), - (0xDCB, "X"), - (0xDCF, "V"), - (0xDD5, "X"), - (0xDD6, "V"), - (0xDD7, "X"), - (0xDD8, "V"), - (0xDE0, "X"), - (0xDE6, "V"), - (0xDF0, "X"), - (0xDF2, "V"), - (0xDF5, "X"), - (0xE01, "V"), - (0xE33, "M", "ํา"), - (0xE34, "V"), - (0xE3B, "X"), - (0xE3F, "V"), - (0xE5C, "X"), - (0xE81, "V"), - (0xE83, "X"), - (0xE84, "V"), - (0xE85, "X"), - (0xE86, "V"), - (0xE8B, "X"), - (0xE8C, "V"), - (0xEA4, "X"), - (0xEA5, "V"), - (0xEA6, "X"), - (0xEA7, "V"), - (0xEB3, "M", "ໍາ"), - (0xEB4, "V"), - ] - - -def _seg_13() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xEBE, "X"), - (0xEC0, "V"), - (0xEC5, "X"), - (0xEC6, "V"), - (0xEC7, "X"), - (0xEC8, "V"), - (0xECF, "X"), - (0xED0, "V"), - (0xEDA, "X"), - (0xEDC, "M", "ຫນ"), - (0xEDD, "M", "ຫມ"), - (0xEDE, "V"), - (0xEE0, "X"), - (0xF00, "V"), - (0xF0C, "M", "་"), - (0xF0D, "V"), - (0xF43, "M", "གྷ"), - (0xF44, "V"), - (0xF48, "X"), - (0xF49, "V"), - (0xF4D, "M", "ཌྷ"), - (0xF4E, "V"), - (0xF52, "M", "དྷ"), - (0xF53, "V"), - (0xF57, "M", "བྷ"), - (0xF58, "V"), - (0xF5C, "M", "ཛྷ"), - (0xF5D, "V"), - (0xF69, "M", "ཀྵ"), - (0xF6A, "V"), - (0xF6D, "X"), - (0xF71, "V"), - (0xF73, "M", "ཱི"), - (0xF74, "V"), - (0xF75, "M", "ཱུ"), - (0xF76, "M", "ྲྀ"), - (0xF77, "M", "ྲཱྀ"), - (0xF78, "M", "ླྀ"), - (0xF79, "M", "ླཱྀ"), - (0xF7A, "V"), - (0xF81, "M", "ཱྀ"), - (0xF82, "V"), - (0xF93, "M", "ྒྷ"), - (0xF94, "V"), - (0xF98, "X"), - (0xF99, "V"), - (0xF9D, "M", "ྜྷ"), - (0xF9E, "V"), - (0xFA2, "M", "ྡྷ"), - (0xFA3, "V"), - (0xFA7, "M", "ྦྷ"), - (0xFA8, "V"), - (0xFAC, "M", "ྫྷ"), - (0xFAD, "V"), - (0xFB9, "M", "ྐྵ"), - (0xFBA, "V"), - (0xFBD, "X"), - (0xFBE, "V"), - (0xFCD, "X"), - (0xFCE, "V"), - (0xFDB, "X"), - (0x1000, "V"), - (0x10A0, "M", "ⴀ"), - (0x10A1, "M", "ⴁ"), - (0x10A2, "M", "ⴂ"), - (0x10A3, "M", "ⴃ"), - (0x10A4, "M", "ⴄ"), - (0x10A5, "M", "ⴅ"), - (0x10A6, "M", "ⴆ"), - (0x10A7, "M", "ⴇ"), - (0x10A8, "M", "ⴈ"), - (0x10A9, "M", "ⴉ"), - (0x10AA, "M", "ⴊ"), - (0x10AB, "M", "ⴋ"), - (0x10AC, "M", "ⴌ"), - (0x10AD, "M", "ⴍ"), - (0x10AE, "M", "ⴎ"), - (0x10AF, "M", "ⴏ"), - (0x10B0, "M", "ⴐ"), - (0x10B1, "M", "ⴑ"), - (0x10B2, "M", "ⴒ"), - (0x10B3, "M", "ⴓ"), - (0x10B4, "M", "ⴔ"), - (0x10B5, "M", "ⴕ"), - (0x10B6, "M", "ⴖ"), - (0x10B7, "M", "ⴗ"), - (0x10B8, "M", "ⴘ"), - (0x10B9, "M", "ⴙ"), - (0x10BA, "M", "ⴚ"), - (0x10BB, "M", "ⴛ"), - (0x10BC, "M", "ⴜ"), - (0x10BD, "M", "ⴝ"), - (0x10BE, "M", "ⴞ"), - (0x10BF, "M", "ⴟ"), - (0x10C0, "M", "ⴠ"), - (0x10C1, "M", "ⴡ"), - (0x10C2, "M", "ⴢ"), - (0x10C3, "M", "ⴣ"), - (0x10C4, "M", "ⴤ"), - (0x10C5, "M", "ⴥ"), - ] - - -def _seg_14() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x10C6, "X"), - (0x10C7, "M", "ⴧ"), - (0x10C8, "X"), - (0x10CD, "M", "ⴭ"), - (0x10CE, "X"), - (0x10D0, "V"), - (0x10FC, "M", "ნ"), - (0x10FD, "V"), - (0x115F, "I"), - (0x1161, "V"), - (0x1249, "X"), - (0x124A, "V"), - (0x124E, "X"), - (0x1250, "V"), - (0x1257, "X"), - (0x1258, "V"), - (0x1259, "X"), - (0x125A, "V"), - (0x125E, "X"), - (0x1260, "V"), - (0x1289, "X"), - (0x128A, "V"), - (0x128E, "X"), - (0x1290, "V"), - (0x12B1, "X"), - (0x12B2, "V"), - (0x12B6, "X"), - (0x12B8, "V"), - (0x12BF, "X"), - (0x12C0, "V"), - (0x12C1, "X"), - (0x12C2, "V"), - (0x12C6, "X"), - (0x12C8, "V"), - (0x12D7, "X"), - (0x12D8, "V"), - (0x1311, "X"), - (0x1312, "V"), - (0x1316, "X"), - (0x1318, "V"), - (0x135B, "X"), - (0x135D, "V"), - (0x137D, "X"), - (0x1380, "V"), - (0x139A, "X"), - (0x13A0, "V"), - (0x13F6, "X"), - (0x13F8, "M", "Ᏸ"), - (0x13F9, "M", "Ᏹ"), - (0x13FA, "M", "Ᏺ"), - (0x13FB, "M", "Ᏻ"), - (0x13FC, "M", "Ᏼ"), - (0x13FD, "M", "Ᏽ"), - (0x13FE, "X"), - (0x1400, "V"), - (0x1680, "X"), - (0x1681, "V"), - (0x169D, "X"), - (0x16A0, "V"), - (0x16F9, "X"), - (0x1700, "V"), - (0x1716, "X"), - (0x171F, "V"), - (0x1737, "X"), - (0x1740, "V"), - (0x1754, "X"), - (0x1760, "V"), - (0x176D, "X"), - (0x176E, "V"), - (0x1771, "X"), - (0x1772, "V"), - (0x1774, "X"), - (0x1780, "V"), - (0x17B4, "I"), - (0x17B6, "V"), - (0x17DE, "X"), - (0x17E0, "V"), - (0x17EA, "X"), - (0x17F0, "V"), - (0x17FA, "X"), - (0x1800, "V"), - (0x180B, "I"), - (0x1810, "V"), - (0x181A, "X"), - (0x1820, "V"), - (0x1879, "X"), - (0x1880, "V"), - (0x18AB, "X"), - (0x18B0, "V"), - (0x18F6, "X"), - (0x1900, "V"), - (0x191F, "X"), - (0x1920, "V"), - (0x192C, "X"), - (0x1930, "V"), - (0x193C, "X"), - (0x1940, "V"), - (0x1941, "X"), - (0x1944, "V"), - (0x196E, "X"), - ] - - -def _seg_15() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1970, "V"), - (0x1975, "X"), - (0x1980, "V"), - (0x19AC, "X"), - (0x19B0, "V"), - (0x19CA, "X"), - (0x19D0, "V"), - (0x19DB, "X"), - (0x19DE, "V"), - (0x1A1C, "X"), - (0x1A1E, "V"), - (0x1A5F, "X"), - (0x1A60, "V"), - (0x1A7D, "X"), - (0x1A7F, "V"), - (0x1A8A, "X"), - (0x1A90, "V"), - (0x1A9A, "X"), - (0x1AA0, "V"), - (0x1AAE, "X"), - (0x1AB0, "V"), - (0x1ACF, "X"), - (0x1B00, "V"), - (0x1B4D, "X"), - (0x1B4E, "V"), - (0x1BF4, "X"), - (0x1BFC, "V"), - (0x1C38, "X"), - (0x1C3B, "V"), - (0x1C4A, "X"), - (0x1C4D, "V"), - (0x1C80, "M", "в"), - (0x1C81, "M", "д"), - (0x1C82, "M", "о"), - (0x1C83, "M", "с"), - (0x1C84, "M", "т"), - (0x1C86, "M", "ъ"), - (0x1C87, "M", "ѣ"), - (0x1C88, "M", "ꙋ"), - (0x1C89, "M", "ᲊ"), - (0x1C8A, "V"), - (0x1C8B, "X"), - (0x1C90, "M", "ა"), - (0x1C91, "M", "ბ"), - (0x1C92, "M", "გ"), - (0x1C93, "M", "დ"), - (0x1C94, "M", "ე"), - (0x1C95, "M", "ვ"), - (0x1C96, "M", "ზ"), - (0x1C97, "M", "თ"), - (0x1C98, "M", "ი"), - (0x1C99, "M", "კ"), - (0x1C9A, "M", "ლ"), - (0x1C9B, "M", "მ"), - (0x1C9C, "M", "ნ"), - (0x1C9D, "M", "ო"), - (0x1C9E, "M", "პ"), - (0x1C9F, "M", "ჟ"), - (0x1CA0, "M", "რ"), - (0x1CA1, "M", "ს"), - (0x1CA2, "M", "ტ"), - (0x1CA3, "M", "უ"), - (0x1CA4, "M", "ფ"), - (0x1CA5, "M", "ქ"), - (0x1CA6, "M", "ღ"), - (0x1CA7, "M", "ყ"), - (0x1CA8, "M", "შ"), - (0x1CA9, "M", "ჩ"), - (0x1CAA, "M", "ც"), - (0x1CAB, "M", "ძ"), - (0x1CAC, "M", "წ"), - (0x1CAD, "M", "ჭ"), - (0x1CAE, "M", "ხ"), - (0x1CAF, "M", "ჯ"), - (0x1CB0, "M", "ჰ"), - (0x1CB1, "M", "ჱ"), - (0x1CB2, "M", "ჲ"), - (0x1CB3, "M", "ჳ"), - (0x1CB4, "M", "ჴ"), - (0x1CB5, "M", "ჵ"), - (0x1CB6, "M", "ჶ"), - (0x1CB7, "M", "ჷ"), - (0x1CB8, "M", "ჸ"), - (0x1CB9, "M", "ჹ"), - (0x1CBA, "M", "ჺ"), - (0x1CBB, "X"), - (0x1CBD, "M", "ჽ"), - (0x1CBE, "M", "ჾ"), - (0x1CBF, "M", "ჿ"), - (0x1CC0, "V"), - (0x1CC8, "X"), - (0x1CD0, "V"), - (0x1CFB, "X"), - (0x1D00, "V"), - (0x1D2C, "M", "a"), - (0x1D2D, "M", "æ"), - (0x1D2E, "M", "b"), - (0x1D2F, "V"), - (0x1D30, "M", "d"), - (0x1D31, "M", "e"), - ] - - -def _seg_16() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D32, "M", "ǝ"), - (0x1D33, "M", "g"), - (0x1D34, "M", "h"), - (0x1D35, "M", "i"), - (0x1D36, "M", "j"), - (0x1D37, "M", "k"), - (0x1D38, "M", "l"), - (0x1D39, "M", "m"), - (0x1D3A, "M", "n"), - (0x1D3B, "V"), - (0x1D3C, "M", "o"), - (0x1D3D, "M", "ȣ"), - (0x1D3E, "M", "p"), - (0x1D3F, "M", "r"), - (0x1D40, "M", "t"), - (0x1D41, "M", "u"), - (0x1D42, "M", "w"), - (0x1D43, "M", "a"), - (0x1D44, "M", "ɐ"), - (0x1D45, "M", "ɑ"), - (0x1D46, "M", "ᴂ"), - (0x1D47, "M", "b"), - (0x1D48, "M", "d"), - (0x1D49, "M", "e"), - (0x1D4A, "M", "ə"), - (0x1D4B, "M", "ɛ"), - (0x1D4C, "M", "ɜ"), - (0x1D4D, "M", "g"), - (0x1D4E, "V"), - (0x1D4F, "M", "k"), - (0x1D50, "M", "m"), - (0x1D51, "M", "ŋ"), - (0x1D52, "M", "o"), - (0x1D53, "M", "ɔ"), - (0x1D54, "M", "ᴖ"), - (0x1D55, "M", "ᴗ"), - (0x1D56, "M", "p"), - (0x1D57, "M", "t"), - (0x1D58, "M", "u"), - (0x1D59, "M", "ᴝ"), - (0x1D5A, "M", "ɯ"), - (0x1D5B, "M", "v"), - (0x1D5C, "M", "ᴥ"), - (0x1D5D, "M", "β"), - (0x1D5E, "M", "γ"), - (0x1D5F, "M", "δ"), - (0x1D60, "M", "φ"), - (0x1D61, "M", "χ"), - (0x1D62, "M", "i"), - (0x1D63, "M", "r"), - (0x1D64, "M", "u"), - (0x1D65, "M", "v"), - (0x1D66, "M", "β"), - (0x1D67, "M", "γ"), - (0x1D68, "M", "ρ"), - (0x1D69, "M", "φ"), - (0x1D6A, "M", "χ"), - (0x1D6B, "V"), - (0x1D78, "M", "н"), - (0x1D79, "V"), - (0x1D9B, "M", "ɒ"), - (0x1D9C, "M", "c"), - (0x1D9D, "M", "ɕ"), - (0x1D9E, "M", "ð"), - (0x1D9F, "M", "ɜ"), - (0x1DA0, "M", "f"), - (0x1DA1, "M", "ɟ"), - (0x1DA2, "M", "ɡ"), - (0x1DA3, "M", "ɥ"), - (0x1DA4, "M", "ɨ"), - (0x1DA5, "M", "ɩ"), - (0x1DA6, "M", "ɪ"), - (0x1DA7, "M", "ᵻ"), - (0x1DA8, "M", "ʝ"), - (0x1DA9, "M", "ɭ"), - (0x1DAA, "M", "ᶅ"), - (0x1DAB, "M", "ʟ"), - (0x1DAC, "M", "ɱ"), - (0x1DAD, "M", "ɰ"), - (0x1DAE, "M", "ɲ"), - (0x1DAF, "M", "ɳ"), - (0x1DB0, "M", "ɴ"), - (0x1DB1, "M", "ɵ"), - (0x1DB2, "M", "ɸ"), - (0x1DB3, "M", "ʂ"), - (0x1DB4, "M", "ʃ"), - (0x1DB5, "M", "ƫ"), - (0x1DB6, "M", "ʉ"), - (0x1DB7, "M", "ʊ"), - (0x1DB8, "M", "ᴜ"), - (0x1DB9, "M", "ʋ"), - (0x1DBA, "M", "ʌ"), - (0x1DBB, "M", "z"), - (0x1DBC, "M", "ʐ"), - (0x1DBD, "M", "ʑ"), - (0x1DBE, "M", "ʒ"), - (0x1DBF, "M", "θ"), - (0x1DC0, "V"), - (0x1E00, "M", "ḁ"), - (0x1E01, "V"), - ] - - -def _seg_17() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1E02, "M", "ḃ"), - (0x1E03, "V"), - (0x1E04, "M", "ḅ"), - (0x1E05, "V"), - (0x1E06, "M", "ḇ"), - (0x1E07, "V"), - (0x1E08, "M", "ḉ"), - (0x1E09, "V"), - (0x1E0A, "M", "ḋ"), - (0x1E0B, "V"), - (0x1E0C, "M", "ḍ"), - (0x1E0D, "V"), - (0x1E0E, "M", "ḏ"), - (0x1E0F, "V"), - (0x1E10, "M", "ḑ"), - (0x1E11, "V"), - (0x1E12, "M", "ḓ"), - (0x1E13, "V"), - (0x1E14, "M", "ḕ"), - (0x1E15, "V"), - (0x1E16, "M", "ḗ"), - (0x1E17, "V"), - (0x1E18, "M", "ḙ"), - (0x1E19, "V"), - (0x1E1A, "M", "ḛ"), - (0x1E1B, "V"), - (0x1E1C, "M", "ḝ"), - (0x1E1D, "V"), - (0x1E1E, "M", "ḟ"), - (0x1E1F, "V"), - (0x1E20, "M", "ḡ"), - (0x1E21, "V"), - (0x1E22, "M", "ḣ"), - (0x1E23, "V"), - (0x1E24, "M", "ḥ"), - (0x1E25, "V"), - (0x1E26, "M", "ḧ"), - (0x1E27, "V"), - (0x1E28, "M", "ḩ"), - (0x1E29, "V"), - (0x1E2A, "M", "ḫ"), - (0x1E2B, "V"), - (0x1E2C, "M", "ḭ"), - (0x1E2D, "V"), - (0x1E2E, "M", "ḯ"), - (0x1E2F, "V"), - (0x1E30, "M", "ḱ"), - (0x1E31, "V"), - (0x1E32, "M", "ḳ"), - (0x1E33, "V"), - (0x1E34, "M", "ḵ"), - (0x1E35, "V"), - (0x1E36, "M", "ḷ"), - (0x1E37, "V"), - (0x1E38, "M", "ḹ"), - (0x1E39, "V"), - (0x1E3A, "M", "ḻ"), - (0x1E3B, "V"), - (0x1E3C, "M", "ḽ"), - (0x1E3D, "V"), - (0x1E3E, "M", "ḿ"), - (0x1E3F, "V"), - (0x1E40, "M", "ṁ"), - (0x1E41, "V"), - (0x1E42, "M", "ṃ"), - (0x1E43, "V"), - (0x1E44, "M", "ṅ"), - (0x1E45, "V"), - (0x1E46, "M", "ṇ"), - (0x1E47, "V"), - (0x1E48, "M", "ṉ"), - (0x1E49, "V"), - (0x1E4A, "M", "ṋ"), - (0x1E4B, "V"), - (0x1E4C, "M", "ṍ"), - (0x1E4D, "V"), - (0x1E4E, "M", "ṏ"), - (0x1E4F, "V"), - (0x1E50, "M", "ṑ"), - (0x1E51, "V"), - (0x1E52, "M", "ṓ"), - (0x1E53, "V"), - (0x1E54, "M", "ṕ"), - (0x1E55, "V"), - (0x1E56, "M", "ṗ"), - (0x1E57, "V"), - (0x1E58, "M", "ṙ"), - (0x1E59, "V"), - (0x1E5A, "M", "ṛ"), - (0x1E5B, "V"), - (0x1E5C, "M", "ṝ"), - (0x1E5D, "V"), - (0x1E5E, "M", "ṟ"), - (0x1E5F, "V"), - (0x1E60, "M", "ṡ"), - (0x1E61, "V"), - (0x1E62, "M", "ṣ"), - (0x1E63, "V"), - (0x1E64, "M", "ṥ"), - (0x1E65, "V"), - ] - - -def _seg_18() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1E66, "M", "ṧ"), - (0x1E67, "V"), - (0x1E68, "M", "ṩ"), - (0x1E69, "V"), - (0x1E6A, "M", "ṫ"), - (0x1E6B, "V"), - (0x1E6C, "M", "ṭ"), - (0x1E6D, "V"), - (0x1E6E, "M", "ṯ"), - (0x1E6F, "V"), - (0x1E70, "M", "ṱ"), - (0x1E71, "V"), - (0x1E72, "M", "ṳ"), - (0x1E73, "V"), - (0x1E74, "M", "ṵ"), - (0x1E75, "V"), - (0x1E76, "M", "ṷ"), - (0x1E77, "V"), - (0x1E78, "M", "ṹ"), - (0x1E79, "V"), - (0x1E7A, "M", "ṻ"), - (0x1E7B, "V"), - (0x1E7C, "M", "ṽ"), - (0x1E7D, "V"), - (0x1E7E, "M", "ṿ"), - (0x1E7F, "V"), - (0x1E80, "M", "ẁ"), - (0x1E81, "V"), - (0x1E82, "M", "ẃ"), - (0x1E83, "V"), - (0x1E84, "M", "ẅ"), - (0x1E85, "V"), - (0x1E86, "M", "ẇ"), - (0x1E87, "V"), - (0x1E88, "M", "ẉ"), - (0x1E89, "V"), - (0x1E8A, "M", "ẋ"), - (0x1E8B, "V"), - (0x1E8C, "M", "ẍ"), - (0x1E8D, "V"), - (0x1E8E, "M", "ẏ"), - (0x1E8F, "V"), - (0x1E90, "M", "ẑ"), - (0x1E91, "V"), - (0x1E92, "M", "ẓ"), - (0x1E93, "V"), - (0x1E94, "M", "ẕ"), - (0x1E95, "V"), - (0x1E9A, "M", "aʾ"), - (0x1E9B, "M", "ṡ"), - (0x1E9C, "V"), - (0x1E9E, "M", "ß"), - (0x1E9F, "V"), - (0x1EA0, "M", "ạ"), - (0x1EA1, "V"), - (0x1EA2, "M", "ả"), - (0x1EA3, "V"), - (0x1EA4, "M", "ấ"), - (0x1EA5, "V"), - (0x1EA6, "M", "ầ"), - (0x1EA7, "V"), - (0x1EA8, "M", "ẩ"), - (0x1EA9, "V"), - (0x1EAA, "M", "ẫ"), - (0x1EAB, "V"), - (0x1EAC, "M", "ậ"), - (0x1EAD, "V"), - (0x1EAE, "M", "ắ"), - (0x1EAF, "V"), - (0x1EB0, "M", "ằ"), - (0x1EB1, "V"), - (0x1EB2, "M", "ẳ"), - (0x1EB3, "V"), - (0x1EB4, "M", "ẵ"), - (0x1EB5, "V"), - (0x1EB6, "M", "ặ"), - (0x1EB7, "V"), - (0x1EB8, "M", "ẹ"), - (0x1EB9, "V"), - (0x1EBA, "M", "ẻ"), - (0x1EBB, "V"), - (0x1EBC, "M", "ẽ"), - (0x1EBD, "V"), - (0x1EBE, "M", "ế"), - (0x1EBF, "V"), - (0x1EC0, "M", "ề"), - (0x1EC1, "V"), - (0x1EC2, "M", "ể"), - (0x1EC3, "V"), - (0x1EC4, "M", "ễ"), - (0x1EC5, "V"), - (0x1EC6, "M", "ệ"), - (0x1EC7, "V"), - (0x1EC8, "M", "ỉ"), - (0x1EC9, "V"), - (0x1ECA, "M", "ị"), - (0x1ECB, "V"), - (0x1ECC, "M", "ọ"), - (0x1ECD, "V"), - (0x1ECE, "M", "ỏ"), - ] - - -def _seg_19() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1ECF, "V"), - (0x1ED0, "M", "ố"), - (0x1ED1, "V"), - (0x1ED2, "M", "ồ"), - (0x1ED3, "V"), - (0x1ED4, "M", "ổ"), - (0x1ED5, "V"), - (0x1ED6, "M", "ỗ"), - (0x1ED7, "V"), - (0x1ED8, "M", "ộ"), - (0x1ED9, "V"), - (0x1EDA, "M", "ớ"), - (0x1EDB, "V"), - (0x1EDC, "M", "ờ"), - (0x1EDD, "V"), - (0x1EDE, "M", "ở"), - (0x1EDF, "V"), - (0x1EE0, "M", "ỡ"), - (0x1EE1, "V"), - (0x1EE2, "M", "ợ"), - (0x1EE3, "V"), - (0x1EE4, "M", "ụ"), - (0x1EE5, "V"), - (0x1EE6, "M", "ủ"), - (0x1EE7, "V"), - (0x1EE8, "M", "ứ"), - (0x1EE9, "V"), - (0x1EEA, "M", "ừ"), - (0x1EEB, "V"), - (0x1EEC, "M", "ử"), - (0x1EED, "V"), - (0x1EEE, "M", "ữ"), - (0x1EEF, "V"), - (0x1EF0, "M", "ự"), - (0x1EF1, "V"), - (0x1EF2, "M", "ỳ"), - (0x1EF3, "V"), - (0x1EF4, "M", "ỵ"), - (0x1EF5, "V"), - (0x1EF6, "M", "ỷ"), - (0x1EF7, "V"), - (0x1EF8, "M", "ỹ"), - (0x1EF9, "V"), - (0x1EFA, "M", "ỻ"), - (0x1EFB, "V"), - (0x1EFC, "M", "ỽ"), - (0x1EFD, "V"), - (0x1EFE, "M", "ỿ"), - (0x1EFF, "V"), - (0x1F08, "M", "ἀ"), - (0x1F09, "M", "ἁ"), - (0x1F0A, "M", "ἂ"), - (0x1F0B, "M", "ἃ"), - (0x1F0C, "M", "ἄ"), - (0x1F0D, "M", "ἅ"), - (0x1F0E, "M", "ἆ"), - (0x1F0F, "M", "ἇ"), - (0x1F10, "V"), - (0x1F16, "X"), - (0x1F18, "M", "ἐ"), - (0x1F19, "M", "ἑ"), - (0x1F1A, "M", "ἒ"), - (0x1F1B, "M", "ἓ"), - (0x1F1C, "M", "ἔ"), - (0x1F1D, "M", "ἕ"), - (0x1F1E, "X"), - (0x1F20, "V"), - (0x1F28, "M", "ἠ"), - (0x1F29, "M", "ἡ"), - (0x1F2A, "M", "ἢ"), - (0x1F2B, "M", "ἣ"), - (0x1F2C, "M", "ἤ"), - (0x1F2D, "M", "ἥ"), - (0x1F2E, "M", "ἦ"), - (0x1F2F, "M", "ἧ"), - (0x1F30, "V"), - (0x1F38, "M", "ἰ"), - (0x1F39, "M", "ἱ"), - (0x1F3A, "M", "ἲ"), - (0x1F3B, "M", "ἳ"), - (0x1F3C, "M", "ἴ"), - (0x1F3D, "M", "ἵ"), - (0x1F3E, "M", "ἶ"), - (0x1F3F, "M", "ἷ"), - (0x1F40, "V"), - (0x1F46, "X"), - (0x1F48, "M", "ὀ"), - (0x1F49, "M", "ὁ"), - (0x1F4A, "M", "ὂ"), - (0x1F4B, "M", "ὃ"), - (0x1F4C, "M", "ὄ"), - (0x1F4D, "M", "ὅ"), - (0x1F4E, "X"), - (0x1F50, "V"), - (0x1F58, "X"), - (0x1F59, "M", "ὑ"), - (0x1F5A, "X"), - (0x1F5B, "M", "ὓ"), - (0x1F5C, "X"), - (0x1F5D, "M", "ὕ"), - ] - - -def _seg_20() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1F5E, "X"), - (0x1F5F, "M", "ὗ"), - (0x1F60, "V"), - (0x1F68, "M", "ὠ"), - (0x1F69, "M", "ὡ"), - (0x1F6A, "M", "ὢ"), - (0x1F6B, "M", "ὣ"), - (0x1F6C, "M", "ὤ"), - (0x1F6D, "M", "ὥ"), - (0x1F6E, "M", "ὦ"), - (0x1F6F, "M", "ὧ"), - (0x1F70, "V"), - (0x1F71, "M", "ά"), - (0x1F72, "V"), - (0x1F73, "M", "έ"), - (0x1F74, "V"), - (0x1F75, "M", "ή"), - (0x1F76, "V"), - (0x1F77, "M", "ί"), - (0x1F78, "V"), - (0x1F79, "M", "ό"), - (0x1F7A, "V"), - (0x1F7B, "M", "ύ"), - (0x1F7C, "V"), - (0x1F7D, "M", "ώ"), - (0x1F7E, "X"), - (0x1F80, "M", "ἀι"), - (0x1F81, "M", "ἁι"), - (0x1F82, "M", "ἂι"), - (0x1F83, "M", "ἃι"), - (0x1F84, "M", "ἄι"), - (0x1F85, "M", "ἅι"), - (0x1F86, "M", "ἆι"), - (0x1F87, "M", "ἇι"), - (0x1F88, "M", "ἀι"), - (0x1F89, "M", "ἁι"), - (0x1F8A, "M", "ἂι"), - (0x1F8B, "M", "ἃι"), - (0x1F8C, "M", "ἄι"), - (0x1F8D, "M", "ἅι"), - (0x1F8E, "M", "ἆι"), - (0x1F8F, "M", "ἇι"), - (0x1F90, "M", "ἠι"), - (0x1F91, "M", "ἡι"), - (0x1F92, "M", "ἢι"), - (0x1F93, "M", "ἣι"), - (0x1F94, "M", "ἤι"), - (0x1F95, "M", "ἥι"), - (0x1F96, "M", "ἦι"), - (0x1F97, "M", "ἧι"), - (0x1F98, "M", "ἠι"), - (0x1F99, "M", "ἡι"), - (0x1F9A, "M", "ἢι"), - (0x1F9B, "M", "ἣι"), - (0x1F9C, "M", "ἤι"), - (0x1F9D, "M", "ἥι"), - (0x1F9E, "M", "ἦι"), - (0x1F9F, "M", "ἧι"), - (0x1FA0, "M", "ὠι"), - (0x1FA1, "M", "ὡι"), - (0x1FA2, "M", "ὢι"), - (0x1FA3, "M", "ὣι"), - (0x1FA4, "M", "ὤι"), - (0x1FA5, "M", "ὥι"), - (0x1FA6, "M", "ὦι"), - (0x1FA7, "M", "ὧι"), - (0x1FA8, "M", "ὠι"), - (0x1FA9, "M", "ὡι"), - (0x1FAA, "M", "ὢι"), - (0x1FAB, "M", "ὣι"), - (0x1FAC, "M", "ὤι"), - (0x1FAD, "M", "ὥι"), - (0x1FAE, "M", "ὦι"), - (0x1FAF, "M", "ὧι"), - (0x1FB0, "V"), - (0x1FB2, "M", "ὰι"), - (0x1FB3, "M", "αι"), - (0x1FB4, "M", "άι"), - (0x1FB5, "X"), - (0x1FB6, "V"), - (0x1FB7, "M", "ᾶι"), - (0x1FB8, "M", "ᾰ"), - (0x1FB9, "M", "ᾱ"), - (0x1FBA, "M", "ὰ"), - (0x1FBB, "M", "ά"), - (0x1FBC, "M", "αι"), - (0x1FBD, "M", " ̓"), - (0x1FBE, "M", "ι"), - (0x1FBF, "M", " ̓"), - (0x1FC0, "M", " ͂"), - (0x1FC1, "M", " ̈͂"), - (0x1FC2, "M", "ὴι"), - (0x1FC3, "M", "ηι"), - (0x1FC4, "M", "ήι"), - (0x1FC5, "X"), - (0x1FC6, "V"), - (0x1FC7, "M", "ῆι"), - (0x1FC8, "M", "ὲ"), - (0x1FC9, "M", "έ"), - (0x1FCA, "M", "ὴ"), - ] - - -def _seg_21() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1FCB, "M", "ή"), - (0x1FCC, "M", "ηι"), - (0x1FCD, "M", " ̓̀"), - (0x1FCE, "M", " ̓́"), - (0x1FCF, "M", " ̓͂"), - (0x1FD0, "V"), - (0x1FD3, "M", "ΐ"), - (0x1FD4, "X"), - (0x1FD6, "V"), - (0x1FD8, "M", "ῐ"), - (0x1FD9, "M", "ῑ"), - (0x1FDA, "M", "ὶ"), - (0x1FDB, "M", "ί"), - (0x1FDC, "X"), - (0x1FDD, "M", " ̔̀"), - (0x1FDE, "M", " ̔́"), - (0x1FDF, "M", " ̔͂"), - (0x1FE0, "V"), - (0x1FE3, "M", "ΰ"), - (0x1FE4, "V"), - (0x1FE8, "M", "ῠ"), - (0x1FE9, "M", "ῡ"), - (0x1FEA, "M", "ὺ"), - (0x1FEB, "M", "ύ"), - (0x1FEC, "M", "ῥ"), - (0x1FED, "M", " ̈̀"), - (0x1FEE, "M", " ̈́"), - (0x1FEF, "M", "`"), - (0x1FF0, "X"), - (0x1FF2, "M", "ὼι"), - (0x1FF3, "M", "ωι"), - (0x1FF4, "M", "ώι"), - (0x1FF5, "X"), - (0x1FF6, "V"), - (0x1FF7, "M", "ῶι"), - (0x1FF8, "M", "ὸ"), - (0x1FF9, "M", "ό"), - (0x1FFA, "M", "ὼ"), - (0x1FFB, "M", "ώ"), - (0x1FFC, "M", "ωι"), - (0x1FFD, "M", " ́"), - (0x1FFE, "M", " ̔"), - (0x1FFF, "X"), - (0x2000, "M", " "), - (0x200B, "I"), - (0x200C, "D", ""), - (0x200E, "X"), - (0x2010, "V"), - (0x2011, "M", "‐"), - (0x2012, "V"), - (0x2017, "M", " ̳"), - (0x2018, "V"), - (0x2024, "X"), - (0x2027, "V"), - (0x2028, "X"), - (0x202F, "M", " "), - (0x2030, "V"), - (0x2033, "M", "′′"), - (0x2034, "M", "′′′"), - (0x2035, "V"), - (0x2036, "M", "‵‵"), - (0x2037, "M", "‵‵‵"), - (0x2038, "V"), - (0x203C, "M", "!!"), - (0x203D, "V"), - (0x203E, "M", " ̅"), - (0x203F, "V"), - (0x2047, "M", "??"), - (0x2048, "M", "?!"), - (0x2049, "M", "!?"), - (0x204A, "V"), - (0x2057, "M", "′′′′"), - (0x2058, "V"), - (0x205F, "M", " "), - (0x2060, "I"), - (0x2065, "X"), - (0x206A, "I"), - (0x2070, "M", "0"), - (0x2071, "M", "i"), - (0x2072, "X"), - (0x2074, "M", "4"), - (0x2075, "M", "5"), - (0x2076, "M", "6"), - (0x2077, "M", "7"), - (0x2078, "M", "8"), - (0x2079, "M", "9"), - (0x207A, "M", "+"), - (0x207B, "M", "−"), - (0x207C, "M", "="), - (0x207D, "M", "("), - (0x207E, "M", ")"), - (0x207F, "M", "n"), - (0x2080, "M", "0"), - (0x2081, "M", "1"), - (0x2082, "M", "2"), - (0x2083, "M", "3"), - (0x2084, "M", "4"), - (0x2085, "M", "5"), - (0x2086, "M", "6"), - (0x2087, "M", "7"), - ] - - -def _seg_22() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2088, "M", "8"), - (0x2089, "M", "9"), - (0x208A, "M", "+"), - (0x208B, "M", "−"), - (0x208C, "M", "="), - (0x208D, "M", "("), - (0x208E, "M", ")"), - (0x208F, "X"), - (0x2090, "M", "a"), - (0x2091, "M", "e"), - (0x2092, "M", "o"), - (0x2093, "M", "x"), - (0x2094, "M", "ə"), - (0x2095, "M", "h"), - (0x2096, "M", "k"), - (0x2097, "M", "l"), - (0x2098, "M", "m"), - (0x2099, "M", "n"), - (0x209A, "M", "p"), - (0x209B, "M", "s"), - (0x209C, "M", "t"), - (0x209D, "X"), - (0x20A0, "V"), - (0x20A8, "M", "rs"), - (0x20A9, "V"), - (0x20C1, "X"), - (0x20D0, "V"), - (0x20F1, "X"), - (0x2100, "M", "a/c"), - (0x2101, "M", "a/s"), - (0x2102, "M", "c"), - (0x2103, "M", "°c"), - (0x2104, "V"), - (0x2105, "M", "c/o"), - (0x2106, "M", "c/u"), - (0x2107, "M", "ɛ"), - (0x2108, "V"), - (0x2109, "M", "°f"), - (0x210A, "M", "g"), - (0x210B, "M", "h"), - (0x210F, "M", "ħ"), - (0x2110, "M", "i"), - (0x2112, "M", "l"), - (0x2114, "V"), - (0x2115, "M", "n"), - (0x2116, "M", "no"), - (0x2117, "V"), - (0x2119, "M", "p"), - (0x211A, "M", "q"), - (0x211B, "M", "r"), - (0x211E, "V"), - (0x2120, "M", "sm"), - (0x2121, "M", "tel"), - (0x2122, "M", "tm"), - (0x2123, "V"), - (0x2124, "M", "z"), - (0x2125, "V"), - (0x2126, "M", "ω"), - (0x2127, "V"), - (0x2128, "M", "z"), - (0x2129, "V"), - (0x212A, "M", "k"), - (0x212B, "M", "å"), - (0x212C, "M", "b"), - (0x212D, "M", "c"), - (0x212E, "V"), - (0x212F, "M", "e"), - (0x2131, "M", "f"), - (0x2132, "M", "ⅎ"), - (0x2133, "M", "m"), - (0x2134, "M", "o"), - (0x2135, "M", "א"), - (0x2136, "M", "ב"), - (0x2137, "M", "ג"), - (0x2138, "M", "ד"), - (0x2139, "M", "i"), - (0x213A, "V"), - (0x213B, "M", "fax"), - (0x213C, "M", "π"), - (0x213D, "M", "γ"), - (0x213F, "M", "π"), - (0x2140, "M", "∑"), - (0x2141, "V"), - (0x2145, "M", "d"), - (0x2147, "M", "e"), - (0x2148, "M", "i"), - (0x2149, "M", "j"), - (0x214A, "V"), - (0x2150, "M", "1⁄7"), - (0x2151, "M", "1⁄9"), - (0x2152, "M", "1⁄10"), - (0x2153, "M", "1⁄3"), - (0x2154, "M", "2⁄3"), - (0x2155, "M", "1⁄5"), - (0x2156, "M", "2⁄5"), - (0x2157, "M", "3⁄5"), - (0x2158, "M", "4⁄5"), - (0x2159, "M", "1⁄6"), - (0x215A, "M", "5⁄6"), - (0x215B, "M", "1⁄8"), - ] - - -def _seg_23() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x215C, "M", "3⁄8"), - (0x215D, "M", "5⁄8"), - (0x215E, "M", "7⁄8"), - (0x215F, "M", "1⁄"), - (0x2160, "M", "i"), - (0x2161, "M", "ii"), - (0x2162, "M", "iii"), - (0x2163, "M", "iv"), - (0x2164, "M", "v"), - (0x2165, "M", "vi"), - (0x2166, "M", "vii"), - (0x2167, "M", "viii"), - (0x2168, "M", "ix"), - (0x2169, "M", "x"), - (0x216A, "M", "xi"), - (0x216B, "M", "xii"), - (0x216C, "M", "l"), - (0x216D, "M", "c"), - (0x216E, "M", "d"), - (0x216F, "M", "m"), - (0x2170, "M", "i"), - (0x2171, "M", "ii"), - (0x2172, "M", "iii"), - (0x2173, "M", "iv"), - (0x2174, "M", "v"), - (0x2175, "M", "vi"), - (0x2176, "M", "vii"), - (0x2177, "M", "viii"), - (0x2178, "M", "ix"), - (0x2179, "M", "x"), - (0x217A, "M", "xi"), - (0x217B, "M", "xii"), - (0x217C, "M", "l"), - (0x217D, "M", "c"), - (0x217E, "M", "d"), - (0x217F, "M", "m"), - (0x2180, "V"), - (0x2183, "M", "ↄ"), - (0x2184, "V"), - (0x2189, "M", "0⁄3"), - (0x218A, "V"), - (0x218C, "X"), - (0x2190, "V"), - (0x222C, "M", "∫∫"), - (0x222D, "M", "∫∫∫"), - (0x222E, "V"), - (0x222F, "M", "∮∮"), - (0x2230, "M", "∮∮∮"), - (0x2231, "V"), - (0x2329, "M", "〈"), - (0x232A, "M", "〉"), - (0x232B, "V"), - (0x242A, "X"), - (0x2440, "V"), - (0x244B, "X"), - (0x2460, "M", "1"), - (0x2461, "M", "2"), - (0x2462, "M", "3"), - (0x2463, "M", "4"), - (0x2464, "M", "5"), - (0x2465, "M", "6"), - (0x2466, "M", "7"), - (0x2467, "M", "8"), - (0x2468, "M", "9"), - (0x2469, "M", "10"), - (0x246A, "M", "11"), - (0x246B, "M", "12"), - (0x246C, "M", "13"), - (0x246D, "M", "14"), - (0x246E, "M", "15"), - (0x246F, "M", "16"), - (0x2470, "M", "17"), - (0x2471, "M", "18"), - (0x2472, "M", "19"), - (0x2473, "M", "20"), - (0x2474, "M", "(1)"), - (0x2475, "M", "(2)"), - (0x2476, "M", "(3)"), - (0x2477, "M", "(4)"), - (0x2478, "M", "(5)"), - (0x2479, "M", "(6)"), - (0x247A, "M", "(7)"), - (0x247B, "M", "(8)"), - (0x247C, "M", "(9)"), - (0x247D, "M", "(10)"), - (0x247E, "M", "(11)"), - (0x247F, "M", "(12)"), - (0x2480, "M", "(13)"), - (0x2481, "M", "(14)"), - (0x2482, "M", "(15)"), - (0x2483, "M", "(16)"), - (0x2484, "M", "(17)"), - (0x2485, "M", "(18)"), - (0x2486, "M", "(19)"), - (0x2487, "M", "(20)"), - (0x2488, "X"), - (0x249C, "M", "(a)"), - (0x249D, "M", "(b)"), - (0x249E, "M", "(c)"), - (0x249F, "M", "(d)"), - ] - - -def _seg_24() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x24A0, "M", "(e)"), - (0x24A1, "M", "(f)"), - (0x24A2, "M", "(g)"), - (0x24A3, "M", "(h)"), - (0x24A4, "M", "(i)"), - (0x24A5, "M", "(j)"), - (0x24A6, "M", "(k)"), - (0x24A7, "M", "(l)"), - (0x24A8, "M", "(m)"), - (0x24A9, "M", "(n)"), - (0x24AA, "M", "(o)"), - (0x24AB, "M", "(p)"), - (0x24AC, "M", "(q)"), - (0x24AD, "M", "(r)"), - (0x24AE, "M", "(s)"), - (0x24AF, "M", "(t)"), - (0x24B0, "M", "(u)"), - (0x24B1, "M", "(v)"), - (0x24B2, "M", "(w)"), - (0x24B3, "M", "(x)"), - (0x24B4, "M", "(y)"), - (0x24B5, "M", "(z)"), - (0x24B6, "M", "a"), - (0x24B7, "M", "b"), - (0x24B8, "M", "c"), - (0x24B9, "M", "d"), - (0x24BA, "M", "e"), - (0x24BB, "M", "f"), - (0x24BC, "M", "g"), - (0x24BD, "M", "h"), - (0x24BE, "M", "i"), - (0x24BF, "M", "j"), - (0x24C0, "M", "k"), - (0x24C1, "M", "l"), - (0x24C2, "M", "m"), - (0x24C3, "M", "n"), - (0x24C4, "M", "o"), - (0x24C5, "M", "p"), - (0x24C6, "M", "q"), - (0x24C7, "M", "r"), - (0x24C8, "M", "s"), - (0x24C9, "M", "t"), - (0x24CA, "M", "u"), - (0x24CB, "M", "v"), - (0x24CC, "M", "w"), - (0x24CD, "M", "x"), - (0x24CE, "M", "y"), - (0x24CF, "M", "z"), - (0x24D0, "M", "a"), - (0x24D1, "M", "b"), - (0x24D2, "M", "c"), - (0x24D3, "M", "d"), - (0x24D4, "M", "e"), - (0x24D5, "M", "f"), - (0x24D6, "M", "g"), - (0x24D7, "M", "h"), - (0x24D8, "M", "i"), - (0x24D9, "M", "j"), - (0x24DA, "M", "k"), - (0x24DB, "M", "l"), - (0x24DC, "M", "m"), - (0x24DD, "M", "n"), - (0x24DE, "M", "o"), - (0x24DF, "M", "p"), - (0x24E0, "M", "q"), - (0x24E1, "M", "r"), - (0x24E2, "M", "s"), - (0x24E3, "M", "t"), - (0x24E4, "M", "u"), - (0x24E5, "M", "v"), - (0x24E6, "M", "w"), - (0x24E7, "M", "x"), - (0x24E8, "M", "y"), - (0x24E9, "M", "z"), - (0x24EA, "M", "0"), - (0x24EB, "V"), - (0x2A0C, "M", "∫∫∫∫"), - (0x2A0D, "V"), - (0x2A74, "M", "::="), - (0x2A75, "M", "=="), - (0x2A76, "M", "==="), - (0x2A77, "V"), - (0x2ADC, "M", "⫝̸"), - (0x2ADD, "V"), - (0x2B74, "X"), - (0x2B76, "V"), - (0x2B96, "X"), - (0x2B97, "V"), - (0x2C00, "M", "ⰰ"), - (0x2C01, "M", "ⰱ"), - (0x2C02, "M", "ⰲ"), - (0x2C03, "M", "ⰳ"), - (0x2C04, "M", "ⰴ"), - (0x2C05, "M", "ⰵ"), - (0x2C06, "M", "ⰶ"), - (0x2C07, "M", "ⰷ"), - (0x2C08, "M", "ⰸ"), - (0x2C09, "M", "ⰹ"), - (0x2C0A, "M", "ⰺ"), - (0x2C0B, "M", "ⰻ"), - ] - - -def _seg_25() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2C0C, "M", "ⰼ"), - (0x2C0D, "M", "ⰽ"), - (0x2C0E, "M", "ⰾ"), - (0x2C0F, "M", "ⰿ"), - (0x2C10, "M", "ⱀ"), - (0x2C11, "M", "ⱁ"), - (0x2C12, "M", "ⱂ"), - (0x2C13, "M", "ⱃ"), - (0x2C14, "M", "ⱄ"), - (0x2C15, "M", "ⱅ"), - (0x2C16, "M", "ⱆ"), - (0x2C17, "M", "ⱇ"), - (0x2C18, "M", "ⱈ"), - (0x2C19, "M", "ⱉ"), - (0x2C1A, "M", "ⱊ"), - (0x2C1B, "M", "ⱋ"), - (0x2C1C, "M", "ⱌ"), - (0x2C1D, "M", "ⱍ"), - (0x2C1E, "M", "ⱎ"), - (0x2C1F, "M", "ⱏ"), - (0x2C20, "M", "ⱐ"), - (0x2C21, "M", "ⱑ"), - (0x2C22, "M", "ⱒ"), - (0x2C23, "M", "ⱓ"), - (0x2C24, "M", "ⱔ"), - (0x2C25, "M", "ⱕ"), - (0x2C26, "M", "ⱖ"), - (0x2C27, "M", "ⱗ"), - (0x2C28, "M", "ⱘ"), - (0x2C29, "M", "ⱙ"), - (0x2C2A, "M", "ⱚ"), - (0x2C2B, "M", "ⱛ"), - (0x2C2C, "M", "ⱜ"), - (0x2C2D, "M", "ⱝ"), - (0x2C2E, "M", "ⱞ"), - (0x2C2F, "M", "ⱟ"), - (0x2C30, "V"), - (0x2C60, "M", "ⱡ"), - (0x2C61, "V"), - (0x2C62, "M", "ɫ"), - (0x2C63, "M", "ᵽ"), - (0x2C64, "M", "ɽ"), - (0x2C65, "V"), - (0x2C67, "M", "ⱨ"), - (0x2C68, "V"), - (0x2C69, "M", "ⱪ"), - (0x2C6A, "V"), - (0x2C6B, "M", "ⱬ"), - (0x2C6C, "V"), - (0x2C6D, "M", "ɑ"), - (0x2C6E, "M", "ɱ"), - (0x2C6F, "M", "ɐ"), - (0x2C70, "M", "ɒ"), - (0x2C71, "V"), - (0x2C72, "M", "ⱳ"), - (0x2C73, "V"), - (0x2C75, "M", "ⱶ"), - (0x2C76, "V"), - (0x2C7C, "M", "j"), - (0x2C7D, "M", "v"), - (0x2C7E, "M", "ȿ"), - (0x2C7F, "M", "ɀ"), - (0x2C80, "M", "ⲁ"), - (0x2C81, "V"), - (0x2C82, "M", "ⲃ"), - (0x2C83, "V"), - (0x2C84, "M", "ⲅ"), - (0x2C85, "V"), - (0x2C86, "M", "ⲇ"), - (0x2C87, "V"), - (0x2C88, "M", "ⲉ"), - (0x2C89, "V"), - (0x2C8A, "M", "ⲋ"), - (0x2C8B, "V"), - (0x2C8C, "M", "ⲍ"), - (0x2C8D, "V"), - (0x2C8E, "M", "ⲏ"), - (0x2C8F, "V"), - (0x2C90, "M", "ⲑ"), - (0x2C91, "V"), - (0x2C92, "M", "ⲓ"), - (0x2C93, "V"), - (0x2C94, "M", "ⲕ"), - (0x2C95, "V"), - (0x2C96, "M", "ⲗ"), - (0x2C97, "V"), - (0x2C98, "M", "ⲙ"), - (0x2C99, "V"), - (0x2C9A, "M", "ⲛ"), - (0x2C9B, "V"), - (0x2C9C, "M", "ⲝ"), - (0x2C9D, "V"), - (0x2C9E, "M", "ⲟ"), - (0x2C9F, "V"), - (0x2CA0, "M", "ⲡ"), - (0x2CA1, "V"), - (0x2CA2, "M", "ⲣ"), - (0x2CA3, "V"), - (0x2CA4, "M", "ⲥ"), - (0x2CA5, "V"), - ] - - -def _seg_26() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2CA6, "M", "ⲧ"), - (0x2CA7, "V"), - (0x2CA8, "M", "ⲩ"), - (0x2CA9, "V"), - (0x2CAA, "M", "ⲫ"), - (0x2CAB, "V"), - (0x2CAC, "M", "ⲭ"), - (0x2CAD, "V"), - (0x2CAE, "M", "ⲯ"), - (0x2CAF, "V"), - (0x2CB0, "M", "ⲱ"), - (0x2CB1, "V"), - (0x2CB2, "M", "ⲳ"), - (0x2CB3, "V"), - (0x2CB4, "M", "ⲵ"), - (0x2CB5, "V"), - (0x2CB6, "M", "ⲷ"), - (0x2CB7, "V"), - (0x2CB8, "M", "ⲹ"), - (0x2CB9, "V"), - (0x2CBA, "M", "ⲻ"), - (0x2CBB, "V"), - (0x2CBC, "M", "ⲽ"), - (0x2CBD, "V"), - (0x2CBE, "M", "ⲿ"), - (0x2CBF, "V"), - (0x2CC0, "M", "ⳁ"), - (0x2CC1, "V"), - (0x2CC2, "M", "ⳃ"), - (0x2CC3, "V"), - (0x2CC4, "M", "ⳅ"), - (0x2CC5, "V"), - (0x2CC6, "M", "ⳇ"), - (0x2CC7, "V"), - (0x2CC8, "M", "ⳉ"), - (0x2CC9, "V"), - (0x2CCA, "M", "ⳋ"), - (0x2CCB, "V"), - (0x2CCC, "M", "ⳍ"), - (0x2CCD, "V"), - (0x2CCE, "M", "ⳏ"), - (0x2CCF, "V"), - (0x2CD0, "M", "ⳑ"), - (0x2CD1, "V"), - (0x2CD2, "M", "ⳓ"), - (0x2CD3, "V"), - (0x2CD4, "M", "ⳕ"), - (0x2CD5, "V"), - (0x2CD6, "M", "ⳗ"), - (0x2CD7, "V"), - (0x2CD8, "M", "ⳙ"), - (0x2CD9, "V"), - (0x2CDA, "M", "ⳛ"), - (0x2CDB, "V"), - (0x2CDC, "M", "ⳝ"), - (0x2CDD, "V"), - (0x2CDE, "M", "ⳟ"), - (0x2CDF, "V"), - (0x2CE0, "M", "ⳡ"), - (0x2CE1, "V"), - (0x2CE2, "M", "ⳣ"), - (0x2CE3, "V"), - (0x2CEB, "M", "ⳬ"), - (0x2CEC, "V"), - (0x2CED, "M", "ⳮ"), - (0x2CEE, "V"), - (0x2CF2, "M", "ⳳ"), - (0x2CF3, "V"), - (0x2CF4, "X"), - (0x2CF9, "V"), - (0x2D26, "X"), - (0x2D27, "V"), - (0x2D28, "X"), - (0x2D2D, "V"), - (0x2D2E, "X"), - (0x2D30, "V"), - (0x2D68, "X"), - (0x2D6F, "M", "ⵡ"), - (0x2D70, "V"), - (0x2D71, "X"), - (0x2D7F, "V"), - (0x2D97, "X"), - (0x2DA0, "V"), - (0x2DA7, "X"), - (0x2DA8, "V"), - (0x2DAF, "X"), - (0x2DB0, "V"), - (0x2DB7, "X"), - (0x2DB8, "V"), - (0x2DBF, "X"), - (0x2DC0, "V"), - (0x2DC7, "X"), - (0x2DC8, "V"), - (0x2DCF, "X"), - (0x2DD0, "V"), - (0x2DD7, "X"), - (0x2DD8, "V"), - (0x2DDF, "X"), - (0x2DE0, "V"), - (0x2E5E, "X"), - ] - - -def _seg_27() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2E80, "V"), - (0x2E9A, "X"), - (0x2E9B, "V"), - (0x2E9F, "M", "母"), - (0x2EA0, "V"), - (0x2EF3, "M", "龟"), - (0x2EF4, "X"), - (0x2F00, "M", "一"), - (0x2F01, "M", "丨"), - (0x2F02, "M", "丶"), - (0x2F03, "M", "丿"), - (0x2F04, "M", "乙"), - (0x2F05, "M", "亅"), - (0x2F06, "M", "二"), - (0x2F07, "M", "亠"), - (0x2F08, "M", "人"), - (0x2F09, "M", "儿"), - (0x2F0A, "M", "入"), - (0x2F0B, "M", "八"), - (0x2F0C, "M", "冂"), - (0x2F0D, "M", "冖"), - (0x2F0E, "M", "冫"), - (0x2F0F, "M", "几"), - (0x2F10, "M", "凵"), - (0x2F11, "M", "刀"), - (0x2F12, "M", "力"), - (0x2F13, "M", "勹"), - (0x2F14, "M", "匕"), - (0x2F15, "M", "匚"), - (0x2F16, "M", "匸"), - (0x2F17, "M", "十"), - (0x2F18, "M", "卜"), - (0x2F19, "M", "卩"), - (0x2F1A, "M", "厂"), - (0x2F1B, "M", "厶"), - (0x2F1C, "M", "又"), - (0x2F1D, "M", "口"), - (0x2F1E, "M", "囗"), - (0x2F1F, "M", "土"), - (0x2F20, "M", "士"), - (0x2F21, "M", "夂"), - (0x2F22, "M", "夊"), - (0x2F23, "M", "夕"), - (0x2F24, "M", "大"), - (0x2F25, "M", "女"), - (0x2F26, "M", "子"), - (0x2F27, "M", "宀"), - (0x2F28, "M", "寸"), - (0x2F29, "M", "小"), - (0x2F2A, "M", "尢"), - (0x2F2B, "M", "尸"), - (0x2F2C, "M", "屮"), - (0x2F2D, "M", "山"), - (0x2F2E, "M", "巛"), - (0x2F2F, "M", "工"), - (0x2F30, "M", "己"), - (0x2F31, "M", "巾"), - (0x2F32, "M", "干"), - (0x2F33, "M", "幺"), - (0x2F34, "M", "广"), - (0x2F35, "M", "廴"), - (0x2F36, "M", "廾"), - (0x2F37, "M", "弋"), - (0x2F38, "M", "弓"), - (0x2F39, "M", "彐"), - (0x2F3A, "M", "彡"), - (0x2F3B, "M", "彳"), - (0x2F3C, "M", "心"), - (0x2F3D, "M", "戈"), - (0x2F3E, "M", "戶"), - (0x2F3F, "M", "手"), - (0x2F40, "M", "支"), - (0x2F41, "M", "攴"), - (0x2F42, "M", "文"), - (0x2F43, "M", "斗"), - (0x2F44, "M", "斤"), - (0x2F45, "M", "方"), - (0x2F46, "M", "无"), - (0x2F47, "M", "日"), - (0x2F48, "M", "曰"), - (0x2F49, "M", "月"), - (0x2F4A, "M", "木"), - (0x2F4B, "M", "欠"), - (0x2F4C, "M", "止"), - (0x2F4D, "M", "歹"), - (0x2F4E, "M", "殳"), - (0x2F4F, "M", "毋"), - (0x2F50, "M", "比"), - (0x2F51, "M", "毛"), - (0x2F52, "M", "氏"), - (0x2F53, "M", "气"), - (0x2F54, "M", "水"), - (0x2F55, "M", "火"), - (0x2F56, "M", "爪"), - (0x2F57, "M", "父"), - (0x2F58, "M", "爻"), - (0x2F59, "M", "爿"), - (0x2F5A, "M", "片"), - (0x2F5B, "M", "牙"), - (0x2F5C, "M", "牛"), - ] - - -def _seg_28() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F5D, "M", "犬"), - (0x2F5E, "M", "玄"), - (0x2F5F, "M", "玉"), - (0x2F60, "M", "瓜"), - (0x2F61, "M", "瓦"), - (0x2F62, "M", "甘"), - (0x2F63, "M", "生"), - (0x2F64, "M", "用"), - (0x2F65, "M", "田"), - (0x2F66, "M", "疋"), - (0x2F67, "M", "疒"), - (0x2F68, "M", "癶"), - (0x2F69, "M", "白"), - (0x2F6A, "M", "皮"), - (0x2F6B, "M", "皿"), - (0x2F6C, "M", "目"), - (0x2F6D, "M", "矛"), - (0x2F6E, "M", "矢"), - (0x2F6F, "M", "石"), - (0x2F70, "M", "示"), - (0x2F71, "M", "禸"), - (0x2F72, "M", "禾"), - (0x2F73, "M", "穴"), - (0x2F74, "M", "立"), - (0x2F75, "M", "竹"), - (0x2F76, "M", "米"), - (0x2F77, "M", "糸"), - (0x2F78, "M", "缶"), - (0x2F79, "M", "网"), - (0x2F7A, "M", "羊"), - (0x2F7B, "M", "羽"), - (0x2F7C, "M", "老"), - (0x2F7D, "M", "而"), - (0x2F7E, "M", "耒"), - (0x2F7F, "M", "耳"), - (0x2F80, "M", "聿"), - (0x2F81, "M", "肉"), - (0x2F82, "M", "臣"), - (0x2F83, "M", "自"), - (0x2F84, "M", "至"), - (0x2F85, "M", "臼"), - (0x2F86, "M", "舌"), - (0x2F87, "M", "舛"), - (0x2F88, "M", "舟"), - (0x2F89, "M", "艮"), - (0x2F8A, "M", "色"), - (0x2F8B, "M", "艸"), - (0x2F8C, "M", "虍"), - (0x2F8D, "M", "虫"), - (0x2F8E, "M", "血"), - (0x2F8F, "M", "行"), - (0x2F90, "M", "衣"), - (0x2F91, "M", "襾"), - (0x2F92, "M", "見"), - (0x2F93, "M", "角"), - (0x2F94, "M", "言"), - (0x2F95, "M", "谷"), - (0x2F96, "M", "豆"), - (0x2F97, "M", "豕"), - (0x2F98, "M", "豸"), - (0x2F99, "M", "貝"), - (0x2F9A, "M", "赤"), - (0x2F9B, "M", "走"), - (0x2F9C, "M", "足"), - (0x2F9D, "M", "身"), - (0x2F9E, "M", "車"), - (0x2F9F, "M", "辛"), - (0x2FA0, "M", "辰"), - (0x2FA1, "M", "辵"), - (0x2FA2, "M", "邑"), - (0x2FA3, "M", "酉"), - (0x2FA4, "M", "釆"), - (0x2FA5, "M", "里"), - (0x2FA6, "M", "金"), - (0x2FA7, "M", "長"), - (0x2FA8, "M", "門"), - (0x2FA9, "M", "阜"), - (0x2FAA, "M", "隶"), - (0x2FAB, "M", "隹"), - (0x2FAC, "M", "雨"), - (0x2FAD, "M", "靑"), - (0x2FAE, "M", "非"), - (0x2FAF, "M", "面"), - (0x2FB0, "M", "革"), - (0x2FB1, "M", "韋"), - (0x2FB2, "M", "韭"), - (0x2FB3, "M", "音"), - (0x2FB4, "M", "頁"), - (0x2FB5, "M", "風"), - (0x2FB6, "M", "飛"), - (0x2FB7, "M", "食"), - (0x2FB8, "M", "首"), - (0x2FB9, "M", "香"), - (0x2FBA, "M", "馬"), - (0x2FBB, "M", "骨"), - (0x2FBC, "M", "高"), - (0x2FBD, "M", "髟"), - (0x2FBE, "M", "鬥"), - (0x2FBF, "M", "鬯"), - (0x2FC0, "M", "鬲"), - ] - - -def _seg_29() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2FC1, "M", "鬼"), - (0x2FC2, "M", "魚"), - (0x2FC3, "M", "鳥"), - (0x2FC4, "M", "鹵"), - (0x2FC5, "M", "鹿"), - (0x2FC6, "M", "麥"), - (0x2FC7, "M", "麻"), - (0x2FC8, "M", "黃"), - (0x2FC9, "M", "黍"), - (0x2FCA, "M", "黑"), - (0x2FCB, "M", "黹"), - (0x2FCC, "M", "黽"), - (0x2FCD, "M", "鼎"), - (0x2FCE, "M", "鼓"), - (0x2FCF, "M", "鼠"), - (0x2FD0, "M", "鼻"), - (0x2FD1, "M", "齊"), - (0x2FD2, "M", "齒"), - (0x2FD3, "M", "龍"), - (0x2FD4, "M", "龜"), - (0x2FD5, "M", "龠"), - (0x2FD6, "X"), - (0x3000, "M", " "), - (0x3001, "V"), - (0x3002, "M", "."), - (0x3003, "V"), - (0x3036, "M", "〒"), - (0x3037, "V"), - (0x3038, "M", "十"), - (0x3039, "M", "卄"), - (0x303A, "M", "卅"), - (0x303B, "V"), - (0x3040, "X"), - (0x3041, "V"), - (0x3097, "X"), - (0x3099, "V"), - (0x309B, "M", " ゙"), - (0x309C, "M", " ゚"), - (0x309D, "V"), - (0x309F, "M", "より"), - (0x30A0, "V"), - (0x30FF, "M", "コト"), - (0x3100, "X"), - (0x3105, "V"), - (0x3130, "X"), - (0x3131, "M", "ᄀ"), - (0x3132, "M", "ᄁ"), - (0x3133, "M", "ᆪ"), - (0x3134, "M", "ᄂ"), - (0x3135, "M", "ᆬ"), - (0x3136, "M", "ᆭ"), - (0x3137, "M", "ᄃ"), - (0x3138, "M", "ᄄ"), - (0x3139, "M", "ᄅ"), - (0x313A, "M", "ᆰ"), - (0x313B, "M", "ᆱ"), - (0x313C, "M", "ᆲ"), - (0x313D, "M", "ᆳ"), - (0x313E, "M", "ᆴ"), - (0x313F, "M", "ᆵ"), - (0x3140, "M", "ᄚ"), - (0x3141, "M", "ᄆ"), - (0x3142, "M", "ᄇ"), - (0x3143, "M", "ᄈ"), - (0x3144, "M", "ᄡ"), - (0x3145, "M", "ᄉ"), - (0x3146, "M", "ᄊ"), - (0x3147, "M", "ᄋ"), - (0x3148, "M", "ᄌ"), - (0x3149, "M", "ᄍ"), - (0x314A, "M", "ᄎ"), - (0x314B, "M", "ᄏ"), - (0x314C, "M", "ᄐ"), - (0x314D, "M", "ᄑ"), - (0x314E, "M", "ᄒ"), - (0x314F, "M", "ᅡ"), - (0x3150, "M", "ᅢ"), - (0x3151, "M", "ᅣ"), - (0x3152, "M", "ᅤ"), - (0x3153, "M", "ᅥ"), - (0x3154, "M", "ᅦ"), - (0x3155, "M", "ᅧ"), - (0x3156, "M", "ᅨ"), - (0x3157, "M", "ᅩ"), - (0x3158, "M", "ᅪ"), - (0x3159, "M", "ᅫ"), - (0x315A, "M", "ᅬ"), - (0x315B, "M", "ᅭ"), - (0x315C, "M", "ᅮ"), - (0x315D, "M", "ᅯ"), - (0x315E, "M", "ᅰ"), - (0x315F, "M", "ᅱ"), - (0x3160, "M", "ᅲ"), - (0x3161, "M", "ᅳ"), - (0x3162, "M", "ᅴ"), - (0x3163, "M", "ᅵ"), - (0x3164, "I"), - (0x3165, "M", "ᄔ"), - (0x3166, "M", "ᄕ"), - (0x3167, "M", "ᇇ"), - ] - - -def _seg_30() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x3168, "M", "ᇈ"), - (0x3169, "M", "ᇌ"), - (0x316A, "M", "ᇎ"), - (0x316B, "M", "ᇓ"), - (0x316C, "M", "ᇗ"), - (0x316D, "M", "ᇙ"), - (0x316E, "M", "ᄜ"), - (0x316F, "M", "ᇝ"), - (0x3170, "M", "ᇟ"), - (0x3171, "M", "ᄝ"), - (0x3172, "M", "ᄞ"), - (0x3173, "M", "ᄠ"), - (0x3174, "M", "ᄢ"), - (0x3175, "M", "ᄣ"), - (0x3176, "M", "ᄧ"), - (0x3177, "M", "ᄩ"), - (0x3178, "M", "ᄫ"), - (0x3179, "M", "ᄬ"), - (0x317A, "M", "ᄭ"), - (0x317B, "M", "ᄮ"), - (0x317C, "M", "ᄯ"), - (0x317D, "M", "ᄲ"), - (0x317E, "M", "ᄶ"), - (0x317F, "M", "ᅀ"), - (0x3180, "M", "ᅇ"), - (0x3181, "M", "ᅌ"), - (0x3182, "M", "ᇱ"), - (0x3183, "M", "ᇲ"), - (0x3184, "M", "ᅗ"), - (0x3185, "M", "ᅘ"), - (0x3186, "M", "ᅙ"), - (0x3187, "M", "ᆄ"), - (0x3188, "M", "ᆅ"), - (0x3189, "M", "ᆈ"), - (0x318A, "M", "ᆑ"), - (0x318B, "M", "ᆒ"), - (0x318C, "M", "ᆔ"), - (0x318D, "M", "ᆞ"), - (0x318E, "M", "ᆡ"), - (0x318F, "X"), - (0x3190, "V"), - (0x3192, "M", "一"), - (0x3193, "M", "二"), - (0x3194, "M", "三"), - (0x3195, "M", "四"), - (0x3196, "M", "上"), - (0x3197, "M", "中"), - (0x3198, "M", "下"), - (0x3199, "M", "甲"), - (0x319A, "M", "乙"), - (0x319B, "M", "丙"), - (0x319C, "M", "丁"), - (0x319D, "M", "天"), - (0x319E, "M", "地"), - (0x319F, "M", "人"), - (0x31A0, "V"), - (0x31E6, "X"), - (0x31F0, "V"), - (0x3200, "M", "(ᄀ)"), - (0x3201, "M", "(ᄂ)"), - (0x3202, "M", "(ᄃ)"), - (0x3203, "M", "(ᄅ)"), - (0x3204, "M", "(ᄆ)"), - (0x3205, "M", "(ᄇ)"), - (0x3206, "M", "(ᄉ)"), - (0x3207, "M", "(ᄋ)"), - (0x3208, "M", "(ᄌ)"), - (0x3209, "M", "(ᄎ)"), - (0x320A, "M", "(ᄏ)"), - (0x320B, "M", "(ᄐ)"), - (0x320C, "M", "(ᄑ)"), - (0x320D, "M", "(ᄒ)"), - (0x320E, "M", "(가)"), - (0x320F, "M", "(나)"), - (0x3210, "M", "(다)"), - (0x3211, "M", "(라)"), - (0x3212, "M", "(마)"), - (0x3213, "M", "(바)"), - (0x3214, "M", "(사)"), - (0x3215, "M", "(아)"), - (0x3216, "M", "(자)"), - (0x3217, "M", "(차)"), - (0x3218, "M", "(카)"), - (0x3219, "M", "(타)"), - (0x321A, "M", "(파)"), - (0x321B, "M", "(하)"), - (0x321C, "M", "(주)"), - (0x321D, "M", "(오전)"), - (0x321E, "M", "(오후)"), - (0x321F, "X"), - (0x3220, "M", "(一)"), - (0x3221, "M", "(二)"), - (0x3222, "M", "(三)"), - (0x3223, "M", "(四)"), - (0x3224, "M", "(五)"), - (0x3225, "M", "(六)"), - (0x3226, "M", "(七)"), - (0x3227, "M", "(八)"), - (0x3228, "M", "(九)"), - (0x3229, "M", "(十)"), - ] - - -def _seg_31() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x322A, "M", "(月)"), - (0x322B, "M", "(火)"), - (0x322C, "M", "(水)"), - (0x322D, "M", "(木)"), - (0x322E, "M", "(金)"), - (0x322F, "M", "(土)"), - (0x3230, "M", "(日)"), - (0x3231, "M", "(株)"), - (0x3232, "M", "(有)"), - (0x3233, "M", "(社)"), - (0x3234, "M", "(名)"), - (0x3235, "M", "(特)"), - (0x3236, "M", "(財)"), - (0x3237, "M", "(祝)"), - (0x3238, "M", "(労)"), - (0x3239, "M", "(代)"), - (0x323A, "M", "(呼)"), - (0x323B, "M", "(学)"), - (0x323C, "M", "(監)"), - (0x323D, "M", "(企)"), - (0x323E, "M", "(資)"), - (0x323F, "M", "(協)"), - (0x3240, "M", "(祭)"), - (0x3241, "M", "(休)"), - (0x3242, "M", "(自)"), - (0x3243, "M", "(至)"), - (0x3244, "M", "問"), - (0x3245, "M", "幼"), - (0x3246, "M", "文"), - (0x3247, "M", "箏"), - (0x3248, "V"), - (0x3250, "M", "pte"), - (0x3251, "M", "21"), - (0x3252, "M", "22"), - (0x3253, "M", "23"), - (0x3254, "M", "24"), - (0x3255, "M", "25"), - (0x3256, "M", "26"), - (0x3257, "M", "27"), - (0x3258, "M", "28"), - (0x3259, "M", "29"), - (0x325A, "M", "30"), - (0x325B, "M", "31"), - (0x325C, "M", "32"), - (0x325D, "M", "33"), - (0x325E, "M", "34"), - (0x325F, "M", "35"), - (0x3260, "M", "ᄀ"), - (0x3261, "M", "ᄂ"), - (0x3262, "M", "ᄃ"), - (0x3263, "M", "ᄅ"), - (0x3264, "M", "ᄆ"), - (0x3265, "M", "ᄇ"), - (0x3266, "M", "ᄉ"), - (0x3267, "M", "ᄋ"), - (0x3268, "M", "ᄌ"), - (0x3269, "M", "ᄎ"), - (0x326A, "M", "ᄏ"), - (0x326B, "M", "ᄐ"), - (0x326C, "M", "ᄑ"), - (0x326D, "M", "ᄒ"), - (0x326E, "M", "가"), - (0x326F, "M", "나"), - (0x3270, "M", "다"), - (0x3271, "M", "라"), - (0x3272, "M", "마"), - (0x3273, "M", "바"), - (0x3274, "M", "사"), - (0x3275, "M", "아"), - (0x3276, "M", "자"), - (0x3277, "M", "차"), - (0x3278, "M", "카"), - (0x3279, "M", "타"), - (0x327A, "M", "파"), - (0x327B, "M", "하"), - (0x327C, "M", "참고"), - (0x327D, "M", "주의"), - (0x327E, "M", "우"), - (0x327F, "V"), - (0x3280, "M", "一"), - (0x3281, "M", "二"), - (0x3282, "M", "三"), - (0x3283, "M", "四"), - (0x3284, "M", "五"), - (0x3285, "M", "六"), - (0x3286, "M", "七"), - (0x3287, "M", "八"), - (0x3288, "M", "九"), - (0x3289, "M", "十"), - (0x328A, "M", "月"), - (0x328B, "M", "火"), - (0x328C, "M", "水"), - (0x328D, "M", "木"), - (0x328E, "M", "金"), - (0x328F, "M", "土"), - (0x3290, "M", "日"), - (0x3291, "M", "株"), - (0x3292, "M", "有"), - (0x3293, "M", "社"), - (0x3294, "M", "名"), - ] - - -def _seg_32() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x3295, "M", "特"), - (0x3296, "M", "財"), - (0x3297, "M", "祝"), - (0x3298, "M", "労"), - (0x3299, "M", "秘"), - (0x329A, "M", "男"), - (0x329B, "M", "女"), - (0x329C, "M", "適"), - (0x329D, "M", "優"), - (0x329E, "M", "印"), - (0x329F, "M", "注"), - (0x32A0, "M", "項"), - (0x32A1, "M", "休"), - (0x32A2, "M", "写"), - (0x32A3, "M", "正"), - (0x32A4, "M", "上"), - (0x32A5, "M", "中"), - (0x32A6, "M", "下"), - (0x32A7, "M", "左"), - (0x32A8, "M", "右"), - (0x32A9, "M", "医"), - (0x32AA, "M", "宗"), - (0x32AB, "M", "学"), - (0x32AC, "M", "監"), - (0x32AD, "M", "企"), - (0x32AE, "M", "資"), - (0x32AF, "M", "協"), - (0x32B0, "M", "夜"), - (0x32B1, "M", "36"), - (0x32B2, "M", "37"), - (0x32B3, "M", "38"), - (0x32B4, "M", "39"), - (0x32B5, "M", "40"), - (0x32B6, "M", "41"), - (0x32B7, "M", "42"), - (0x32B8, "M", "43"), - (0x32B9, "M", "44"), - (0x32BA, "M", "45"), - (0x32BB, "M", "46"), - (0x32BC, "M", "47"), - (0x32BD, "M", "48"), - (0x32BE, "M", "49"), - (0x32BF, "M", "50"), - (0x32C0, "M", "1月"), - (0x32C1, "M", "2月"), - (0x32C2, "M", "3月"), - (0x32C3, "M", "4月"), - (0x32C4, "M", "5月"), - (0x32C5, "M", "6月"), - (0x32C6, "M", "7月"), - (0x32C7, "M", "8月"), - (0x32C8, "M", "9月"), - (0x32C9, "M", "10月"), - (0x32CA, "M", "11月"), - (0x32CB, "M", "12月"), - (0x32CC, "M", "hg"), - (0x32CD, "M", "erg"), - (0x32CE, "M", "ev"), - (0x32CF, "M", "ltd"), - (0x32D0, "M", "ア"), - (0x32D1, "M", "イ"), - (0x32D2, "M", "ウ"), - (0x32D3, "M", "エ"), - (0x32D4, "M", "オ"), - (0x32D5, "M", "カ"), - (0x32D6, "M", "キ"), - (0x32D7, "M", "ク"), - (0x32D8, "M", "ケ"), - (0x32D9, "M", "コ"), - (0x32DA, "M", "サ"), - (0x32DB, "M", "シ"), - (0x32DC, "M", "ス"), - (0x32DD, "M", "セ"), - (0x32DE, "M", "ソ"), - (0x32DF, "M", "タ"), - (0x32E0, "M", "チ"), - (0x32E1, "M", "ツ"), - (0x32E2, "M", "テ"), - (0x32E3, "M", "ト"), - (0x32E4, "M", "ナ"), - (0x32E5, "M", "ニ"), - (0x32E6, "M", "ヌ"), - (0x32E7, "M", "ネ"), - (0x32E8, "M", "ノ"), - (0x32E9, "M", "ハ"), - (0x32EA, "M", "ヒ"), - (0x32EB, "M", "フ"), - (0x32EC, "M", "ヘ"), - (0x32ED, "M", "ホ"), - (0x32EE, "M", "マ"), - (0x32EF, "M", "ミ"), - (0x32F0, "M", "ム"), - (0x32F1, "M", "メ"), - (0x32F2, "M", "モ"), - (0x32F3, "M", "ヤ"), - (0x32F4, "M", "ユ"), - (0x32F5, "M", "ヨ"), - (0x32F6, "M", "ラ"), - (0x32F7, "M", "リ"), - (0x32F8, "M", "ル"), - ] - - -def _seg_33() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x32F9, "M", "レ"), - (0x32FA, "M", "ロ"), - (0x32FB, "M", "ワ"), - (0x32FC, "M", "ヰ"), - (0x32FD, "M", "ヱ"), - (0x32FE, "M", "ヲ"), - (0x32FF, "M", "令和"), - (0x3300, "M", "アパート"), - (0x3301, "M", "アルファ"), - (0x3302, "M", "アンペア"), - (0x3303, "M", "アール"), - (0x3304, "M", "イニング"), - (0x3305, "M", "インチ"), - (0x3306, "M", "ウォン"), - (0x3307, "M", "エスクード"), - (0x3308, "M", "エーカー"), - (0x3309, "M", "オンス"), - (0x330A, "M", "オーム"), - (0x330B, "M", "カイリ"), - (0x330C, "M", "カラット"), - (0x330D, "M", "カロリー"), - (0x330E, "M", "ガロン"), - (0x330F, "M", "ガンマ"), - (0x3310, "M", "ギガ"), - (0x3311, "M", "ギニー"), - (0x3312, "M", "キュリー"), - (0x3313, "M", "ギルダー"), - (0x3314, "M", "キロ"), - (0x3315, "M", "キログラム"), - (0x3316, "M", "キロメートル"), - (0x3317, "M", "キロワット"), - (0x3318, "M", "グラム"), - (0x3319, "M", "グラムトン"), - (0x331A, "M", "クルゼイロ"), - (0x331B, "M", "クローネ"), - (0x331C, "M", "ケース"), - (0x331D, "M", "コルナ"), - (0x331E, "M", "コーポ"), - (0x331F, "M", "サイクル"), - (0x3320, "M", "サンチーム"), - (0x3321, "M", "シリング"), - (0x3322, "M", "センチ"), - (0x3323, "M", "セント"), - (0x3324, "M", "ダース"), - (0x3325, "M", "デシ"), - (0x3326, "M", "ドル"), - (0x3327, "M", "トン"), - (0x3328, "M", "ナノ"), - (0x3329, "M", "ノット"), - (0x332A, "M", "ハイツ"), - (0x332B, "M", "パーセント"), - (0x332C, "M", "パーツ"), - (0x332D, "M", "バーレル"), - (0x332E, "M", "ピアストル"), - (0x332F, "M", "ピクル"), - (0x3330, "M", "ピコ"), - (0x3331, "M", "ビル"), - (0x3332, "M", "ファラッド"), - (0x3333, "M", "フィート"), - (0x3334, "M", "ブッシェル"), - (0x3335, "M", "フラン"), - (0x3336, "M", "ヘクタール"), - (0x3337, "M", "ペソ"), - (0x3338, "M", "ペニヒ"), - (0x3339, "M", "ヘルツ"), - (0x333A, "M", "ペンス"), - (0x333B, "M", "ページ"), - (0x333C, "M", "ベータ"), - (0x333D, "M", "ポイント"), - (0x333E, "M", "ボルト"), - (0x333F, "M", "ホン"), - (0x3340, "M", "ポンド"), - (0x3341, "M", "ホール"), - (0x3342, "M", "ホーン"), - (0x3343, "M", "マイクロ"), - (0x3344, "M", "マイル"), - (0x3345, "M", "マッハ"), - (0x3346, "M", "マルク"), - (0x3347, "M", "マンション"), - (0x3348, "M", "ミクロン"), - (0x3349, "M", "ミリ"), - (0x334A, "M", "ミリバール"), - (0x334B, "M", "メガ"), - (0x334C, "M", "メガトン"), - (0x334D, "M", "メートル"), - (0x334E, "M", "ヤード"), - (0x334F, "M", "ヤール"), - (0x3350, "M", "ユアン"), - (0x3351, "M", "リットル"), - (0x3352, "M", "リラ"), - (0x3353, "M", "ルピー"), - (0x3354, "M", "ルーブル"), - (0x3355, "M", "レム"), - (0x3356, "M", "レントゲン"), - (0x3357, "M", "ワット"), - (0x3358, "M", "0点"), - (0x3359, "M", "1点"), - (0x335A, "M", "2点"), - (0x335B, "M", "3点"), - (0x335C, "M", "4点"), - ] - - -def _seg_34() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x335D, "M", "5点"), - (0x335E, "M", "6点"), - (0x335F, "M", "7点"), - (0x3360, "M", "8点"), - (0x3361, "M", "9点"), - (0x3362, "M", "10点"), - (0x3363, "M", "11点"), - (0x3364, "M", "12点"), - (0x3365, "M", "13点"), - (0x3366, "M", "14点"), - (0x3367, "M", "15点"), - (0x3368, "M", "16点"), - (0x3369, "M", "17点"), - (0x336A, "M", "18点"), - (0x336B, "M", "19点"), - (0x336C, "M", "20点"), - (0x336D, "M", "21点"), - (0x336E, "M", "22点"), - (0x336F, "M", "23点"), - (0x3370, "M", "24点"), - (0x3371, "M", "hpa"), - (0x3372, "M", "da"), - (0x3373, "M", "au"), - (0x3374, "M", "bar"), - (0x3375, "M", "ov"), - (0x3376, "M", "pc"), - (0x3377, "M", "dm"), - (0x3378, "M", "dm2"), - (0x3379, "M", "dm3"), - (0x337A, "M", "iu"), - (0x337B, "M", "平成"), - (0x337C, "M", "昭和"), - (0x337D, "M", "大正"), - (0x337E, "M", "明治"), - (0x337F, "M", "株式会社"), - (0x3380, "M", "pa"), - (0x3381, "M", "na"), - (0x3382, "M", "μa"), - (0x3383, "M", "ma"), - (0x3384, "M", "ka"), - (0x3385, "M", "kb"), - (0x3386, "M", "mb"), - (0x3387, "M", "gb"), - (0x3388, "M", "cal"), - (0x3389, "M", "kcal"), - (0x338A, "M", "pf"), - (0x338B, "M", "nf"), - (0x338C, "M", "μf"), - (0x338D, "M", "μg"), - (0x338E, "M", "mg"), - (0x338F, "M", "kg"), - (0x3390, "M", "hz"), - (0x3391, "M", "khz"), - (0x3392, "M", "mhz"), - (0x3393, "M", "ghz"), - (0x3394, "M", "thz"), - (0x3395, "M", "μl"), - (0x3396, "M", "ml"), - (0x3397, "M", "dl"), - (0x3398, "M", "kl"), - (0x3399, "M", "fm"), - (0x339A, "M", "nm"), - (0x339B, "M", "μm"), - (0x339C, "M", "mm"), - (0x339D, "M", "cm"), - (0x339E, "M", "km"), - (0x339F, "M", "mm2"), - (0x33A0, "M", "cm2"), - (0x33A1, "M", "m2"), - (0x33A2, "M", "km2"), - (0x33A3, "M", "mm3"), - (0x33A4, "M", "cm3"), - (0x33A5, "M", "m3"), - (0x33A6, "M", "km3"), - (0x33A7, "M", "m∕s"), - (0x33A8, "M", "m∕s2"), - (0x33A9, "M", "pa"), - (0x33AA, "M", "kpa"), - (0x33AB, "M", "mpa"), - (0x33AC, "M", "gpa"), - (0x33AD, "M", "rad"), - (0x33AE, "M", "rad∕s"), - (0x33AF, "M", "rad∕s2"), - (0x33B0, "M", "ps"), - (0x33B1, "M", "ns"), - (0x33B2, "M", "μs"), - (0x33B3, "M", "ms"), - (0x33B4, "M", "pv"), - (0x33B5, "M", "nv"), - (0x33B6, "M", "μv"), - (0x33B7, "M", "mv"), - (0x33B8, "M", "kv"), - (0x33B9, "M", "mv"), - (0x33BA, "M", "pw"), - (0x33BB, "M", "nw"), - (0x33BC, "M", "μw"), - (0x33BD, "M", "mw"), - (0x33BE, "M", "kw"), - (0x33BF, "M", "mw"), - (0x33C0, "M", "kω"), - ] - - -def _seg_35() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x33C1, "M", "mω"), - (0x33C2, "X"), - (0x33C3, "M", "bq"), - (0x33C4, "M", "cc"), - (0x33C5, "M", "cd"), - (0x33C6, "M", "c∕kg"), - (0x33C7, "X"), - (0x33C8, "M", "db"), - (0x33C9, "M", "gy"), - (0x33CA, "M", "ha"), - (0x33CB, "M", "hp"), - (0x33CC, "M", "in"), - (0x33CD, "M", "kk"), - (0x33CE, "M", "km"), - (0x33CF, "M", "kt"), - (0x33D0, "M", "lm"), - (0x33D1, "M", "ln"), - (0x33D2, "M", "log"), - (0x33D3, "M", "lx"), - (0x33D4, "M", "mb"), - (0x33D5, "M", "mil"), - (0x33D6, "M", "mol"), - (0x33D7, "M", "ph"), - (0x33D8, "X"), - (0x33D9, "M", "ppm"), - (0x33DA, "M", "pr"), - (0x33DB, "M", "sr"), - (0x33DC, "M", "sv"), - (0x33DD, "M", "wb"), - (0x33DE, "M", "v∕m"), - (0x33DF, "M", "a∕m"), - (0x33E0, "M", "1日"), - (0x33E1, "M", "2日"), - (0x33E2, "M", "3日"), - (0x33E3, "M", "4日"), - (0x33E4, "M", "5日"), - (0x33E5, "M", "6日"), - (0x33E6, "M", "7日"), - (0x33E7, "M", "8日"), - (0x33E8, "M", "9日"), - (0x33E9, "M", "10日"), - (0x33EA, "M", "11日"), - (0x33EB, "M", "12日"), - (0x33EC, "M", "13日"), - (0x33ED, "M", "14日"), - (0x33EE, "M", "15日"), - (0x33EF, "M", "16日"), - (0x33F0, "M", "17日"), - (0x33F1, "M", "18日"), - (0x33F2, "M", "19日"), - (0x33F3, "M", "20日"), - (0x33F4, "M", "21日"), - (0x33F5, "M", "22日"), - (0x33F6, "M", "23日"), - (0x33F7, "M", "24日"), - (0x33F8, "M", "25日"), - (0x33F9, "M", "26日"), - (0x33FA, "M", "27日"), - (0x33FB, "M", "28日"), - (0x33FC, "M", "29日"), - (0x33FD, "M", "30日"), - (0x33FE, "M", "31日"), - (0x33FF, "M", "gal"), - (0x3400, "V"), - (0xA48D, "X"), - (0xA490, "V"), - (0xA4C7, "X"), - (0xA4D0, "V"), - (0xA62C, "X"), - (0xA640, "M", "ꙁ"), - (0xA641, "V"), - (0xA642, "M", "ꙃ"), - (0xA643, "V"), - (0xA644, "M", "ꙅ"), - (0xA645, "V"), - (0xA646, "M", "ꙇ"), - (0xA647, "V"), - (0xA648, "M", "ꙉ"), - (0xA649, "V"), - (0xA64A, "M", "ꙋ"), - (0xA64B, "V"), - (0xA64C, "M", "ꙍ"), - (0xA64D, "V"), - (0xA64E, "M", "ꙏ"), - (0xA64F, "V"), - (0xA650, "M", "ꙑ"), - (0xA651, "V"), - (0xA652, "M", "ꙓ"), - (0xA653, "V"), - (0xA654, "M", "ꙕ"), - (0xA655, "V"), - (0xA656, "M", "ꙗ"), - (0xA657, "V"), - (0xA658, "M", "ꙙ"), - (0xA659, "V"), - (0xA65A, "M", "ꙛ"), - (0xA65B, "V"), - (0xA65C, "M", "ꙝ"), - (0xA65D, "V"), - (0xA65E, "M", "ꙟ"), - ] - - -def _seg_36() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xA65F, "V"), - (0xA660, "M", "ꙡ"), - (0xA661, "V"), - (0xA662, "M", "ꙣ"), - (0xA663, "V"), - (0xA664, "M", "ꙥ"), - (0xA665, "V"), - (0xA666, "M", "ꙧ"), - (0xA667, "V"), - (0xA668, "M", "ꙩ"), - (0xA669, "V"), - (0xA66A, "M", "ꙫ"), - (0xA66B, "V"), - (0xA66C, "M", "ꙭ"), - (0xA66D, "V"), - (0xA680, "M", "ꚁ"), - (0xA681, "V"), - (0xA682, "M", "ꚃ"), - (0xA683, "V"), - (0xA684, "M", "ꚅ"), - (0xA685, "V"), - (0xA686, "M", "ꚇ"), - (0xA687, "V"), - (0xA688, "M", "ꚉ"), - (0xA689, "V"), - (0xA68A, "M", "ꚋ"), - (0xA68B, "V"), - (0xA68C, "M", "ꚍ"), - (0xA68D, "V"), - (0xA68E, "M", "ꚏ"), - (0xA68F, "V"), - (0xA690, "M", "ꚑ"), - (0xA691, "V"), - (0xA692, "M", "ꚓ"), - (0xA693, "V"), - (0xA694, "M", "ꚕ"), - (0xA695, "V"), - (0xA696, "M", "ꚗ"), - (0xA697, "V"), - (0xA698, "M", "ꚙ"), - (0xA699, "V"), - (0xA69A, "M", "ꚛ"), - (0xA69B, "V"), - (0xA69C, "M", "ъ"), - (0xA69D, "M", "ь"), - (0xA69E, "V"), - (0xA6F8, "X"), - (0xA700, "V"), - (0xA722, "M", "ꜣ"), - (0xA723, "V"), - (0xA724, "M", "ꜥ"), - (0xA725, "V"), - (0xA726, "M", "ꜧ"), - (0xA727, "V"), - (0xA728, "M", "ꜩ"), - (0xA729, "V"), - (0xA72A, "M", "ꜫ"), - (0xA72B, "V"), - (0xA72C, "M", "ꜭ"), - (0xA72D, "V"), - (0xA72E, "M", "ꜯ"), - (0xA72F, "V"), - (0xA732, "M", "ꜳ"), - (0xA733, "V"), - (0xA734, "M", "ꜵ"), - (0xA735, "V"), - (0xA736, "M", "ꜷ"), - (0xA737, "V"), - (0xA738, "M", "ꜹ"), - (0xA739, "V"), - (0xA73A, "M", "ꜻ"), - (0xA73B, "V"), - (0xA73C, "M", "ꜽ"), - (0xA73D, "V"), - (0xA73E, "M", "ꜿ"), - (0xA73F, "V"), - (0xA740, "M", "ꝁ"), - (0xA741, "V"), - (0xA742, "M", "ꝃ"), - (0xA743, "V"), - (0xA744, "M", "ꝅ"), - (0xA745, "V"), - (0xA746, "M", "ꝇ"), - (0xA747, "V"), - (0xA748, "M", "ꝉ"), - (0xA749, "V"), - (0xA74A, "M", "ꝋ"), - (0xA74B, "V"), - (0xA74C, "M", "ꝍ"), - (0xA74D, "V"), - (0xA74E, "M", "ꝏ"), - (0xA74F, "V"), - (0xA750, "M", "ꝑ"), - (0xA751, "V"), - (0xA752, "M", "ꝓ"), - (0xA753, "V"), - (0xA754, "M", "ꝕ"), - (0xA755, "V"), - (0xA756, "M", "ꝗ"), - (0xA757, "V"), - ] - - -def _seg_37() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xA758, "M", "ꝙ"), - (0xA759, "V"), - (0xA75A, "M", "ꝛ"), - (0xA75B, "V"), - (0xA75C, "M", "ꝝ"), - (0xA75D, "V"), - (0xA75E, "M", "ꝟ"), - (0xA75F, "V"), - (0xA760, "M", "ꝡ"), - (0xA761, "V"), - (0xA762, "M", "ꝣ"), - (0xA763, "V"), - (0xA764, "M", "ꝥ"), - (0xA765, "V"), - (0xA766, "M", "ꝧ"), - (0xA767, "V"), - (0xA768, "M", "ꝩ"), - (0xA769, "V"), - (0xA76A, "M", "ꝫ"), - (0xA76B, "V"), - (0xA76C, "M", "ꝭ"), - (0xA76D, "V"), - (0xA76E, "M", "ꝯ"), - (0xA76F, "V"), - (0xA770, "M", "ꝯ"), - (0xA771, "V"), - (0xA779, "M", "ꝺ"), - (0xA77A, "V"), - (0xA77B, "M", "ꝼ"), - (0xA77C, "V"), - (0xA77D, "M", "ᵹ"), - (0xA77E, "M", "ꝿ"), - (0xA77F, "V"), - (0xA780, "M", "ꞁ"), - (0xA781, "V"), - (0xA782, "M", "ꞃ"), - (0xA783, "V"), - (0xA784, "M", "ꞅ"), - (0xA785, "V"), - (0xA786, "M", "ꞇ"), - (0xA787, "V"), - (0xA78B, "M", "ꞌ"), - (0xA78C, "V"), - (0xA78D, "M", "ɥ"), - (0xA78E, "V"), - (0xA790, "M", "ꞑ"), - (0xA791, "V"), - (0xA792, "M", "ꞓ"), - (0xA793, "V"), - (0xA796, "M", "ꞗ"), - (0xA797, "V"), - (0xA798, "M", "ꞙ"), - (0xA799, "V"), - (0xA79A, "M", "ꞛ"), - (0xA79B, "V"), - (0xA79C, "M", "ꞝ"), - (0xA79D, "V"), - (0xA79E, "M", "ꞟ"), - (0xA79F, "V"), - (0xA7A0, "M", "ꞡ"), - (0xA7A1, "V"), - (0xA7A2, "M", "ꞣ"), - (0xA7A3, "V"), - (0xA7A4, "M", "ꞥ"), - (0xA7A5, "V"), - (0xA7A6, "M", "ꞧ"), - (0xA7A7, "V"), - (0xA7A8, "M", "ꞩ"), - (0xA7A9, "V"), - (0xA7AA, "M", "ɦ"), - (0xA7AB, "M", "ɜ"), - (0xA7AC, "M", "ɡ"), - (0xA7AD, "M", "ɬ"), - (0xA7AE, "M", "ɪ"), - (0xA7AF, "V"), - (0xA7B0, "M", "ʞ"), - (0xA7B1, "M", "ʇ"), - (0xA7B2, "M", "ʝ"), - (0xA7B3, "M", "ꭓ"), - (0xA7B4, "M", "ꞵ"), - (0xA7B5, "V"), - (0xA7B6, "M", "ꞷ"), - (0xA7B7, "V"), - (0xA7B8, "M", "ꞹ"), - (0xA7B9, "V"), - (0xA7BA, "M", "ꞻ"), - (0xA7BB, "V"), - (0xA7BC, "M", "ꞽ"), - (0xA7BD, "V"), - (0xA7BE, "M", "ꞿ"), - (0xA7BF, "V"), - (0xA7C0, "M", "ꟁ"), - (0xA7C1, "V"), - (0xA7C2, "M", "ꟃ"), - (0xA7C3, "V"), - (0xA7C4, "M", "ꞔ"), - (0xA7C5, "M", "ʂ"), - (0xA7C6, "M", "ᶎ"), - (0xA7C7, "M", "ꟈ"), - (0xA7C8, "V"), - ] - - -def _seg_38() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xA7C9, "M", "ꟊ"), - (0xA7CA, "V"), - (0xA7CB, "M", "ɤ"), - (0xA7CC, "M", "ꟍ"), - (0xA7CD, "V"), - (0xA7CE, "X"), - (0xA7D0, "M", "ꟑ"), - (0xA7D1, "V"), - (0xA7D2, "X"), - (0xA7D3, "V"), - (0xA7D4, "X"), - (0xA7D5, "V"), - (0xA7D6, "M", "ꟗ"), - (0xA7D7, "V"), - (0xA7D8, "M", "ꟙ"), - (0xA7D9, "V"), - (0xA7DA, "M", "ꟛ"), - (0xA7DB, "V"), - (0xA7DC, "M", "ƛ"), - (0xA7DD, "X"), - (0xA7F2, "M", "c"), - (0xA7F3, "M", "f"), - (0xA7F4, "M", "q"), - (0xA7F5, "M", "ꟶ"), - (0xA7F6, "V"), - (0xA7F8, "M", "ħ"), - (0xA7F9, "M", "œ"), - (0xA7FA, "V"), - (0xA82D, "X"), - (0xA830, "V"), - (0xA83A, "X"), - (0xA840, "V"), - (0xA878, "X"), - (0xA880, "V"), - (0xA8C6, "X"), - (0xA8CE, "V"), - (0xA8DA, "X"), - (0xA8E0, "V"), - (0xA954, "X"), - (0xA95F, "V"), - (0xA97D, "X"), - (0xA980, "V"), - (0xA9CE, "X"), - (0xA9CF, "V"), - (0xA9DA, "X"), - (0xA9DE, "V"), - (0xA9FF, "X"), - (0xAA00, "V"), - (0xAA37, "X"), - (0xAA40, "V"), - (0xAA4E, "X"), - (0xAA50, "V"), - (0xAA5A, "X"), - (0xAA5C, "V"), - (0xAAC3, "X"), - (0xAADB, "V"), - (0xAAF7, "X"), - (0xAB01, "V"), - (0xAB07, "X"), - (0xAB09, "V"), - (0xAB0F, "X"), - (0xAB11, "V"), - (0xAB17, "X"), - (0xAB20, "V"), - (0xAB27, "X"), - (0xAB28, "V"), - (0xAB2F, "X"), - (0xAB30, "V"), - (0xAB5C, "M", "ꜧ"), - (0xAB5D, "M", "ꬷ"), - (0xAB5E, "M", "ɫ"), - (0xAB5F, "M", "ꭒ"), - (0xAB60, "V"), - (0xAB69, "M", "ʍ"), - (0xAB6A, "V"), - (0xAB6C, "X"), - (0xAB70, "M", "Ꭰ"), - (0xAB71, "M", "Ꭱ"), - (0xAB72, "M", "Ꭲ"), - (0xAB73, "M", "Ꭳ"), - (0xAB74, "M", "Ꭴ"), - (0xAB75, "M", "Ꭵ"), - (0xAB76, "M", "Ꭶ"), - (0xAB77, "M", "Ꭷ"), - (0xAB78, "M", "Ꭸ"), - (0xAB79, "M", "Ꭹ"), - (0xAB7A, "M", "Ꭺ"), - (0xAB7B, "M", "Ꭻ"), - (0xAB7C, "M", "Ꭼ"), - (0xAB7D, "M", "Ꭽ"), - (0xAB7E, "M", "Ꭾ"), - (0xAB7F, "M", "Ꭿ"), - (0xAB80, "M", "Ꮀ"), - (0xAB81, "M", "Ꮁ"), - (0xAB82, "M", "Ꮂ"), - (0xAB83, "M", "Ꮃ"), - (0xAB84, "M", "Ꮄ"), - (0xAB85, "M", "Ꮅ"), - (0xAB86, "M", "Ꮆ"), - (0xAB87, "M", "Ꮇ"), - ] - - -def _seg_39() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xAB88, "M", "Ꮈ"), - (0xAB89, "M", "Ꮉ"), - (0xAB8A, "M", "Ꮊ"), - (0xAB8B, "M", "Ꮋ"), - (0xAB8C, "M", "Ꮌ"), - (0xAB8D, "M", "Ꮍ"), - (0xAB8E, "M", "Ꮎ"), - (0xAB8F, "M", "Ꮏ"), - (0xAB90, "M", "Ꮐ"), - (0xAB91, "M", "Ꮑ"), - (0xAB92, "M", "Ꮒ"), - (0xAB93, "M", "Ꮓ"), - (0xAB94, "M", "Ꮔ"), - (0xAB95, "M", "Ꮕ"), - (0xAB96, "M", "Ꮖ"), - (0xAB97, "M", "Ꮗ"), - (0xAB98, "M", "Ꮘ"), - (0xAB99, "M", "Ꮙ"), - (0xAB9A, "M", "Ꮚ"), - (0xAB9B, "M", "Ꮛ"), - (0xAB9C, "M", "Ꮜ"), - (0xAB9D, "M", "Ꮝ"), - (0xAB9E, "M", "Ꮞ"), - (0xAB9F, "M", "Ꮟ"), - (0xABA0, "M", "Ꮠ"), - (0xABA1, "M", "Ꮡ"), - (0xABA2, "M", "Ꮢ"), - (0xABA3, "M", "Ꮣ"), - (0xABA4, "M", "Ꮤ"), - (0xABA5, "M", "Ꮥ"), - (0xABA6, "M", "Ꮦ"), - (0xABA7, "M", "Ꮧ"), - (0xABA8, "M", "Ꮨ"), - (0xABA9, "M", "Ꮩ"), - (0xABAA, "M", "Ꮪ"), - (0xABAB, "M", "Ꮫ"), - (0xABAC, "M", "Ꮬ"), - (0xABAD, "M", "Ꮭ"), - (0xABAE, "M", "Ꮮ"), - (0xABAF, "M", "Ꮯ"), - (0xABB0, "M", "Ꮰ"), - (0xABB1, "M", "Ꮱ"), - (0xABB2, "M", "Ꮲ"), - (0xABB3, "M", "Ꮳ"), - (0xABB4, "M", "Ꮴ"), - (0xABB5, "M", "Ꮵ"), - (0xABB6, "M", "Ꮶ"), - (0xABB7, "M", "Ꮷ"), - (0xABB8, "M", "Ꮸ"), - (0xABB9, "M", "Ꮹ"), - (0xABBA, "M", "Ꮺ"), - (0xABBB, "M", "Ꮻ"), - (0xABBC, "M", "Ꮼ"), - (0xABBD, "M", "Ꮽ"), - (0xABBE, "M", "Ꮾ"), - (0xABBF, "M", "Ꮿ"), - (0xABC0, "V"), - (0xABEE, "X"), - (0xABF0, "V"), - (0xABFA, "X"), - (0xAC00, "V"), - (0xD7A4, "X"), - (0xD7B0, "V"), - (0xD7C7, "X"), - (0xD7CB, "V"), - (0xD7FC, "X"), - (0xF900, "M", "豈"), - (0xF901, "M", "更"), - (0xF902, "M", "車"), - (0xF903, "M", "賈"), - (0xF904, "M", "滑"), - (0xF905, "M", "串"), - (0xF906, "M", "句"), - (0xF907, "M", "龜"), - (0xF909, "M", "契"), - (0xF90A, "M", "金"), - (0xF90B, "M", "喇"), - (0xF90C, "M", "奈"), - (0xF90D, "M", "懶"), - (0xF90E, "M", "癩"), - (0xF90F, "M", "羅"), - (0xF910, "M", "蘿"), - (0xF911, "M", "螺"), - (0xF912, "M", "裸"), - (0xF913, "M", "邏"), - (0xF914, "M", "樂"), - (0xF915, "M", "洛"), - (0xF916, "M", "烙"), - (0xF917, "M", "珞"), - (0xF918, "M", "落"), - (0xF919, "M", "酪"), - (0xF91A, "M", "駱"), - (0xF91B, "M", "亂"), - (0xF91C, "M", "卵"), - (0xF91D, "M", "欄"), - (0xF91E, "M", "爛"), - (0xF91F, "M", "蘭"), - (0xF920, "M", "鸞"), - (0xF921, "M", "嵐"), - (0xF922, "M", "濫"), - ] - - -def _seg_40() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xF923, "M", "藍"), - (0xF924, "M", "襤"), - (0xF925, "M", "拉"), - (0xF926, "M", "臘"), - (0xF927, "M", "蠟"), - (0xF928, "M", "廊"), - (0xF929, "M", "朗"), - (0xF92A, "M", "浪"), - (0xF92B, "M", "狼"), - (0xF92C, "M", "郎"), - (0xF92D, "M", "來"), - (0xF92E, "M", "冷"), - (0xF92F, "M", "勞"), - (0xF930, "M", "擄"), - (0xF931, "M", "櫓"), - (0xF932, "M", "爐"), - (0xF933, "M", "盧"), - (0xF934, "M", "老"), - (0xF935, "M", "蘆"), - (0xF936, "M", "虜"), - (0xF937, "M", "路"), - (0xF938, "M", "露"), - (0xF939, "M", "魯"), - (0xF93A, "M", "鷺"), - (0xF93B, "M", "碌"), - (0xF93C, "M", "祿"), - (0xF93D, "M", "綠"), - (0xF93E, "M", "菉"), - (0xF93F, "M", "錄"), - (0xF940, "M", "鹿"), - (0xF941, "M", "論"), - (0xF942, "M", "壟"), - (0xF943, "M", "弄"), - (0xF944, "M", "籠"), - (0xF945, "M", "聾"), - (0xF946, "M", "牢"), - (0xF947, "M", "磊"), - (0xF948, "M", "賂"), - (0xF949, "M", "雷"), - (0xF94A, "M", "壘"), - (0xF94B, "M", "屢"), - (0xF94C, "M", "樓"), - (0xF94D, "M", "淚"), - (0xF94E, "M", "漏"), - (0xF94F, "M", "累"), - (0xF950, "M", "縷"), - (0xF951, "M", "陋"), - (0xF952, "M", "勒"), - (0xF953, "M", "肋"), - (0xF954, "M", "凜"), - (0xF955, "M", "凌"), - (0xF956, "M", "稜"), - (0xF957, "M", "綾"), - (0xF958, "M", "菱"), - (0xF959, "M", "陵"), - (0xF95A, "M", "讀"), - (0xF95B, "M", "拏"), - (0xF95C, "M", "樂"), - (0xF95D, "M", "諾"), - (0xF95E, "M", "丹"), - (0xF95F, "M", "寧"), - (0xF960, "M", "怒"), - (0xF961, "M", "率"), - (0xF962, "M", "異"), - (0xF963, "M", "北"), - (0xF964, "M", "磻"), - (0xF965, "M", "便"), - (0xF966, "M", "復"), - (0xF967, "M", "不"), - (0xF968, "M", "泌"), - (0xF969, "M", "數"), - (0xF96A, "M", "索"), - (0xF96B, "M", "參"), - (0xF96C, "M", "塞"), - (0xF96D, "M", "省"), - (0xF96E, "M", "葉"), - (0xF96F, "M", "說"), - (0xF970, "M", "殺"), - (0xF971, "M", "辰"), - (0xF972, "M", "沈"), - (0xF973, "M", "拾"), - (0xF974, "M", "若"), - (0xF975, "M", "掠"), - (0xF976, "M", "略"), - (0xF977, "M", "亮"), - (0xF978, "M", "兩"), - (0xF979, "M", "凉"), - (0xF97A, "M", "梁"), - (0xF97B, "M", "糧"), - (0xF97C, "M", "良"), - (0xF97D, "M", "諒"), - (0xF97E, "M", "量"), - (0xF97F, "M", "勵"), - (0xF980, "M", "呂"), - (0xF981, "M", "女"), - (0xF982, "M", "廬"), - (0xF983, "M", "旅"), - (0xF984, "M", "濾"), - (0xF985, "M", "礪"), - (0xF986, "M", "閭"), - ] - - -def _seg_41() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xF987, "M", "驪"), - (0xF988, "M", "麗"), - (0xF989, "M", "黎"), - (0xF98A, "M", "力"), - (0xF98B, "M", "曆"), - (0xF98C, "M", "歷"), - (0xF98D, "M", "轢"), - (0xF98E, "M", "年"), - (0xF98F, "M", "憐"), - (0xF990, "M", "戀"), - (0xF991, "M", "撚"), - (0xF992, "M", "漣"), - (0xF993, "M", "煉"), - (0xF994, "M", "璉"), - (0xF995, "M", "秊"), - (0xF996, "M", "練"), - (0xF997, "M", "聯"), - (0xF998, "M", "輦"), - (0xF999, "M", "蓮"), - (0xF99A, "M", "連"), - (0xF99B, "M", "鍊"), - (0xF99C, "M", "列"), - (0xF99D, "M", "劣"), - (0xF99E, "M", "咽"), - (0xF99F, "M", "烈"), - (0xF9A0, "M", "裂"), - (0xF9A1, "M", "說"), - (0xF9A2, "M", "廉"), - (0xF9A3, "M", "念"), - (0xF9A4, "M", "捻"), - (0xF9A5, "M", "殮"), - (0xF9A6, "M", "簾"), - (0xF9A7, "M", "獵"), - (0xF9A8, "M", "令"), - (0xF9A9, "M", "囹"), - (0xF9AA, "M", "寧"), - (0xF9AB, "M", "嶺"), - (0xF9AC, "M", "怜"), - (0xF9AD, "M", "玲"), - (0xF9AE, "M", "瑩"), - (0xF9AF, "M", "羚"), - (0xF9B0, "M", "聆"), - (0xF9B1, "M", "鈴"), - (0xF9B2, "M", "零"), - (0xF9B3, "M", "靈"), - (0xF9B4, "M", "領"), - (0xF9B5, "M", "例"), - (0xF9B6, "M", "禮"), - (0xF9B7, "M", "醴"), - (0xF9B8, "M", "隸"), - (0xF9B9, "M", "惡"), - (0xF9BA, "M", "了"), - (0xF9BB, "M", "僚"), - (0xF9BC, "M", "寮"), - (0xF9BD, "M", "尿"), - (0xF9BE, "M", "料"), - (0xF9BF, "M", "樂"), - (0xF9C0, "M", "燎"), - (0xF9C1, "M", "療"), - (0xF9C2, "M", "蓼"), - (0xF9C3, "M", "遼"), - (0xF9C4, "M", "龍"), - (0xF9C5, "M", "暈"), - (0xF9C6, "M", "阮"), - (0xF9C7, "M", "劉"), - (0xF9C8, "M", "杻"), - (0xF9C9, "M", "柳"), - (0xF9CA, "M", "流"), - (0xF9CB, "M", "溜"), - (0xF9CC, "M", "琉"), - (0xF9CD, "M", "留"), - (0xF9CE, "M", "硫"), - (0xF9CF, "M", "紐"), - (0xF9D0, "M", "類"), - (0xF9D1, "M", "六"), - (0xF9D2, "M", "戮"), - (0xF9D3, "M", "陸"), - (0xF9D4, "M", "倫"), - (0xF9D5, "M", "崙"), - (0xF9D6, "M", "淪"), - (0xF9D7, "M", "輪"), - (0xF9D8, "M", "律"), - (0xF9D9, "M", "慄"), - (0xF9DA, "M", "栗"), - (0xF9DB, "M", "率"), - (0xF9DC, "M", "隆"), - (0xF9DD, "M", "利"), - (0xF9DE, "M", "吏"), - (0xF9DF, "M", "履"), - (0xF9E0, "M", "易"), - (0xF9E1, "M", "李"), - (0xF9E2, "M", "梨"), - (0xF9E3, "M", "泥"), - (0xF9E4, "M", "理"), - (0xF9E5, "M", "痢"), - (0xF9E6, "M", "罹"), - (0xF9E7, "M", "裏"), - (0xF9E8, "M", "裡"), - (0xF9E9, "M", "里"), - (0xF9EA, "M", "離"), - ] - - -def _seg_42() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xF9EB, "M", "匿"), - (0xF9EC, "M", "溺"), - (0xF9ED, "M", "吝"), - (0xF9EE, "M", "燐"), - (0xF9EF, "M", "璘"), - (0xF9F0, "M", "藺"), - (0xF9F1, "M", "隣"), - (0xF9F2, "M", "鱗"), - (0xF9F3, "M", "麟"), - (0xF9F4, "M", "林"), - (0xF9F5, "M", "淋"), - (0xF9F6, "M", "臨"), - (0xF9F7, "M", "立"), - (0xF9F8, "M", "笠"), - (0xF9F9, "M", "粒"), - (0xF9FA, "M", "狀"), - (0xF9FB, "M", "炙"), - (0xF9FC, "M", "識"), - (0xF9FD, "M", "什"), - (0xF9FE, "M", "茶"), - (0xF9FF, "M", "刺"), - (0xFA00, "M", "切"), - (0xFA01, "M", "度"), - (0xFA02, "M", "拓"), - (0xFA03, "M", "糖"), - (0xFA04, "M", "宅"), - (0xFA05, "M", "洞"), - (0xFA06, "M", "暴"), - (0xFA07, "M", "輻"), - (0xFA08, "M", "行"), - (0xFA09, "M", "降"), - (0xFA0A, "M", "見"), - (0xFA0B, "M", "廓"), - (0xFA0C, "M", "兀"), - (0xFA0D, "M", "嗀"), - (0xFA0E, "V"), - (0xFA10, "M", "塚"), - (0xFA11, "V"), - (0xFA12, "M", "晴"), - (0xFA13, "V"), - (0xFA15, "M", "凞"), - (0xFA16, "M", "猪"), - (0xFA17, "M", "益"), - (0xFA18, "M", "礼"), - (0xFA19, "M", "神"), - (0xFA1A, "M", "祥"), - (0xFA1B, "M", "福"), - (0xFA1C, "M", "靖"), - (0xFA1D, "M", "精"), - (0xFA1E, "M", "羽"), - (0xFA1F, "V"), - (0xFA20, "M", "蘒"), - (0xFA21, "V"), - (0xFA22, "M", "諸"), - (0xFA23, "V"), - (0xFA25, "M", "逸"), - (0xFA26, "M", "都"), - (0xFA27, "V"), - (0xFA2A, "M", "飯"), - (0xFA2B, "M", "飼"), - (0xFA2C, "M", "館"), - (0xFA2D, "M", "鶴"), - (0xFA2E, "M", "郞"), - (0xFA2F, "M", "隷"), - (0xFA30, "M", "侮"), - (0xFA31, "M", "僧"), - (0xFA32, "M", "免"), - (0xFA33, "M", "勉"), - (0xFA34, "M", "勤"), - (0xFA35, "M", "卑"), - (0xFA36, "M", "喝"), - (0xFA37, "M", "嘆"), - (0xFA38, "M", "器"), - (0xFA39, "M", "塀"), - (0xFA3A, "M", "墨"), - (0xFA3B, "M", "層"), - (0xFA3C, "M", "屮"), - (0xFA3D, "M", "悔"), - (0xFA3E, "M", "慨"), - (0xFA3F, "M", "憎"), - (0xFA40, "M", "懲"), - (0xFA41, "M", "敏"), - (0xFA42, "M", "既"), - (0xFA43, "M", "暑"), - (0xFA44, "M", "梅"), - (0xFA45, "M", "海"), - (0xFA46, "M", "渚"), - (0xFA47, "M", "漢"), - (0xFA48, "M", "煮"), - (0xFA49, "M", "爫"), - (0xFA4A, "M", "琢"), - (0xFA4B, "M", "碑"), - (0xFA4C, "M", "社"), - (0xFA4D, "M", "祉"), - (0xFA4E, "M", "祈"), - (0xFA4F, "M", "祐"), - (0xFA50, "M", "祖"), - (0xFA51, "M", "祝"), - (0xFA52, "M", "禍"), - (0xFA53, "M", "禎"), - ] - - -def _seg_43() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFA54, "M", "穀"), - (0xFA55, "M", "突"), - (0xFA56, "M", "節"), - (0xFA57, "M", "練"), - (0xFA58, "M", "縉"), - (0xFA59, "M", "繁"), - (0xFA5A, "M", "署"), - (0xFA5B, "M", "者"), - (0xFA5C, "M", "臭"), - (0xFA5D, "M", "艹"), - (0xFA5F, "M", "著"), - (0xFA60, "M", "褐"), - (0xFA61, "M", "視"), - (0xFA62, "M", "謁"), - (0xFA63, "M", "謹"), - (0xFA64, "M", "賓"), - (0xFA65, "M", "贈"), - (0xFA66, "M", "辶"), - (0xFA67, "M", "逸"), - (0xFA68, "M", "難"), - (0xFA69, "M", "響"), - (0xFA6A, "M", "頻"), - (0xFA6B, "M", "恵"), - (0xFA6C, "M", "𤋮"), - (0xFA6D, "M", "舘"), - (0xFA6E, "X"), - (0xFA70, "M", "並"), - (0xFA71, "M", "况"), - (0xFA72, "M", "全"), - (0xFA73, "M", "侀"), - (0xFA74, "M", "充"), - (0xFA75, "M", "冀"), - (0xFA76, "M", "勇"), - (0xFA77, "M", "勺"), - (0xFA78, "M", "喝"), - (0xFA79, "M", "啕"), - (0xFA7A, "M", "喙"), - (0xFA7B, "M", "嗢"), - (0xFA7C, "M", "塚"), - (0xFA7D, "M", "墳"), - (0xFA7E, "M", "奄"), - (0xFA7F, "M", "奔"), - (0xFA80, "M", "婢"), - (0xFA81, "M", "嬨"), - (0xFA82, "M", "廒"), - (0xFA83, "M", "廙"), - (0xFA84, "M", "彩"), - (0xFA85, "M", "徭"), - (0xFA86, "M", "惘"), - (0xFA87, "M", "慎"), - (0xFA88, "M", "愈"), - (0xFA89, "M", "憎"), - (0xFA8A, "M", "慠"), - (0xFA8B, "M", "懲"), - (0xFA8C, "M", "戴"), - (0xFA8D, "M", "揄"), - (0xFA8E, "M", "搜"), - (0xFA8F, "M", "摒"), - (0xFA90, "M", "敖"), - (0xFA91, "M", "晴"), - (0xFA92, "M", "朗"), - (0xFA93, "M", "望"), - (0xFA94, "M", "杖"), - (0xFA95, "M", "歹"), - (0xFA96, "M", "殺"), - (0xFA97, "M", "流"), - (0xFA98, "M", "滛"), - (0xFA99, "M", "滋"), - (0xFA9A, "M", "漢"), - (0xFA9B, "M", "瀞"), - (0xFA9C, "M", "煮"), - (0xFA9D, "M", "瞧"), - (0xFA9E, "M", "爵"), - (0xFA9F, "M", "犯"), - (0xFAA0, "M", "猪"), - (0xFAA1, "M", "瑱"), - (0xFAA2, "M", "甆"), - (0xFAA3, "M", "画"), - (0xFAA4, "M", "瘝"), - (0xFAA5, "M", "瘟"), - (0xFAA6, "M", "益"), - (0xFAA7, "M", "盛"), - (0xFAA8, "M", "直"), - (0xFAA9, "M", "睊"), - (0xFAAA, "M", "着"), - (0xFAAB, "M", "磌"), - (0xFAAC, "M", "窱"), - (0xFAAD, "M", "節"), - (0xFAAE, "M", "类"), - (0xFAAF, "M", "絛"), - (0xFAB0, "M", "練"), - (0xFAB1, "M", "缾"), - (0xFAB2, "M", "者"), - (0xFAB3, "M", "荒"), - (0xFAB4, "M", "華"), - (0xFAB5, "M", "蝹"), - (0xFAB6, "M", "襁"), - (0xFAB7, "M", "覆"), - (0xFAB8, "M", "視"), - (0xFAB9, "M", "調"), - ] - - -def _seg_44() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFABA, "M", "諸"), - (0xFABB, "M", "請"), - (0xFABC, "M", "謁"), - (0xFABD, "M", "諾"), - (0xFABE, "M", "諭"), - (0xFABF, "M", "謹"), - (0xFAC0, "M", "變"), - (0xFAC1, "M", "贈"), - (0xFAC2, "M", "輸"), - (0xFAC3, "M", "遲"), - (0xFAC4, "M", "醙"), - (0xFAC5, "M", "鉶"), - (0xFAC6, "M", "陼"), - (0xFAC7, "M", "難"), - (0xFAC8, "M", "靖"), - (0xFAC9, "M", "韛"), - (0xFACA, "M", "響"), - (0xFACB, "M", "頋"), - (0xFACC, "M", "頻"), - (0xFACD, "M", "鬒"), - (0xFACE, "M", "龜"), - (0xFACF, "M", "𢡊"), - (0xFAD0, "M", "𢡄"), - (0xFAD1, "M", "𣏕"), - (0xFAD2, "M", "㮝"), - (0xFAD3, "M", "䀘"), - (0xFAD4, "M", "䀹"), - (0xFAD5, "M", "𥉉"), - (0xFAD6, "M", "𥳐"), - (0xFAD7, "M", "𧻓"), - (0xFAD8, "M", "齃"), - (0xFAD9, "M", "龎"), - (0xFADA, "X"), - (0xFB00, "M", "ff"), - (0xFB01, "M", "fi"), - (0xFB02, "M", "fl"), - (0xFB03, "M", "ffi"), - (0xFB04, "M", "ffl"), - (0xFB05, "M", "st"), - (0xFB07, "X"), - (0xFB13, "M", "մն"), - (0xFB14, "M", "մե"), - (0xFB15, "M", "մի"), - (0xFB16, "M", "վն"), - (0xFB17, "M", "մխ"), - (0xFB18, "X"), - (0xFB1D, "M", "יִ"), - (0xFB1E, "V"), - (0xFB1F, "M", "ײַ"), - (0xFB20, "M", "ע"), - (0xFB21, "M", "א"), - (0xFB22, "M", "ד"), - (0xFB23, "M", "ה"), - (0xFB24, "M", "כ"), - (0xFB25, "M", "ל"), - (0xFB26, "M", "ם"), - (0xFB27, "M", "ר"), - (0xFB28, "M", "ת"), - (0xFB29, "M", "+"), - (0xFB2A, "M", "שׁ"), - (0xFB2B, "M", "שׂ"), - (0xFB2C, "M", "שּׁ"), - (0xFB2D, "M", "שּׂ"), - (0xFB2E, "M", "אַ"), - (0xFB2F, "M", "אָ"), - (0xFB30, "M", "אּ"), - (0xFB31, "M", "בּ"), - (0xFB32, "M", "גּ"), - (0xFB33, "M", "דּ"), - (0xFB34, "M", "הּ"), - (0xFB35, "M", "וּ"), - (0xFB36, "M", "זּ"), - (0xFB37, "X"), - (0xFB38, "M", "טּ"), - (0xFB39, "M", "יּ"), - (0xFB3A, "M", "ךּ"), - (0xFB3B, "M", "כּ"), - (0xFB3C, "M", "לּ"), - (0xFB3D, "X"), - (0xFB3E, "M", "מּ"), - (0xFB3F, "X"), - (0xFB40, "M", "נּ"), - (0xFB41, "M", "סּ"), - (0xFB42, "X"), - (0xFB43, "M", "ףּ"), - (0xFB44, "M", "פּ"), - (0xFB45, "X"), - (0xFB46, "M", "צּ"), - (0xFB47, "M", "קּ"), - (0xFB48, "M", "רּ"), - (0xFB49, "M", "שּ"), - (0xFB4A, "M", "תּ"), - (0xFB4B, "M", "וֹ"), - (0xFB4C, "M", "בֿ"), - (0xFB4D, "M", "כֿ"), - (0xFB4E, "M", "פֿ"), - (0xFB4F, "M", "אל"), - (0xFB50, "M", "ٱ"), - (0xFB52, "M", "ٻ"), - (0xFB56, "M", "پ"), - ] - - -def _seg_45() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFB5A, "M", "ڀ"), - (0xFB5E, "M", "ٺ"), - (0xFB62, "M", "ٿ"), - (0xFB66, "M", "ٹ"), - (0xFB6A, "M", "ڤ"), - (0xFB6E, "M", "ڦ"), - (0xFB72, "M", "ڄ"), - (0xFB76, "M", "ڃ"), - (0xFB7A, "M", "چ"), - (0xFB7E, "M", "ڇ"), - (0xFB82, "M", "ڍ"), - (0xFB84, "M", "ڌ"), - (0xFB86, "M", "ڎ"), - (0xFB88, "M", "ڈ"), - (0xFB8A, "M", "ژ"), - (0xFB8C, "M", "ڑ"), - (0xFB8E, "M", "ک"), - (0xFB92, "M", "گ"), - (0xFB96, "M", "ڳ"), - (0xFB9A, "M", "ڱ"), - (0xFB9E, "M", "ں"), - (0xFBA0, "M", "ڻ"), - (0xFBA4, "M", "ۀ"), - (0xFBA6, "M", "ہ"), - (0xFBAA, "M", "ھ"), - (0xFBAE, "M", "ے"), - (0xFBB0, "M", "ۓ"), - (0xFBB2, "V"), - (0xFBC3, "X"), - (0xFBD3, "M", "ڭ"), - (0xFBD7, "M", "ۇ"), - (0xFBD9, "M", "ۆ"), - (0xFBDB, "M", "ۈ"), - (0xFBDD, "M", "ۇٴ"), - (0xFBDE, "M", "ۋ"), - (0xFBE0, "M", "ۅ"), - (0xFBE2, "M", "ۉ"), - (0xFBE4, "M", "ې"), - (0xFBE8, "M", "ى"), - (0xFBEA, "M", "ئا"), - (0xFBEC, "M", "ئە"), - (0xFBEE, "M", "ئو"), - (0xFBF0, "M", "ئۇ"), - (0xFBF2, "M", "ئۆ"), - (0xFBF4, "M", "ئۈ"), - (0xFBF6, "M", "ئې"), - (0xFBF9, "M", "ئى"), - (0xFBFC, "M", "ی"), - (0xFC00, "M", "ئج"), - (0xFC01, "M", "ئح"), - (0xFC02, "M", "ئم"), - (0xFC03, "M", "ئى"), - (0xFC04, "M", "ئي"), - (0xFC05, "M", "بج"), - (0xFC06, "M", "بح"), - (0xFC07, "M", "بخ"), - (0xFC08, "M", "بم"), - (0xFC09, "M", "بى"), - (0xFC0A, "M", "بي"), - (0xFC0B, "M", "تج"), - (0xFC0C, "M", "تح"), - (0xFC0D, "M", "تخ"), - (0xFC0E, "M", "تم"), - (0xFC0F, "M", "تى"), - (0xFC10, "M", "تي"), - (0xFC11, "M", "ثج"), - (0xFC12, "M", "ثم"), - (0xFC13, "M", "ثى"), - (0xFC14, "M", "ثي"), - (0xFC15, "M", "جح"), - (0xFC16, "M", "جم"), - (0xFC17, "M", "حج"), - (0xFC18, "M", "حم"), - (0xFC19, "M", "خج"), - (0xFC1A, "M", "خح"), - (0xFC1B, "M", "خم"), - (0xFC1C, "M", "سج"), - (0xFC1D, "M", "سح"), - (0xFC1E, "M", "سخ"), - (0xFC1F, "M", "سم"), - (0xFC20, "M", "صح"), - (0xFC21, "M", "صم"), - (0xFC22, "M", "ضج"), - (0xFC23, "M", "ضح"), - (0xFC24, "M", "ضخ"), - (0xFC25, "M", "ضم"), - (0xFC26, "M", "طح"), - (0xFC27, "M", "طم"), - (0xFC28, "M", "ظم"), - (0xFC29, "M", "عج"), - (0xFC2A, "M", "عم"), - (0xFC2B, "M", "غج"), - (0xFC2C, "M", "غم"), - (0xFC2D, "M", "فج"), - (0xFC2E, "M", "فح"), - (0xFC2F, "M", "فخ"), - (0xFC30, "M", "فم"), - (0xFC31, "M", "فى"), - (0xFC32, "M", "في"), - (0xFC33, "M", "قح"), - ] - - -def _seg_46() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFC34, "M", "قم"), - (0xFC35, "M", "قى"), - (0xFC36, "M", "قي"), - (0xFC37, "M", "كا"), - (0xFC38, "M", "كج"), - (0xFC39, "M", "كح"), - (0xFC3A, "M", "كخ"), - (0xFC3B, "M", "كل"), - (0xFC3C, "M", "كم"), - (0xFC3D, "M", "كى"), - (0xFC3E, "M", "كي"), - (0xFC3F, "M", "لج"), - (0xFC40, "M", "لح"), - (0xFC41, "M", "لخ"), - (0xFC42, "M", "لم"), - (0xFC43, "M", "لى"), - (0xFC44, "M", "لي"), - (0xFC45, "M", "مج"), - (0xFC46, "M", "مح"), - (0xFC47, "M", "مخ"), - (0xFC48, "M", "مم"), - (0xFC49, "M", "مى"), - (0xFC4A, "M", "مي"), - (0xFC4B, "M", "نج"), - (0xFC4C, "M", "نح"), - (0xFC4D, "M", "نخ"), - (0xFC4E, "M", "نم"), - (0xFC4F, "M", "نى"), - (0xFC50, "M", "ني"), - (0xFC51, "M", "هج"), - (0xFC52, "M", "هم"), - (0xFC53, "M", "هى"), - (0xFC54, "M", "هي"), - (0xFC55, "M", "يج"), - (0xFC56, "M", "يح"), - (0xFC57, "M", "يخ"), - (0xFC58, "M", "يم"), - (0xFC59, "M", "يى"), - (0xFC5A, "M", "يي"), - (0xFC5B, "M", "ذٰ"), - (0xFC5C, "M", "رٰ"), - (0xFC5D, "M", "ىٰ"), - (0xFC5E, "M", " ٌّ"), - (0xFC5F, "M", " ٍّ"), - (0xFC60, "M", " َّ"), - (0xFC61, "M", " ُّ"), - (0xFC62, "M", " ِّ"), - (0xFC63, "M", " ّٰ"), - (0xFC64, "M", "ئر"), - (0xFC65, "M", "ئز"), - (0xFC66, "M", "ئم"), - (0xFC67, "M", "ئن"), - (0xFC68, "M", "ئى"), - (0xFC69, "M", "ئي"), - (0xFC6A, "M", "بر"), - (0xFC6B, "M", "بز"), - (0xFC6C, "M", "بم"), - (0xFC6D, "M", "بن"), - (0xFC6E, "M", "بى"), - (0xFC6F, "M", "بي"), - (0xFC70, "M", "تر"), - (0xFC71, "M", "تز"), - (0xFC72, "M", "تم"), - (0xFC73, "M", "تن"), - (0xFC74, "M", "تى"), - (0xFC75, "M", "تي"), - (0xFC76, "M", "ثر"), - (0xFC77, "M", "ثز"), - (0xFC78, "M", "ثم"), - (0xFC79, "M", "ثن"), - (0xFC7A, "M", "ثى"), - (0xFC7B, "M", "ثي"), - (0xFC7C, "M", "فى"), - (0xFC7D, "M", "في"), - (0xFC7E, "M", "قى"), - (0xFC7F, "M", "قي"), - (0xFC80, "M", "كا"), - (0xFC81, "M", "كل"), - (0xFC82, "M", "كم"), - (0xFC83, "M", "كى"), - (0xFC84, "M", "كي"), - (0xFC85, "M", "لم"), - (0xFC86, "M", "لى"), - (0xFC87, "M", "لي"), - (0xFC88, "M", "ما"), - (0xFC89, "M", "مم"), - (0xFC8A, "M", "نر"), - (0xFC8B, "M", "نز"), - (0xFC8C, "M", "نم"), - (0xFC8D, "M", "نن"), - (0xFC8E, "M", "نى"), - (0xFC8F, "M", "ني"), - (0xFC90, "M", "ىٰ"), - (0xFC91, "M", "ير"), - (0xFC92, "M", "يز"), - (0xFC93, "M", "يم"), - (0xFC94, "M", "ين"), - (0xFC95, "M", "يى"), - (0xFC96, "M", "يي"), - (0xFC97, "M", "ئج"), - ] - - -def _seg_47() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFC98, "M", "ئح"), - (0xFC99, "M", "ئخ"), - (0xFC9A, "M", "ئم"), - (0xFC9B, "M", "ئه"), - (0xFC9C, "M", "بج"), - (0xFC9D, "M", "بح"), - (0xFC9E, "M", "بخ"), - (0xFC9F, "M", "بم"), - (0xFCA0, "M", "به"), - (0xFCA1, "M", "تج"), - (0xFCA2, "M", "تح"), - (0xFCA3, "M", "تخ"), - (0xFCA4, "M", "تم"), - (0xFCA5, "M", "ته"), - (0xFCA6, "M", "ثم"), - (0xFCA7, "M", "جح"), - (0xFCA8, "M", "جم"), - (0xFCA9, "M", "حج"), - (0xFCAA, "M", "حم"), - (0xFCAB, "M", "خج"), - (0xFCAC, "M", "خم"), - (0xFCAD, "M", "سج"), - (0xFCAE, "M", "سح"), - (0xFCAF, "M", "سخ"), - (0xFCB0, "M", "سم"), - (0xFCB1, "M", "صح"), - (0xFCB2, "M", "صخ"), - (0xFCB3, "M", "صم"), - (0xFCB4, "M", "ضج"), - (0xFCB5, "M", "ضح"), - (0xFCB6, "M", "ضخ"), - (0xFCB7, "M", "ضم"), - (0xFCB8, "M", "طح"), - (0xFCB9, "M", "ظم"), - (0xFCBA, "M", "عج"), - (0xFCBB, "M", "عم"), - (0xFCBC, "M", "غج"), - (0xFCBD, "M", "غم"), - (0xFCBE, "M", "فج"), - (0xFCBF, "M", "فح"), - (0xFCC0, "M", "فخ"), - (0xFCC1, "M", "فم"), - (0xFCC2, "M", "قح"), - (0xFCC3, "M", "قم"), - (0xFCC4, "M", "كج"), - (0xFCC5, "M", "كح"), - (0xFCC6, "M", "كخ"), - (0xFCC7, "M", "كل"), - (0xFCC8, "M", "كم"), - (0xFCC9, "M", "لج"), - (0xFCCA, "M", "لح"), - (0xFCCB, "M", "لخ"), - (0xFCCC, "M", "لم"), - (0xFCCD, "M", "له"), - (0xFCCE, "M", "مج"), - (0xFCCF, "M", "مح"), - (0xFCD0, "M", "مخ"), - (0xFCD1, "M", "مم"), - (0xFCD2, "M", "نج"), - (0xFCD3, "M", "نح"), - (0xFCD4, "M", "نخ"), - (0xFCD5, "M", "نم"), - (0xFCD6, "M", "نه"), - (0xFCD7, "M", "هج"), - (0xFCD8, "M", "هم"), - (0xFCD9, "M", "هٰ"), - (0xFCDA, "M", "يج"), - (0xFCDB, "M", "يح"), - (0xFCDC, "M", "يخ"), - (0xFCDD, "M", "يم"), - (0xFCDE, "M", "يه"), - (0xFCDF, "M", "ئم"), - (0xFCE0, "M", "ئه"), - (0xFCE1, "M", "بم"), - (0xFCE2, "M", "به"), - (0xFCE3, "M", "تم"), - (0xFCE4, "M", "ته"), - (0xFCE5, "M", "ثم"), - (0xFCE6, "M", "ثه"), - (0xFCE7, "M", "سم"), - (0xFCE8, "M", "سه"), - (0xFCE9, "M", "شم"), - (0xFCEA, "M", "شه"), - (0xFCEB, "M", "كل"), - (0xFCEC, "M", "كم"), - (0xFCED, "M", "لم"), - (0xFCEE, "M", "نم"), - (0xFCEF, "M", "نه"), - (0xFCF0, "M", "يم"), - (0xFCF1, "M", "يه"), - (0xFCF2, "M", "ـَّ"), - (0xFCF3, "M", "ـُّ"), - (0xFCF4, "M", "ـِّ"), - (0xFCF5, "M", "طى"), - (0xFCF6, "M", "طي"), - (0xFCF7, "M", "عى"), - (0xFCF8, "M", "عي"), - (0xFCF9, "M", "غى"), - (0xFCFA, "M", "غي"), - (0xFCFB, "M", "سى"), - ] - - -def _seg_48() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFCFC, "M", "سي"), - (0xFCFD, "M", "شى"), - (0xFCFE, "M", "شي"), - (0xFCFF, "M", "حى"), - (0xFD00, "M", "حي"), - (0xFD01, "M", "جى"), - (0xFD02, "M", "جي"), - (0xFD03, "M", "خى"), - (0xFD04, "M", "خي"), - (0xFD05, "M", "صى"), - (0xFD06, "M", "صي"), - (0xFD07, "M", "ضى"), - (0xFD08, "M", "ضي"), - (0xFD09, "M", "شج"), - (0xFD0A, "M", "شح"), - (0xFD0B, "M", "شخ"), - (0xFD0C, "M", "شم"), - (0xFD0D, "M", "شر"), - (0xFD0E, "M", "سر"), - (0xFD0F, "M", "صر"), - (0xFD10, "M", "ضر"), - (0xFD11, "M", "طى"), - (0xFD12, "M", "طي"), - (0xFD13, "M", "عى"), - (0xFD14, "M", "عي"), - (0xFD15, "M", "غى"), - (0xFD16, "M", "غي"), - (0xFD17, "M", "سى"), - (0xFD18, "M", "سي"), - (0xFD19, "M", "شى"), - (0xFD1A, "M", "شي"), - (0xFD1B, "M", "حى"), - (0xFD1C, "M", "حي"), - (0xFD1D, "M", "جى"), - (0xFD1E, "M", "جي"), - (0xFD1F, "M", "خى"), - (0xFD20, "M", "خي"), - (0xFD21, "M", "صى"), - (0xFD22, "M", "صي"), - (0xFD23, "M", "ضى"), - (0xFD24, "M", "ضي"), - (0xFD25, "M", "شج"), - (0xFD26, "M", "شح"), - (0xFD27, "M", "شخ"), - (0xFD28, "M", "شم"), - (0xFD29, "M", "شر"), - (0xFD2A, "M", "سر"), - (0xFD2B, "M", "صر"), - (0xFD2C, "M", "ضر"), - (0xFD2D, "M", "شج"), - (0xFD2E, "M", "شح"), - (0xFD2F, "M", "شخ"), - (0xFD30, "M", "شم"), - (0xFD31, "M", "سه"), - (0xFD32, "M", "شه"), - (0xFD33, "M", "طم"), - (0xFD34, "M", "سج"), - (0xFD35, "M", "سح"), - (0xFD36, "M", "سخ"), - (0xFD37, "M", "شج"), - (0xFD38, "M", "شح"), - (0xFD39, "M", "شخ"), - (0xFD3A, "M", "طم"), - (0xFD3B, "M", "ظم"), - (0xFD3C, "M", "اً"), - (0xFD3E, "V"), - (0xFD50, "M", "تجم"), - (0xFD51, "M", "تحج"), - (0xFD53, "M", "تحم"), - (0xFD54, "M", "تخم"), - (0xFD55, "M", "تمج"), - (0xFD56, "M", "تمح"), - (0xFD57, "M", "تمخ"), - (0xFD58, "M", "جمح"), - (0xFD5A, "M", "حمي"), - (0xFD5B, "M", "حمى"), - (0xFD5C, "M", "سحج"), - (0xFD5D, "M", "سجح"), - (0xFD5E, "M", "سجى"), - (0xFD5F, "M", "سمح"), - (0xFD61, "M", "سمج"), - (0xFD62, "M", "سمم"), - (0xFD64, "M", "صحح"), - (0xFD66, "M", "صمم"), - (0xFD67, "M", "شحم"), - (0xFD69, "M", "شجي"), - (0xFD6A, "M", "شمخ"), - (0xFD6C, "M", "شمم"), - (0xFD6E, "M", "ضحى"), - (0xFD6F, "M", "ضخم"), - (0xFD71, "M", "طمح"), - (0xFD73, "M", "طمم"), - (0xFD74, "M", "طمي"), - (0xFD75, "M", "عجم"), - (0xFD76, "M", "عمم"), - (0xFD78, "M", "عمى"), - (0xFD79, "M", "غمم"), - (0xFD7A, "M", "غمي"), - (0xFD7B, "M", "غمى"), - (0xFD7C, "M", "فخم"), - ] - - -def _seg_49() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFD7E, "M", "قمح"), - (0xFD7F, "M", "قمم"), - (0xFD80, "M", "لحم"), - (0xFD81, "M", "لحي"), - (0xFD82, "M", "لحى"), - (0xFD83, "M", "لجج"), - (0xFD85, "M", "لخم"), - (0xFD87, "M", "لمح"), - (0xFD89, "M", "محج"), - (0xFD8A, "M", "محم"), - (0xFD8B, "M", "محي"), - (0xFD8C, "M", "مجح"), - (0xFD8D, "M", "مجم"), - (0xFD8E, "M", "مخج"), - (0xFD8F, "M", "مخم"), - (0xFD90, "X"), - (0xFD92, "M", "مجخ"), - (0xFD93, "M", "همج"), - (0xFD94, "M", "همم"), - (0xFD95, "M", "نحم"), - (0xFD96, "M", "نحى"), - (0xFD97, "M", "نجم"), - (0xFD99, "M", "نجى"), - (0xFD9A, "M", "نمي"), - (0xFD9B, "M", "نمى"), - (0xFD9C, "M", "يمم"), - (0xFD9E, "M", "بخي"), - (0xFD9F, "M", "تجي"), - (0xFDA0, "M", "تجى"), - (0xFDA1, "M", "تخي"), - (0xFDA2, "M", "تخى"), - (0xFDA3, "M", "تمي"), - (0xFDA4, "M", "تمى"), - (0xFDA5, "M", "جمي"), - (0xFDA6, "M", "جحى"), - (0xFDA7, "M", "جمى"), - (0xFDA8, "M", "سخى"), - (0xFDA9, "M", "صحي"), - (0xFDAA, "M", "شحي"), - (0xFDAB, "M", "ضحي"), - (0xFDAC, "M", "لجي"), - (0xFDAD, "M", "لمي"), - (0xFDAE, "M", "يحي"), - (0xFDAF, "M", "يجي"), - (0xFDB0, "M", "يمي"), - (0xFDB1, "M", "ممي"), - (0xFDB2, "M", "قمي"), - (0xFDB3, "M", "نحي"), - (0xFDB4, "M", "قمح"), - (0xFDB5, "M", "لحم"), - (0xFDB6, "M", "عمي"), - (0xFDB7, "M", "كمي"), - (0xFDB8, "M", "نجح"), - (0xFDB9, "M", "مخي"), - (0xFDBA, "M", "لجم"), - (0xFDBB, "M", "كمم"), - (0xFDBC, "M", "لجم"), - (0xFDBD, "M", "نجح"), - (0xFDBE, "M", "جحي"), - (0xFDBF, "M", "حجي"), - (0xFDC0, "M", "مجي"), - (0xFDC1, "M", "فمي"), - (0xFDC2, "M", "بحي"), - (0xFDC3, "M", "كمم"), - (0xFDC4, "M", "عجم"), - (0xFDC5, "M", "صمم"), - (0xFDC6, "M", "سخي"), - (0xFDC7, "M", "نجي"), - (0xFDC8, "X"), - (0xFDCF, "V"), - (0xFDD0, "X"), - (0xFDF0, "M", "صلے"), - (0xFDF1, "M", "قلے"), - (0xFDF2, "M", "الله"), - (0xFDF3, "M", "اكبر"), - (0xFDF4, "M", "محمد"), - (0xFDF5, "M", "صلعم"), - (0xFDF6, "M", "رسول"), - (0xFDF7, "M", "عليه"), - (0xFDF8, "M", "وسلم"), - (0xFDF9, "M", "صلى"), - (0xFDFA, "M", "صلى الله عليه وسلم"), - (0xFDFB, "M", "جل جلاله"), - (0xFDFC, "M", "ریال"), - (0xFDFD, "V"), - (0xFE00, "I"), - (0xFE10, "M", ","), - (0xFE11, "M", "、"), - (0xFE12, "X"), - (0xFE13, "M", ":"), - (0xFE14, "M", ";"), - (0xFE15, "M", "!"), - (0xFE16, "M", "?"), - (0xFE17, "M", "〖"), - (0xFE18, "M", "〗"), - (0xFE19, "X"), - (0xFE20, "V"), - (0xFE30, "X"), - (0xFE31, "M", "—"), - (0xFE32, "M", "–"), - ] - - -def _seg_50() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFE33, "M", "_"), - (0xFE35, "M", "("), - (0xFE36, "M", ")"), - (0xFE37, "M", "{"), - (0xFE38, "M", "}"), - (0xFE39, "M", "〔"), - (0xFE3A, "M", "〕"), - (0xFE3B, "M", "【"), - (0xFE3C, "M", "】"), - (0xFE3D, "M", "《"), - (0xFE3E, "M", "》"), - (0xFE3F, "M", "〈"), - (0xFE40, "M", "〉"), - (0xFE41, "M", "「"), - (0xFE42, "M", "」"), - (0xFE43, "M", "『"), - (0xFE44, "M", "』"), - (0xFE45, "V"), - (0xFE47, "M", "["), - (0xFE48, "M", "]"), - (0xFE49, "M", " ̅"), - (0xFE4D, "M", "_"), - (0xFE50, "M", ","), - (0xFE51, "M", "、"), - (0xFE52, "X"), - (0xFE54, "M", ";"), - (0xFE55, "M", ":"), - (0xFE56, "M", "?"), - (0xFE57, "M", "!"), - (0xFE58, "M", "—"), - (0xFE59, "M", "("), - (0xFE5A, "M", ")"), - (0xFE5B, "M", "{"), - (0xFE5C, "M", "}"), - (0xFE5D, "M", "〔"), - (0xFE5E, "M", "〕"), - (0xFE5F, "M", "#"), - (0xFE60, "M", "&"), - (0xFE61, "M", "*"), - (0xFE62, "M", "+"), - (0xFE63, "M", "-"), - (0xFE64, "M", "<"), - (0xFE65, "M", ">"), - (0xFE66, "M", "="), - (0xFE67, "X"), - (0xFE68, "M", "\\"), - (0xFE69, "M", "$"), - (0xFE6A, "M", "%"), - (0xFE6B, "M", "@"), - (0xFE6C, "X"), - (0xFE70, "M", " ً"), - (0xFE71, "M", "ـً"), - (0xFE72, "M", " ٌ"), - (0xFE73, "V"), - (0xFE74, "M", " ٍ"), - (0xFE75, "X"), - (0xFE76, "M", " َ"), - (0xFE77, "M", "ـَ"), - (0xFE78, "M", " ُ"), - (0xFE79, "M", "ـُ"), - (0xFE7A, "M", " ِ"), - (0xFE7B, "M", "ـِ"), - (0xFE7C, "M", " ّ"), - (0xFE7D, "M", "ـّ"), - (0xFE7E, "M", " ْ"), - (0xFE7F, "M", "ـْ"), - (0xFE80, "M", "ء"), - (0xFE81, "M", "آ"), - (0xFE83, "M", "أ"), - (0xFE85, "M", "ؤ"), - (0xFE87, "M", "إ"), - (0xFE89, "M", "ئ"), - (0xFE8D, "M", "ا"), - (0xFE8F, "M", "ب"), - (0xFE93, "M", "ة"), - (0xFE95, "M", "ت"), - (0xFE99, "M", "ث"), - (0xFE9D, "M", "ج"), - (0xFEA1, "M", "ح"), - (0xFEA5, "M", "خ"), - (0xFEA9, "M", "د"), - (0xFEAB, "M", "ذ"), - (0xFEAD, "M", "ر"), - (0xFEAF, "M", "ز"), - (0xFEB1, "M", "س"), - (0xFEB5, "M", "ش"), - (0xFEB9, "M", "ص"), - (0xFEBD, "M", "ض"), - (0xFEC1, "M", "ط"), - (0xFEC5, "M", "ظ"), - (0xFEC9, "M", "ع"), - (0xFECD, "M", "غ"), - (0xFED1, "M", "ف"), - (0xFED5, "M", "ق"), - (0xFED9, "M", "ك"), - (0xFEDD, "M", "ل"), - (0xFEE1, "M", "م"), - (0xFEE5, "M", "ن"), - (0xFEE9, "M", "ه"), - (0xFEED, "M", "و"), - ] - - -def _seg_51() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFEEF, "M", "ى"), - (0xFEF1, "M", "ي"), - (0xFEF5, "M", "لآ"), - (0xFEF7, "M", "لأ"), - (0xFEF9, "M", "لإ"), - (0xFEFB, "M", "لا"), - (0xFEFD, "X"), - (0xFEFF, "I"), - (0xFF00, "X"), - (0xFF01, "M", "!"), - (0xFF02, "M", '"'), - (0xFF03, "M", "#"), - (0xFF04, "M", "$"), - (0xFF05, "M", "%"), - (0xFF06, "M", "&"), - (0xFF07, "M", "'"), - (0xFF08, "M", "("), - (0xFF09, "M", ")"), - (0xFF0A, "M", "*"), - (0xFF0B, "M", "+"), - (0xFF0C, "M", ","), - (0xFF0D, "M", "-"), - (0xFF0E, "M", "."), - (0xFF0F, "M", "/"), - (0xFF10, "M", "0"), - (0xFF11, "M", "1"), - (0xFF12, "M", "2"), - (0xFF13, "M", "3"), - (0xFF14, "M", "4"), - (0xFF15, "M", "5"), - (0xFF16, "M", "6"), - (0xFF17, "M", "7"), - (0xFF18, "M", "8"), - (0xFF19, "M", "9"), - (0xFF1A, "M", ":"), - (0xFF1B, "M", ";"), - (0xFF1C, "M", "<"), - (0xFF1D, "M", "="), - (0xFF1E, "M", ">"), - (0xFF1F, "M", "?"), - (0xFF20, "M", "@"), - (0xFF21, "M", "a"), - (0xFF22, "M", "b"), - (0xFF23, "M", "c"), - (0xFF24, "M", "d"), - (0xFF25, "M", "e"), - (0xFF26, "M", "f"), - (0xFF27, "M", "g"), - (0xFF28, "M", "h"), - (0xFF29, "M", "i"), - (0xFF2A, "M", "j"), - (0xFF2B, "M", "k"), - (0xFF2C, "M", "l"), - (0xFF2D, "M", "m"), - (0xFF2E, "M", "n"), - (0xFF2F, "M", "o"), - (0xFF30, "M", "p"), - (0xFF31, "M", "q"), - (0xFF32, "M", "r"), - (0xFF33, "M", "s"), - (0xFF34, "M", "t"), - (0xFF35, "M", "u"), - (0xFF36, "M", "v"), - (0xFF37, "M", "w"), - (0xFF38, "M", "x"), - (0xFF39, "M", "y"), - (0xFF3A, "M", "z"), - (0xFF3B, "M", "["), - (0xFF3C, "M", "\\"), - (0xFF3D, "M", "]"), - (0xFF3E, "M", "^"), - (0xFF3F, "M", "_"), - (0xFF40, "M", "`"), - (0xFF41, "M", "a"), - (0xFF42, "M", "b"), - (0xFF43, "M", "c"), - (0xFF44, "M", "d"), - (0xFF45, "M", "e"), - (0xFF46, "M", "f"), - (0xFF47, "M", "g"), - (0xFF48, "M", "h"), - (0xFF49, "M", "i"), - (0xFF4A, "M", "j"), - (0xFF4B, "M", "k"), - (0xFF4C, "M", "l"), - (0xFF4D, "M", "m"), - (0xFF4E, "M", "n"), - (0xFF4F, "M", "o"), - (0xFF50, "M", "p"), - (0xFF51, "M", "q"), - (0xFF52, "M", "r"), - (0xFF53, "M", "s"), - (0xFF54, "M", "t"), - (0xFF55, "M", "u"), - (0xFF56, "M", "v"), - (0xFF57, "M", "w"), - (0xFF58, "M", "x"), - (0xFF59, "M", "y"), - (0xFF5A, "M", "z"), - (0xFF5B, "M", "{"), - ] - - -def _seg_52() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFF5C, "M", "|"), - (0xFF5D, "M", "}"), - (0xFF5E, "M", "~"), - (0xFF5F, "M", "⦅"), - (0xFF60, "M", "⦆"), - (0xFF61, "M", "."), - (0xFF62, "M", "「"), - (0xFF63, "M", "」"), - (0xFF64, "M", "、"), - (0xFF65, "M", "・"), - (0xFF66, "M", "ヲ"), - (0xFF67, "M", "ァ"), - (0xFF68, "M", "ィ"), - (0xFF69, "M", "ゥ"), - (0xFF6A, "M", "ェ"), - (0xFF6B, "M", "ォ"), - (0xFF6C, "M", "ャ"), - (0xFF6D, "M", "ュ"), - (0xFF6E, "M", "ョ"), - (0xFF6F, "M", "ッ"), - (0xFF70, "M", "ー"), - (0xFF71, "M", "ア"), - (0xFF72, "M", "イ"), - (0xFF73, "M", "ウ"), - (0xFF74, "M", "エ"), - (0xFF75, "M", "オ"), - (0xFF76, "M", "カ"), - (0xFF77, "M", "キ"), - (0xFF78, "M", "ク"), - (0xFF79, "M", "ケ"), - (0xFF7A, "M", "コ"), - (0xFF7B, "M", "サ"), - (0xFF7C, "M", "シ"), - (0xFF7D, "M", "ス"), - (0xFF7E, "M", "セ"), - (0xFF7F, "M", "ソ"), - (0xFF80, "M", "タ"), - (0xFF81, "M", "チ"), - (0xFF82, "M", "ツ"), - (0xFF83, "M", "テ"), - (0xFF84, "M", "ト"), - (0xFF85, "M", "ナ"), - (0xFF86, "M", "ニ"), - (0xFF87, "M", "ヌ"), - (0xFF88, "M", "ネ"), - (0xFF89, "M", "ノ"), - (0xFF8A, "M", "ハ"), - (0xFF8B, "M", "ヒ"), - (0xFF8C, "M", "フ"), - (0xFF8D, "M", "ヘ"), - (0xFF8E, "M", "ホ"), - (0xFF8F, "M", "マ"), - (0xFF90, "M", "ミ"), - (0xFF91, "M", "ム"), - (0xFF92, "M", "メ"), - (0xFF93, "M", "モ"), - (0xFF94, "M", "ヤ"), - (0xFF95, "M", "ユ"), - (0xFF96, "M", "ヨ"), - (0xFF97, "M", "ラ"), - (0xFF98, "M", "リ"), - (0xFF99, "M", "ル"), - (0xFF9A, "M", "レ"), - (0xFF9B, "M", "ロ"), - (0xFF9C, "M", "ワ"), - (0xFF9D, "M", "ン"), - (0xFF9E, "M", "゙"), - (0xFF9F, "M", "゚"), - (0xFFA0, "I"), - (0xFFA1, "M", "ᄀ"), - (0xFFA2, "M", "ᄁ"), - (0xFFA3, "M", "ᆪ"), - (0xFFA4, "M", "ᄂ"), - (0xFFA5, "M", "ᆬ"), - (0xFFA6, "M", "ᆭ"), - (0xFFA7, "M", "ᄃ"), - (0xFFA8, "M", "ᄄ"), - (0xFFA9, "M", "ᄅ"), - (0xFFAA, "M", "ᆰ"), - (0xFFAB, "M", "ᆱ"), - (0xFFAC, "M", "ᆲ"), - (0xFFAD, "M", "ᆳ"), - (0xFFAE, "M", "ᆴ"), - (0xFFAF, "M", "ᆵ"), - (0xFFB0, "M", "ᄚ"), - (0xFFB1, "M", "ᄆ"), - (0xFFB2, "M", "ᄇ"), - (0xFFB3, "M", "ᄈ"), - (0xFFB4, "M", "ᄡ"), - (0xFFB5, "M", "ᄉ"), - (0xFFB6, "M", "ᄊ"), - (0xFFB7, "M", "ᄋ"), - (0xFFB8, "M", "ᄌ"), - (0xFFB9, "M", "ᄍ"), - (0xFFBA, "M", "ᄎ"), - (0xFFBB, "M", "ᄏ"), - (0xFFBC, "M", "ᄐ"), - (0xFFBD, "M", "ᄑ"), - (0xFFBE, "M", "ᄒ"), - (0xFFBF, "X"), - ] - - -def _seg_53() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0xFFC2, "M", "ᅡ"), - (0xFFC3, "M", "ᅢ"), - (0xFFC4, "M", "ᅣ"), - (0xFFC5, "M", "ᅤ"), - (0xFFC6, "M", "ᅥ"), - (0xFFC7, "M", "ᅦ"), - (0xFFC8, "X"), - (0xFFCA, "M", "ᅧ"), - (0xFFCB, "M", "ᅨ"), - (0xFFCC, "M", "ᅩ"), - (0xFFCD, "M", "ᅪ"), - (0xFFCE, "M", "ᅫ"), - (0xFFCF, "M", "ᅬ"), - (0xFFD0, "X"), - (0xFFD2, "M", "ᅭ"), - (0xFFD3, "M", "ᅮ"), - (0xFFD4, "M", "ᅯ"), - (0xFFD5, "M", "ᅰ"), - (0xFFD6, "M", "ᅱ"), - (0xFFD7, "M", "ᅲ"), - (0xFFD8, "X"), - (0xFFDA, "M", "ᅳ"), - (0xFFDB, "M", "ᅴ"), - (0xFFDC, "M", "ᅵ"), - (0xFFDD, "X"), - (0xFFE0, "M", "¢"), - (0xFFE1, "M", "£"), - (0xFFE2, "M", "¬"), - (0xFFE3, "M", " ̄"), - (0xFFE4, "M", "¦"), - (0xFFE5, "M", "¥"), - (0xFFE6, "M", "₩"), - (0xFFE7, "X"), - (0xFFE8, "M", "│"), - (0xFFE9, "M", "←"), - (0xFFEA, "M", "↑"), - (0xFFEB, "M", "→"), - (0xFFEC, "M", "↓"), - (0xFFED, "M", "■"), - (0xFFEE, "M", "○"), - (0xFFEF, "X"), - (0x10000, "V"), - (0x1000C, "X"), - (0x1000D, "V"), - (0x10027, "X"), - (0x10028, "V"), - (0x1003B, "X"), - (0x1003C, "V"), - (0x1003E, "X"), - (0x1003F, "V"), - (0x1004E, "X"), - (0x10050, "V"), - (0x1005E, "X"), - (0x10080, "V"), - (0x100FB, "X"), - (0x10100, "V"), - (0x10103, "X"), - (0x10107, "V"), - (0x10134, "X"), - (0x10137, "V"), - (0x1018F, "X"), - (0x10190, "V"), - (0x1019D, "X"), - (0x101A0, "V"), - (0x101A1, "X"), - (0x101D0, "V"), - (0x101FE, "X"), - (0x10280, "V"), - (0x1029D, "X"), - (0x102A0, "V"), - (0x102D1, "X"), - (0x102E0, "V"), - (0x102FC, "X"), - (0x10300, "V"), - (0x10324, "X"), - (0x1032D, "V"), - (0x1034B, "X"), - (0x10350, "V"), - (0x1037B, "X"), - (0x10380, "V"), - (0x1039E, "X"), - (0x1039F, "V"), - (0x103C4, "X"), - (0x103C8, "V"), - (0x103D6, "X"), - (0x10400, "M", "𐐨"), - (0x10401, "M", "𐐩"), - (0x10402, "M", "𐐪"), - (0x10403, "M", "𐐫"), - (0x10404, "M", "𐐬"), - (0x10405, "M", "𐐭"), - (0x10406, "M", "𐐮"), - (0x10407, "M", "𐐯"), - (0x10408, "M", "𐐰"), - (0x10409, "M", "𐐱"), - (0x1040A, "M", "𐐲"), - (0x1040B, "M", "𐐳"), - (0x1040C, "M", "𐐴"), - (0x1040D, "M", "𐐵"), - (0x1040E, "M", "𐐶"), - ] - - -def _seg_54() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1040F, "M", "𐐷"), - (0x10410, "M", "𐐸"), - (0x10411, "M", "𐐹"), - (0x10412, "M", "𐐺"), - (0x10413, "M", "𐐻"), - (0x10414, "M", "𐐼"), - (0x10415, "M", "𐐽"), - (0x10416, "M", "𐐾"), - (0x10417, "M", "𐐿"), - (0x10418, "M", "𐑀"), - (0x10419, "M", "𐑁"), - (0x1041A, "M", "𐑂"), - (0x1041B, "M", "𐑃"), - (0x1041C, "M", "𐑄"), - (0x1041D, "M", "𐑅"), - (0x1041E, "M", "𐑆"), - (0x1041F, "M", "𐑇"), - (0x10420, "M", "𐑈"), - (0x10421, "M", "𐑉"), - (0x10422, "M", "𐑊"), - (0x10423, "M", "𐑋"), - (0x10424, "M", "𐑌"), - (0x10425, "M", "𐑍"), - (0x10426, "M", "𐑎"), - (0x10427, "M", "𐑏"), - (0x10428, "V"), - (0x1049E, "X"), - (0x104A0, "V"), - (0x104AA, "X"), - (0x104B0, "M", "𐓘"), - (0x104B1, "M", "𐓙"), - (0x104B2, "M", "𐓚"), - (0x104B3, "M", "𐓛"), - (0x104B4, "M", "𐓜"), - (0x104B5, "M", "𐓝"), - (0x104B6, "M", "𐓞"), - (0x104B7, "M", "𐓟"), - (0x104B8, "M", "𐓠"), - (0x104B9, "M", "𐓡"), - (0x104BA, "M", "𐓢"), - (0x104BB, "M", "𐓣"), - (0x104BC, "M", "𐓤"), - (0x104BD, "M", "𐓥"), - (0x104BE, "M", "𐓦"), - (0x104BF, "M", "𐓧"), - (0x104C0, "M", "𐓨"), - (0x104C1, "M", "𐓩"), - (0x104C2, "M", "𐓪"), - (0x104C3, "M", "𐓫"), - (0x104C4, "M", "𐓬"), - (0x104C5, "M", "𐓭"), - (0x104C6, "M", "𐓮"), - (0x104C7, "M", "𐓯"), - (0x104C8, "M", "𐓰"), - (0x104C9, "M", "𐓱"), - (0x104CA, "M", "𐓲"), - (0x104CB, "M", "𐓳"), - (0x104CC, "M", "𐓴"), - (0x104CD, "M", "𐓵"), - (0x104CE, "M", "𐓶"), - (0x104CF, "M", "𐓷"), - (0x104D0, "M", "𐓸"), - (0x104D1, "M", "𐓹"), - (0x104D2, "M", "𐓺"), - (0x104D3, "M", "𐓻"), - (0x104D4, "X"), - (0x104D8, "V"), - (0x104FC, "X"), - (0x10500, "V"), - (0x10528, "X"), - (0x10530, "V"), - (0x10564, "X"), - (0x1056F, "V"), - (0x10570, "M", "𐖗"), - (0x10571, "M", "𐖘"), - (0x10572, "M", "𐖙"), - (0x10573, "M", "𐖚"), - (0x10574, "M", "𐖛"), - (0x10575, "M", "𐖜"), - (0x10576, "M", "𐖝"), - (0x10577, "M", "𐖞"), - (0x10578, "M", "𐖟"), - (0x10579, "M", "𐖠"), - (0x1057A, "M", "𐖡"), - (0x1057B, "X"), - (0x1057C, "M", "𐖣"), - (0x1057D, "M", "𐖤"), - (0x1057E, "M", "𐖥"), - (0x1057F, "M", "𐖦"), - (0x10580, "M", "𐖧"), - (0x10581, "M", "𐖨"), - (0x10582, "M", "𐖩"), - (0x10583, "M", "𐖪"), - (0x10584, "M", "𐖫"), - (0x10585, "M", "𐖬"), - (0x10586, "M", "𐖭"), - (0x10587, "M", "𐖮"), - (0x10588, "M", "𐖯"), - (0x10589, "M", "𐖰"), - (0x1058A, "M", "𐖱"), - ] - - -def _seg_55() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1058B, "X"), - (0x1058C, "M", "𐖳"), - (0x1058D, "M", "𐖴"), - (0x1058E, "M", "𐖵"), - (0x1058F, "M", "𐖶"), - (0x10590, "M", "𐖷"), - (0x10591, "M", "𐖸"), - (0x10592, "M", "𐖹"), - (0x10593, "X"), - (0x10594, "M", "𐖻"), - (0x10595, "M", "𐖼"), - (0x10596, "X"), - (0x10597, "V"), - (0x105A2, "X"), - (0x105A3, "V"), - (0x105B2, "X"), - (0x105B3, "V"), - (0x105BA, "X"), - (0x105BB, "V"), - (0x105BD, "X"), - (0x105C0, "V"), - (0x105F4, "X"), - (0x10600, "V"), - (0x10737, "X"), - (0x10740, "V"), - (0x10756, "X"), - (0x10760, "V"), - (0x10768, "X"), - (0x10780, "V"), - (0x10781, "M", "ː"), - (0x10782, "M", "ˑ"), - (0x10783, "M", "æ"), - (0x10784, "M", "ʙ"), - (0x10785, "M", "ɓ"), - (0x10786, "X"), - (0x10787, "M", "ʣ"), - (0x10788, "M", "ꭦ"), - (0x10789, "M", "ʥ"), - (0x1078A, "M", "ʤ"), - (0x1078B, "M", "ɖ"), - (0x1078C, "M", "ɗ"), - (0x1078D, "M", "ᶑ"), - (0x1078E, "M", "ɘ"), - (0x1078F, "M", "ɞ"), - (0x10790, "M", "ʩ"), - (0x10791, "M", "ɤ"), - (0x10792, "M", "ɢ"), - (0x10793, "M", "ɠ"), - (0x10794, "M", "ʛ"), - (0x10795, "M", "ħ"), - (0x10796, "M", "ʜ"), - (0x10797, "M", "ɧ"), - (0x10798, "M", "ʄ"), - (0x10799, "M", "ʪ"), - (0x1079A, "M", "ʫ"), - (0x1079B, "M", "ɬ"), - (0x1079C, "M", "𝼄"), - (0x1079D, "M", "ꞎ"), - (0x1079E, "M", "ɮ"), - (0x1079F, "M", "𝼅"), - (0x107A0, "M", "ʎ"), - (0x107A1, "M", "𝼆"), - (0x107A2, "M", "ø"), - (0x107A3, "M", "ɶ"), - (0x107A4, "M", "ɷ"), - (0x107A5, "M", "q"), - (0x107A6, "M", "ɺ"), - (0x107A7, "M", "𝼈"), - (0x107A8, "M", "ɽ"), - (0x107A9, "M", "ɾ"), - (0x107AA, "M", "ʀ"), - (0x107AB, "M", "ʨ"), - (0x107AC, "M", "ʦ"), - (0x107AD, "M", "ꭧ"), - (0x107AE, "M", "ʧ"), - (0x107AF, "M", "ʈ"), - (0x107B0, "M", "ⱱ"), - (0x107B1, "X"), - (0x107B2, "M", "ʏ"), - (0x107B3, "M", "ʡ"), - (0x107B4, "M", "ʢ"), - (0x107B5, "M", "ʘ"), - (0x107B6, "M", "ǀ"), - (0x107B7, "M", "ǁ"), - (0x107B8, "M", "ǂ"), - (0x107B9, "M", "𝼊"), - (0x107BA, "M", "𝼞"), - (0x107BB, "X"), - (0x10800, "V"), - (0x10806, "X"), - (0x10808, "V"), - (0x10809, "X"), - (0x1080A, "V"), - (0x10836, "X"), - (0x10837, "V"), - (0x10839, "X"), - (0x1083C, "V"), - (0x1083D, "X"), - (0x1083F, "V"), - (0x10856, "X"), - ] - - -def _seg_56() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x10857, "V"), - (0x1089F, "X"), - (0x108A7, "V"), - (0x108B0, "X"), - (0x108E0, "V"), - (0x108F3, "X"), - (0x108F4, "V"), - (0x108F6, "X"), - (0x108FB, "V"), - (0x1091C, "X"), - (0x1091F, "V"), - (0x1093A, "X"), - (0x1093F, "V"), - (0x10940, "X"), - (0x10980, "V"), - (0x109B8, "X"), - (0x109BC, "V"), - (0x109D0, "X"), - (0x109D2, "V"), - (0x10A04, "X"), - (0x10A05, "V"), - (0x10A07, "X"), - (0x10A0C, "V"), - (0x10A14, "X"), - (0x10A15, "V"), - (0x10A18, "X"), - (0x10A19, "V"), - (0x10A36, "X"), - (0x10A38, "V"), - (0x10A3B, "X"), - (0x10A3F, "V"), - (0x10A49, "X"), - (0x10A50, "V"), - (0x10A59, "X"), - (0x10A60, "V"), - (0x10AA0, "X"), - (0x10AC0, "V"), - (0x10AE7, "X"), - (0x10AEB, "V"), - (0x10AF7, "X"), - (0x10B00, "V"), - (0x10B36, "X"), - (0x10B39, "V"), - (0x10B56, "X"), - (0x10B58, "V"), - (0x10B73, "X"), - (0x10B78, "V"), - (0x10B92, "X"), - (0x10B99, "V"), - (0x10B9D, "X"), - (0x10BA9, "V"), - (0x10BB0, "X"), - (0x10C00, "V"), - (0x10C49, "X"), - (0x10C80, "M", "𐳀"), - (0x10C81, "M", "𐳁"), - (0x10C82, "M", "𐳂"), - (0x10C83, "M", "𐳃"), - (0x10C84, "M", "𐳄"), - (0x10C85, "M", "𐳅"), - (0x10C86, "M", "𐳆"), - (0x10C87, "M", "𐳇"), - (0x10C88, "M", "𐳈"), - (0x10C89, "M", "𐳉"), - (0x10C8A, "M", "𐳊"), - (0x10C8B, "M", "𐳋"), - (0x10C8C, "M", "𐳌"), - (0x10C8D, "M", "𐳍"), - (0x10C8E, "M", "𐳎"), - (0x10C8F, "M", "𐳏"), - (0x10C90, "M", "𐳐"), - (0x10C91, "M", "𐳑"), - (0x10C92, "M", "𐳒"), - (0x10C93, "M", "𐳓"), - (0x10C94, "M", "𐳔"), - (0x10C95, "M", "𐳕"), - (0x10C96, "M", "𐳖"), - (0x10C97, "M", "𐳗"), - (0x10C98, "M", "𐳘"), - (0x10C99, "M", "𐳙"), - (0x10C9A, "M", "𐳚"), - (0x10C9B, "M", "𐳛"), - (0x10C9C, "M", "𐳜"), - (0x10C9D, "M", "𐳝"), - (0x10C9E, "M", "𐳞"), - (0x10C9F, "M", "𐳟"), - (0x10CA0, "M", "𐳠"), - (0x10CA1, "M", "𐳡"), - (0x10CA2, "M", "𐳢"), - (0x10CA3, "M", "𐳣"), - (0x10CA4, "M", "𐳤"), - (0x10CA5, "M", "𐳥"), - (0x10CA6, "M", "𐳦"), - (0x10CA7, "M", "𐳧"), - (0x10CA8, "M", "𐳨"), - (0x10CA9, "M", "𐳩"), - (0x10CAA, "M", "𐳪"), - (0x10CAB, "M", "𐳫"), - (0x10CAC, "M", "𐳬"), - (0x10CAD, "M", "𐳭"), - ] - - -def _seg_57() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x10CAE, "M", "𐳮"), - (0x10CAF, "M", "𐳯"), - (0x10CB0, "M", "𐳰"), - (0x10CB1, "M", "𐳱"), - (0x10CB2, "M", "𐳲"), - (0x10CB3, "X"), - (0x10CC0, "V"), - (0x10CF3, "X"), - (0x10CFA, "V"), - (0x10D28, "X"), - (0x10D30, "V"), - (0x10D3A, "X"), - (0x10D40, "V"), - (0x10D50, "M", "𐵰"), - (0x10D51, "M", "𐵱"), - (0x10D52, "M", "𐵲"), - (0x10D53, "M", "𐵳"), - (0x10D54, "M", "𐵴"), - (0x10D55, "M", "𐵵"), - (0x10D56, "M", "𐵶"), - (0x10D57, "M", "𐵷"), - (0x10D58, "M", "𐵸"), - (0x10D59, "M", "𐵹"), - (0x10D5A, "M", "𐵺"), - (0x10D5B, "M", "𐵻"), - (0x10D5C, "M", "𐵼"), - (0x10D5D, "M", "𐵽"), - (0x10D5E, "M", "𐵾"), - (0x10D5F, "M", "𐵿"), - (0x10D60, "M", "𐶀"), - (0x10D61, "M", "𐶁"), - (0x10D62, "M", "𐶂"), - (0x10D63, "M", "𐶃"), - (0x10D64, "M", "𐶄"), - (0x10D65, "M", "𐶅"), - (0x10D66, "X"), - (0x10D69, "V"), - (0x10D86, "X"), - (0x10D8E, "V"), - (0x10D90, "X"), - (0x10E60, "V"), - (0x10E7F, "X"), - (0x10E80, "V"), - (0x10EAA, "X"), - (0x10EAB, "V"), - (0x10EAE, "X"), - (0x10EB0, "V"), - (0x10EB2, "X"), - (0x10EC2, "V"), - (0x10EC5, "X"), - (0x10EFC, "V"), - (0x10F28, "X"), - (0x10F30, "V"), - (0x10F5A, "X"), - (0x10F70, "V"), - (0x10F8A, "X"), - (0x10FB0, "V"), - (0x10FCC, "X"), - (0x10FE0, "V"), - (0x10FF7, "X"), - (0x11000, "V"), - (0x1104E, "X"), - (0x11052, "V"), - (0x11076, "X"), - (0x1107F, "V"), - (0x110BD, "X"), - (0x110BE, "V"), - (0x110C3, "X"), - (0x110D0, "V"), - (0x110E9, "X"), - (0x110F0, "V"), - (0x110FA, "X"), - (0x11100, "V"), - (0x11135, "X"), - (0x11136, "V"), - (0x11148, "X"), - (0x11150, "V"), - (0x11177, "X"), - (0x11180, "V"), - (0x111E0, "X"), - (0x111E1, "V"), - (0x111F5, "X"), - (0x11200, "V"), - (0x11212, "X"), - (0x11213, "V"), - (0x11242, "X"), - (0x11280, "V"), - (0x11287, "X"), - (0x11288, "V"), - (0x11289, "X"), - (0x1128A, "V"), - (0x1128E, "X"), - (0x1128F, "V"), - (0x1129E, "X"), - (0x1129F, "V"), - (0x112AA, "X"), - (0x112B0, "V"), - (0x112EB, "X"), - (0x112F0, "V"), - (0x112FA, "X"), - ] - - -def _seg_58() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x11300, "V"), - (0x11304, "X"), - (0x11305, "V"), - (0x1130D, "X"), - (0x1130F, "V"), - (0x11311, "X"), - (0x11313, "V"), - (0x11329, "X"), - (0x1132A, "V"), - (0x11331, "X"), - (0x11332, "V"), - (0x11334, "X"), - (0x11335, "V"), - (0x1133A, "X"), - (0x1133B, "V"), - (0x11345, "X"), - (0x11347, "V"), - (0x11349, "X"), - (0x1134B, "V"), - (0x1134E, "X"), - (0x11350, "V"), - (0x11351, "X"), - (0x11357, "V"), - (0x11358, "X"), - (0x1135D, "V"), - (0x11364, "X"), - (0x11366, "V"), - (0x1136D, "X"), - (0x11370, "V"), - (0x11375, "X"), - (0x11380, "V"), - (0x1138A, "X"), - (0x1138B, "V"), - (0x1138C, "X"), - (0x1138E, "V"), - (0x1138F, "X"), - (0x11390, "V"), - (0x113B6, "X"), - (0x113B7, "V"), - (0x113C1, "X"), - (0x113C2, "V"), - (0x113C3, "X"), - (0x113C5, "V"), - (0x113C6, "X"), - (0x113C7, "V"), - (0x113CB, "X"), - (0x113CC, "V"), - (0x113D6, "X"), - (0x113D7, "V"), - (0x113D9, "X"), - (0x113E1, "V"), - (0x113E3, "X"), - (0x11400, "V"), - (0x1145C, "X"), - (0x1145D, "V"), - (0x11462, "X"), - (0x11480, "V"), - (0x114C8, "X"), - (0x114D0, "V"), - (0x114DA, "X"), - (0x11580, "V"), - (0x115B6, "X"), - (0x115B8, "V"), - (0x115DE, "X"), - (0x11600, "V"), - (0x11645, "X"), - (0x11650, "V"), - (0x1165A, "X"), - (0x11660, "V"), - (0x1166D, "X"), - (0x11680, "V"), - (0x116BA, "X"), - (0x116C0, "V"), - (0x116CA, "X"), - (0x116D0, "V"), - (0x116E4, "X"), - (0x11700, "V"), - (0x1171B, "X"), - (0x1171D, "V"), - (0x1172C, "X"), - (0x11730, "V"), - (0x11747, "X"), - (0x11800, "V"), - (0x1183C, "X"), - (0x118A0, "M", "𑣀"), - (0x118A1, "M", "𑣁"), - (0x118A2, "M", "𑣂"), - (0x118A3, "M", "𑣃"), - (0x118A4, "M", "𑣄"), - (0x118A5, "M", "𑣅"), - (0x118A6, "M", "𑣆"), - (0x118A7, "M", "𑣇"), - (0x118A8, "M", "𑣈"), - (0x118A9, "M", "𑣉"), - (0x118AA, "M", "𑣊"), - (0x118AB, "M", "𑣋"), - (0x118AC, "M", "𑣌"), - (0x118AD, "M", "𑣍"), - (0x118AE, "M", "𑣎"), - (0x118AF, "M", "𑣏"), - ] - - -def _seg_59() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x118B0, "M", "𑣐"), - (0x118B1, "M", "𑣑"), - (0x118B2, "M", "𑣒"), - (0x118B3, "M", "𑣓"), - (0x118B4, "M", "𑣔"), - (0x118B5, "M", "𑣕"), - (0x118B6, "M", "𑣖"), - (0x118B7, "M", "𑣗"), - (0x118B8, "M", "𑣘"), - (0x118B9, "M", "𑣙"), - (0x118BA, "M", "𑣚"), - (0x118BB, "M", "𑣛"), - (0x118BC, "M", "𑣜"), - (0x118BD, "M", "𑣝"), - (0x118BE, "M", "𑣞"), - (0x118BF, "M", "𑣟"), - (0x118C0, "V"), - (0x118F3, "X"), - (0x118FF, "V"), - (0x11907, "X"), - (0x11909, "V"), - (0x1190A, "X"), - (0x1190C, "V"), - (0x11914, "X"), - (0x11915, "V"), - (0x11917, "X"), - (0x11918, "V"), - (0x11936, "X"), - (0x11937, "V"), - (0x11939, "X"), - (0x1193B, "V"), - (0x11947, "X"), - (0x11950, "V"), - (0x1195A, "X"), - (0x119A0, "V"), - (0x119A8, "X"), - (0x119AA, "V"), - (0x119D8, "X"), - (0x119DA, "V"), - (0x119E5, "X"), - (0x11A00, "V"), - (0x11A48, "X"), - (0x11A50, "V"), - (0x11AA3, "X"), - (0x11AB0, "V"), - (0x11AF9, "X"), - (0x11B00, "V"), - (0x11B0A, "X"), - (0x11BC0, "V"), - (0x11BE2, "X"), - (0x11BF0, "V"), - (0x11BFA, "X"), - (0x11C00, "V"), - (0x11C09, "X"), - (0x11C0A, "V"), - (0x11C37, "X"), - (0x11C38, "V"), - (0x11C46, "X"), - (0x11C50, "V"), - (0x11C6D, "X"), - (0x11C70, "V"), - (0x11C90, "X"), - (0x11C92, "V"), - (0x11CA8, "X"), - (0x11CA9, "V"), - (0x11CB7, "X"), - (0x11D00, "V"), - (0x11D07, "X"), - (0x11D08, "V"), - (0x11D0A, "X"), - (0x11D0B, "V"), - (0x11D37, "X"), - (0x11D3A, "V"), - (0x11D3B, "X"), - (0x11D3C, "V"), - (0x11D3E, "X"), - (0x11D3F, "V"), - (0x11D48, "X"), - (0x11D50, "V"), - (0x11D5A, "X"), - (0x11D60, "V"), - (0x11D66, "X"), - (0x11D67, "V"), - (0x11D69, "X"), - (0x11D6A, "V"), - (0x11D8F, "X"), - (0x11D90, "V"), - (0x11D92, "X"), - (0x11D93, "V"), - (0x11D99, "X"), - (0x11DA0, "V"), - (0x11DAA, "X"), - (0x11EE0, "V"), - (0x11EF9, "X"), - (0x11F00, "V"), - (0x11F11, "X"), - (0x11F12, "V"), - (0x11F3B, "X"), - (0x11F3E, "V"), - (0x11F5B, "X"), - ] - - -def _seg_60() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x11FB0, "V"), - (0x11FB1, "X"), - (0x11FC0, "V"), - (0x11FF2, "X"), - (0x11FFF, "V"), - (0x1239A, "X"), - (0x12400, "V"), - (0x1246F, "X"), - (0x12470, "V"), - (0x12475, "X"), - (0x12480, "V"), - (0x12544, "X"), - (0x12F90, "V"), - (0x12FF3, "X"), - (0x13000, "V"), - (0x13430, "X"), - (0x13440, "V"), - (0x13456, "X"), - (0x13460, "V"), - (0x143FB, "X"), - (0x14400, "V"), - (0x14647, "X"), - (0x16100, "V"), - (0x1613A, "X"), - (0x16800, "V"), - (0x16A39, "X"), - (0x16A40, "V"), - (0x16A5F, "X"), - (0x16A60, "V"), - (0x16A6A, "X"), - (0x16A6E, "V"), - (0x16ABF, "X"), - (0x16AC0, "V"), - (0x16ACA, "X"), - (0x16AD0, "V"), - (0x16AEE, "X"), - (0x16AF0, "V"), - (0x16AF6, "X"), - (0x16B00, "V"), - (0x16B46, "X"), - (0x16B50, "V"), - (0x16B5A, "X"), - (0x16B5B, "V"), - (0x16B62, "X"), - (0x16B63, "V"), - (0x16B78, "X"), - (0x16B7D, "V"), - (0x16B90, "X"), - (0x16D40, "V"), - (0x16D7A, "X"), - (0x16E40, "M", "𖹠"), - (0x16E41, "M", "𖹡"), - (0x16E42, "M", "𖹢"), - (0x16E43, "M", "𖹣"), - (0x16E44, "M", "𖹤"), - (0x16E45, "M", "𖹥"), - (0x16E46, "M", "𖹦"), - (0x16E47, "M", "𖹧"), - (0x16E48, "M", "𖹨"), - (0x16E49, "M", "𖹩"), - (0x16E4A, "M", "𖹪"), - (0x16E4B, "M", "𖹫"), - (0x16E4C, "M", "𖹬"), - (0x16E4D, "M", "𖹭"), - (0x16E4E, "M", "𖹮"), - (0x16E4F, "M", "𖹯"), - (0x16E50, "M", "𖹰"), - (0x16E51, "M", "𖹱"), - (0x16E52, "M", "𖹲"), - (0x16E53, "M", "𖹳"), - (0x16E54, "M", "𖹴"), - (0x16E55, "M", "𖹵"), - (0x16E56, "M", "𖹶"), - (0x16E57, "M", "𖹷"), - (0x16E58, "M", "𖹸"), - (0x16E59, "M", "𖹹"), - (0x16E5A, "M", "𖹺"), - (0x16E5B, "M", "𖹻"), - (0x16E5C, "M", "𖹼"), - (0x16E5D, "M", "𖹽"), - (0x16E5E, "M", "𖹾"), - (0x16E5F, "M", "𖹿"), - (0x16E60, "V"), - (0x16E9B, "X"), - (0x16F00, "V"), - (0x16F4B, "X"), - (0x16F4F, "V"), - (0x16F88, "X"), - (0x16F8F, "V"), - (0x16FA0, "X"), - (0x16FE0, "V"), - (0x16FE5, "X"), - (0x16FF0, "V"), - (0x16FF2, "X"), - (0x17000, "V"), - (0x187F8, "X"), - (0x18800, "V"), - (0x18CD6, "X"), - (0x18CFF, "V"), - (0x18D09, "X"), - ] - - -def _seg_61() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1AFF0, "V"), - (0x1AFF4, "X"), - (0x1AFF5, "V"), - (0x1AFFC, "X"), - (0x1AFFD, "V"), - (0x1AFFF, "X"), - (0x1B000, "V"), - (0x1B123, "X"), - (0x1B132, "V"), - (0x1B133, "X"), - (0x1B150, "V"), - (0x1B153, "X"), - (0x1B155, "V"), - (0x1B156, "X"), - (0x1B164, "V"), - (0x1B168, "X"), - (0x1B170, "V"), - (0x1B2FC, "X"), - (0x1BC00, "V"), - (0x1BC6B, "X"), - (0x1BC70, "V"), - (0x1BC7D, "X"), - (0x1BC80, "V"), - (0x1BC89, "X"), - (0x1BC90, "V"), - (0x1BC9A, "X"), - (0x1BC9C, "V"), - (0x1BCA0, "I"), - (0x1BCA4, "X"), - (0x1CC00, "V"), - (0x1CCD6, "M", "a"), - (0x1CCD7, "M", "b"), - (0x1CCD8, "M", "c"), - (0x1CCD9, "M", "d"), - (0x1CCDA, "M", "e"), - (0x1CCDB, "M", "f"), - (0x1CCDC, "M", "g"), - (0x1CCDD, "M", "h"), - (0x1CCDE, "M", "i"), - (0x1CCDF, "M", "j"), - (0x1CCE0, "M", "k"), - (0x1CCE1, "M", "l"), - (0x1CCE2, "M", "m"), - (0x1CCE3, "M", "n"), - (0x1CCE4, "M", "o"), - (0x1CCE5, "M", "p"), - (0x1CCE6, "M", "q"), - (0x1CCE7, "M", "r"), - (0x1CCE8, "M", "s"), - (0x1CCE9, "M", "t"), - (0x1CCEA, "M", "u"), - (0x1CCEB, "M", "v"), - (0x1CCEC, "M", "w"), - (0x1CCED, "M", "x"), - (0x1CCEE, "M", "y"), - (0x1CCEF, "M", "z"), - (0x1CCF0, "M", "0"), - (0x1CCF1, "M", "1"), - (0x1CCF2, "M", "2"), - (0x1CCF3, "M", "3"), - (0x1CCF4, "M", "4"), - (0x1CCF5, "M", "5"), - (0x1CCF6, "M", "6"), - (0x1CCF7, "M", "7"), - (0x1CCF8, "M", "8"), - (0x1CCF9, "M", "9"), - (0x1CCFA, "X"), - (0x1CD00, "V"), - (0x1CEB4, "X"), - (0x1CF00, "V"), - (0x1CF2E, "X"), - (0x1CF30, "V"), - (0x1CF47, "X"), - (0x1CF50, "V"), - (0x1CFC4, "X"), - (0x1D000, "V"), - (0x1D0F6, "X"), - (0x1D100, "V"), - (0x1D127, "X"), - (0x1D129, "V"), - (0x1D15E, "M", "𝅗𝅥"), - (0x1D15F, "M", "𝅘𝅥"), - (0x1D160, "M", "𝅘𝅥𝅮"), - (0x1D161, "M", "𝅘𝅥𝅯"), - (0x1D162, "M", "𝅘𝅥𝅰"), - (0x1D163, "M", "𝅘𝅥𝅱"), - (0x1D164, "M", "𝅘𝅥𝅲"), - (0x1D165, "V"), - (0x1D173, "I"), - (0x1D17B, "V"), - (0x1D1BB, "M", "𝆹𝅥"), - (0x1D1BC, "M", "𝆺𝅥"), - (0x1D1BD, "M", "𝆹𝅥𝅮"), - (0x1D1BE, "M", "𝆺𝅥𝅮"), - (0x1D1BF, "M", "𝆹𝅥𝅯"), - (0x1D1C0, "M", "𝆺𝅥𝅯"), - (0x1D1C1, "V"), - (0x1D1EB, "X"), - (0x1D200, "V"), - (0x1D246, "X"), - ] - - -def _seg_62() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D2C0, "V"), - (0x1D2D4, "X"), - (0x1D2E0, "V"), - (0x1D2F4, "X"), - (0x1D300, "V"), - (0x1D357, "X"), - (0x1D360, "V"), - (0x1D379, "X"), - (0x1D400, "M", "a"), - (0x1D401, "M", "b"), - (0x1D402, "M", "c"), - (0x1D403, "M", "d"), - (0x1D404, "M", "e"), - (0x1D405, "M", "f"), - (0x1D406, "M", "g"), - (0x1D407, "M", "h"), - (0x1D408, "M", "i"), - (0x1D409, "M", "j"), - (0x1D40A, "M", "k"), - (0x1D40B, "M", "l"), - (0x1D40C, "M", "m"), - (0x1D40D, "M", "n"), - (0x1D40E, "M", "o"), - (0x1D40F, "M", "p"), - (0x1D410, "M", "q"), - (0x1D411, "M", "r"), - (0x1D412, "M", "s"), - (0x1D413, "M", "t"), - (0x1D414, "M", "u"), - (0x1D415, "M", "v"), - (0x1D416, "M", "w"), - (0x1D417, "M", "x"), - (0x1D418, "M", "y"), - (0x1D419, "M", "z"), - (0x1D41A, "M", "a"), - (0x1D41B, "M", "b"), - (0x1D41C, "M", "c"), - (0x1D41D, "M", "d"), - (0x1D41E, "M", "e"), - (0x1D41F, "M", "f"), - (0x1D420, "M", "g"), - (0x1D421, "M", "h"), - (0x1D422, "M", "i"), - (0x1D423, "M", "j"), - (0x1D424, "M", "k"), - (0x1D425, "M", "l"), - (0x1D426, "M", "m"), - (0x1D427, "M", "n"), - (0x1D428, "M", "o"), - (0x1D429, "M", "p"), - (0x1D42A, "M", "q"), - (0x1D42B, "M", "r"), - (0x1D42C, "M", "s"), - (0x1D42D, "M", "t"), - (0x1D42E, "M", "u"), - (0x1D42F, "M", "v"), - (0x1D430, "M", "w"), - (0x1D431, "M", "x"), - (0x1D432, "M", "y"), - (0x1D433, "M", "z"), - (0x1D434, "M", "a"), - (0x1D435, "M", "b"), - (0x1D436, "M", "c"), - (0x1D437, "M", "d"), - (0x1D438, "M", "e"), - (0x1D439, "M", "f"), - (0x1D43A, "M", "g"), - (0x1D43B, "M", "h"), - (0x1D43C, "M", "i"), - (0x1D43D, "M", "j"), - (0x1D43E, "M", "k"), - (0x1D43F, "M", "l"), - (0x1D440, "M", "m"), - (0x1D441, "M", "n"), - (0x1D442, "M", "o"), - (0x1D443, "M", "p"), - (0x1D444, "M", "q"), - (0x1D445, "M", "r"), - (0x1D446, "M", "s"), - (0x1D447, "M", "t"), - (0x1D448, "M", "u"), - (0x1D449, "M", "v"), - (0x1D44A, "M", "w"), - (0x1D44B, "M", "x"), - (0x1D44C, "M", "y"), - (0x1D44D, "M", "z"), - (0x1D44E, "M", "a"), - (0x1D44F, "M", "b"), - (0x1D450, "M", "c"), - (0x1D451, "M", "d"), - (0x1D452, "M", "e"), - (0x1D453, "M", "f"), - (0x1D454, "M", "g"), - (0x1D455, "X"), - (0x1D456, "M", "i"), - (0x1D457, "M", "j"), - (0x1D458, "M", "k"), - (0x1D459, "M", "l"), - (0x1D45A, "M", "m"), - (0x1D45B, "M", "n"), - ] - - -def _seg_63() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D45C, "M", "o"), - (0x1D45D, "M", "p"), - (0x1D45E, "M", "q"), - (0x1D45F, "M", "r"), - (0x1D460, "M", "s"), - (0x1D461, "M", "t"), - (0x1D462, "M", "u"), - (0x1D463, "M", "v"), - (0x1D464, "M", "w"), - (0x1D465, "M", "x"), - (0x1D466, "M", "y"), - (0x1D467, "M", "z"), - (0x1D468, "M", "a"), - (0x1D469, "M", "b"), - (0x1D46A, "M", "c"), - (0x1D46B, "M", "d"), - (0x1D46C, "M", "e"), - (0x1D46D, "M", "f"), - (0x1D46E, "M", "g"), - (0x1D46F, "M", "h"), - (0x1D470, "M", "i"), - (0x1D471, "M", "j"), - (0x1D472, "M", "k"), - (0x1D473, "M", "l"), - (0x1D474, "M", "m"), - (0x1D475, "M", "n"), - (0x1D476, "M", "o"), - (0x1D477, "M", "p"), - (0x1D478, "M", "q"), - (0x1D479, "M", "r"), - (0x1D47A, "M", "s"), - (0x1D47B, "M", "t"), - (0x1D47C, "M", "u"), - (0x1D47D, "M", "v"), - (0x1D47E, "M", "w"), - (0x1D47F, "M", "x"), - (0x1D480, "M", "y"), - (0x1D481, "M", "z"), - (0x1D482, "M", "a"), - (0x1D483, "M", "b"), - (0x1D484, "M", "c"), - (0x1D485, "M", "d"), - (0x1D486, "M", "e"), - (0x1D487, "M", "f"), - (0x1D488, "M", "g"), - (0x1D489, "M", "h"), - (0x1D48A, "M", "i"), - (0x1D48B, "M", "j"), - (0x1D48C, "M", "k"), - (0x1D48D, "M", "l"), - (0x1D48E, "M", "m"), - (0x1D48F, "M", "n"), - (0x1D490, "M", "o"), - (0x1D491, "M", "p"), - (0x1D492, "M", "q"), - (0x1D493, "M", "r"), - (0x1D494, "M", "s"), - (0x1D495, "M", "t"), - (0x1D496, "M", "u"), - (0x1D497, "M", "v"), - (0x1D498, "M", "w"), - (0x1D499, "M", "x"), - (0x1D49A, "M", "y"), - (0x1D49B, "M", "z"), - (0x1D49C, "M", "a"), - (0x1D49D, "X"), - (0x1D49E, "M", "c"), - (0x1D49F, "M", "d"), - (0x1D4A0, "X"), - (0x1D4A2, "M", "g"), - (0x1D4A3, "X"), - (0x1D4A5, "M", "j"), - (0x1D4A6, "M", "k"), - (0x1D4A7, "X"), - (0x1D4A9, "M", "n"), - (0x1D4AA, "M", "o"), - (0x1D4AB, "M", "p"), - (0x1D4AC, "M", "q"), - (0x1D4AD, "X"), - (0x1D4AE, "M", "s"), - (0x1D4AF, "M", "t"), - (0x1D4B0, "M", "u"), - (0x1D4B1, "M", "v"), - (0x1D4B2, "M", "w"), - (0x1D4B3, "M", "x"), - (0x1D4B4, "M", "y"), - (0x1D4B5, "M", "z"), - (0x1D4B6, "M", "a"), - (0x1D4B7, "M", "b"), - (0x1D4B8, "M", "c"), - (0x1D4B9, "M", "d"), - (0x1D4BA, "X"), - (0x1D4BB, "M", "f"), - (0x1D4BC, "X"), - (0x1D4BD, "M", "h"), - (0x1D4BE, "M", "i"), - (0x1D4BF, "M", "j"), - (0x1D4C0, "M", "k"), - (0x1D4C1, "M", "l"), - (0x1D4C2, "M", "m"), - ] - - -def _seg_64() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D4C3, "M", "n"), - (0x1D4C4, "X"), - (0x1D4C5, "M", "p"), - (0x1D4C6, "M", "q"), - (0x1D4C7, "M", "r"), - (0x1D4C8, "M", "s"), - (0x1D4C9, "M", "t"), - (0x1D4CA, "M", "u"), - (0x1D4CB, "M", "v"), - (0x1D4CC, "M", "w"), - (0x1D4CD, "M", "x"), - (0x1D4CE, "M", "y"), - (0x1D4CF, "M", "z"), - (0x1D4D0, "M", "a"), - (0x1D4D1, "M", "b"), - (0x1D4D2, "M", "c"), - (0x1D4D3, "M", "d"), - (0x1D4D4, "M", "e"), - (0x1D4D5, "M", "f"), - (0x1D4D6, "M", "g"), - (0x1D4D7, "M", "h"), - (0x1D4D8, "M", "i"), - (0x1D4D9, "M", "j"), - (0x1D4DA, "M", "k"), - (0x1D4DB, "M", "l"), - (0x1D4DC, "M", "m"), - (0x1D4DD, "M", "n"), - (0x1D4DE, "M", "o"), - (0x1D4DF, "M", "p"), - (0x1D4E0, "M", "q"), - (0x1D4E1, "M", "r"), - (0x1D4E2, "M", "s"), - (0x1D4E3, "M", "t"), - (0x1D4E4, "M", "u"), - (0x1D4E5, "M", "v"), - (0x1D4E6, "M", "w"), - (0x1D4E7, "M", "x"), - (0x1D4E8, "M", "y"), - (0x1D4E9, "M", "z"), - (0x1D4EA, "M", "a"), - (0x1D4EB, "M", "b"), - (0x1D4EC, "M", "c"), - (0x1D4ED, "M", "d"), - (0x1D4EE, "M", "e"), - (0x1D4EF, "M", "f"), - (0x1D4F0, "M", "g"), - (0x1D4F1, "M", "h"), - (0x1D4F2, "M", "i"), - (0x1D4F3, "M", "j"), - (0x1D4F4, "M", "k"), - (0x1D4F5, "M", "l"), - (0x1D4F6, "M", "m"), - (0x1D4F7, "M", "n"), - (0x1D4F8, "M", "o"), - (0x1D4F9, "M", "p"), - (0x1D4FA, "M", "q"), - (0x1D4FB, "M", "r"), - (0x1D4FC, "M", "s"), - (0x1D4FD, "M", "t"), - (0x1D4FE, "M", "u"), - (0x1D4FF, "M", "v"), - (0x1D500, "M", "w"), - (0x1D501, "M", "x"), - (0x1D502, "M", "y"), - (0x1D503, "M", "z"), - (0x1D504, "M", "a"), - (0x1D505, "M", "b"), - (0x1D506, "X"), - (0x1D507, "M", "d"), - (0x1D508, "M", "e"), - (0x1D509, "M", "f"), - (0x1D50A, "M", "g"), - (0x1D50B, "X"), - (0x1D50D, "M", "j"), - (0x1D50E, "M", "k"), - (0x1D50F, "M", "l"), - (0x1D510, "M", "m"), - (0x1D511, "M", "n"), - (0x1D512, "M", "o"), - (0x1D513, "M", "p"), - (0x1D514, "M", "q"), - (0x1D515, "X"), - (0x1D516, "M", "s"), - (0x1D517, "M", "t"), - (0x1D518, "M", "u"), - (0x1D519, "M", "v"), - (0x1D51A, "M", "w"), - (0x1D51B, "M", "x"), - (0x1D51C, "M", "y"), - (0x1D51D, "X"), - (0x1D51E, "M", "a"), - (0x1D51F, "M", "b"), - (0x1D520, "M", "c"), - (0x1D521, "M", "d"), - (0x1D522, "M", "e"), - (0x1D523, "M", "f"), - (0x1D524, "M", "g"), - (0x1D525, "M", "h"), - (0x1D526, "M", "i"), - (0x1D527, "M", "j"), - ] - - -def _seg_65() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D528, "M", "k"), - (0x1D529, "M", "l"), - (0x1D52A, "M", "m"), - (0x1D52B, "M", "n"), - (0x1D52C, "M", "o"), - (0x1D52D, "M", "p"), - (0x1D52E, "M", "q"), - (0x1D52F, "M", "r"), - (0x1D530, "M", "s"), - (0x1D531, "M", "t"), - (0x1D532, "M", "u"), - (0x1D533, "M", "v"), - (0x1D534, "M", "w"), - (0x1D535, "M", "x"), - (0x1D536, "M", "y"), - (0x1D537, "M", "z"), - (0x1D538, "M", "a"), - (0x1D539, "M", "b"), - (0x1D53A, "X"), - (0x1D53B, "M", "d"), - (0x1D53C, "M", "e"), - (0x1D53D, "M", "f"), - (0x1D53E, "M", "g"), - (0x1D53F, "X"), - (0x1D540, "M", "i"), - (0x1D541, "M", "j"), - (0x1D542, "M", "k"), - (0x1D543, "M", "l"), - (0x1D544, "M", "m"), - (0x1D545, "X"), - (0x1D546, "M", "o"), - (0x1D547, "X"), - (0x1D54A, "M", "s"), - (0x1D54B, "M", "t"), - (0x1D54C, "M", "u"), - (0x1D54D, "M", "v"), - (0x1D54E, "M", "w"), - (0x1D54F, "M", "x"), - (0x1D550, "M", "y"), - (0x1D551, "X"), - (0x1D552, "M", "a"), - (0x1D553, "M", "b"), - (0x1D554, "M", "c"), - (0x1D555, "M", "d"), - (0x1D556, "M", "e"), - (0x1D557, "M", "f"), - (0x1D558, "M", "g"), - (0x1D559, "M", "h"), - (0x1D55A, "M", "i"), - (0x1D55B, "M", "j"), - (0x1D55C, "M", "k"), - (0x1D55D, "M", "l"), - (0x1D55E, "M", "m"), - (0x1D55F, "M", "n"), - (0x1D560, "M", "o"), - (0x1D561, "M", "p"), - (0x1D562, "M", "q"), - (0x1D563, "M", "r"), - (0x1D564, "M", "s"), - (0x1D565, "M", "t"), - (0x1D566, "M", "u"), - (0x1D567, "M", "v"), - (0x1D568, "M", "w"), - (0x1D569, "M", "x"), - (0x1D56A, "M", "y"), - (0x1D56B, "M", "z"), - (0x1D56C, "M", "a"), - (0x1D56D, "M", "b"), - (0x1D56E, "M", "c"), - (0x1D56F, "M", "d"), - (0x1D570, "M", "e"), - (0x1D571, "M", "f"), - (0x1D572, "M", "g"), - (0x1D573, "M", "h"), - (0x1D574, "M", "i"), - (0x1D575, "M", "j"), - (0x1D576, "M", "k"), - (0x1D577, "M", "l"), - (0x1D578, "M", "m"), - (0x1D579, "M", "n"), - (0x1D57A, "M", "o"), - (0x1D57B, "M", "p"), - (0x1D57C, "M", "q"), - (0x1D57D, "M", "r"), - (0x1D57E, "M", "s"), - (0x1D57F, "M", "t"), - (0x1D580, "M", "u"), - (0x1D581, "M", "v"), - (0x1D582, "M", "w"), - (0x1D583, "M", "x"), - (0x1D584, "M", "y"), - (0x1D585, "M", "z"), - (0x1D586, "M", "a"), - (0x1D587, "M", "b"), - (0x1D588, "M", "c"), - (0x1D589, "M", "d"), - (0x1D58A, "M", "e"), - (0x1D58B, "M", "f"), - (0x1D58C, "M", "g"), - (0x1D58D, "M", "h"), - ] - - -def _seg_66() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D58E, "M", "i"), - (0x1D58F, "M", "j"), - (0x1D590, "M", "k"), - (0x1D591, "M", "l"), - (0x1D592, "M", "m"), - (0x1D593, "M", "n"), - (0x1D594, "M", "o"), - (0x1D595, "M", "p"), - (0x1D596, "M", "q"), - (0x1D597, "M", "r"), - (0x1D598, "M", "s"), - (0x1D599, "M", "t"), - (0x1D59A, "M", "u"), - (0x1D59B, "M", "v"), - (0x1D59C, "M", "w"), - (0x1D59D, "M", "x"), - (0x1D59E, "M", "y"), - (0x1D59F, "M", "z"), - (0x1D5A0, "M", "a"), - (0x1D5A1, "M", "b"), - (0x1D5A2, "M", "c"), - (0x1D5A3, "M", "d"), - (0x1D5A4, "M", "e"), - (0x1D5A5, "M", "f"), - (0x1D5A6, "M", "g"), - (0x1D5A7, "M", "h"), - (0x1D5A8, "M", "i"), - (0x1D5A9, "M", "j"), - (0x1D5AA, "M", "k"), - (0x1D5AB, "M", "l"), - (0x1D5AC, "M", "m"), - (0x1D5AD, "M", "n"), - (0x1D5AE, "M", "o"), - (0x1D5AF, "M", "p"), - (0x1D5B0, "M", "q"), - (0x1D5B1, "M", "r"), - (0x1D5B2, "M", "s"), - (0x1D5B3, "M", "t"), - (0x1D5B4, "M", "u"), - (0x1D5B5, "M", "v"), - (0x1D5B6, "M", "w"), - (0x1D5B7, "M", "x"), - (0x1D5B8, "M", "y"), - (0x1D5B9, "M", "z"), - (0x1D5BA, "M", "a"), - (0x1D5BB, "M", "b"), - (0x1D5BC, "M", "c"), - (0x1D5BD, "M", "d"), - (0x1D5BE, "M", "e"), - (0x1D5BF, "M", "f"), - (0x1D5C0, "M", "g"), - (0x1D5C1, "M", "h"), - (0x1D5C2, "M", "i"), - (0x1D5C3, "M", "j"), - (0x1D5C4, "M", "k"), - (0x1D5C5, "M", "l"), - (0x1D5C6, "M", "m"), - (0x1D5C7, "M", "n"), - (0x1D5C8, "M", "o"), - (0x1D5C9, "M", "p"), - (0x1D5CA, "M", "q"), - (0x1D5CB, "M", "r"), - (0x1D5CC, "M", "s"), - (0x1D5CD, "M", "t"), - (0x1D5CE, "M", "u"), - (0x1D5CF, "M", "v"), - (0x1D5D0, "M", "w"), - (0x1D5D1, "M", "x"), - (0x1D5D2, "M", "y"), - (0x1D5D3, "M", "z"), - (0x1D5D4, "M", "a"), - (0x1D5D5, "M", "b"), - (0x1D5D6, "M", "c"), - (0x1D5D7, "M", "d"), - (0x1D5D8, "M", "e"), - (0x1D5D9, "M", "f"), - (0x1D5DA, "M", "g"), - (0x1D5DB, "M", "h"), - (0x1D5DC, "M", "i"), - (0x1D5DD, "M", "j"), - (0x1D5DE, "M", "k"), - (0x1D5DF, "M", "l"), - (0x1D5E0, "M", "m"), - (0x1D5E1, "M", "n"), - (0x1D5E2, "M", "o"), - (0x1D5E3, "M", "p"), - (0x1D5E4, "M", "q"), - (0x1D5E5, "M", "r"), - (0x1D5E6, "M", "s"), - (0x1D5E7, "M", "t"), - (0x1D5E8, "M", "u"), - (0x1D5E9, "M", "v"), - (0x1D5EA, "M", "w"), - (0x1D5EB, "M", "x"), - (0x1D5EC, "M", "y"), - (0x1D5ED, "M", "z"), - (0x1D5EE, "M", "a"), - (0x1D5EF, "M", "b"), - (0x1D5F0, "M", "c"), - (0x1D5F1, "M", "d"), - ] - - -def _seg_67() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D5F2, "M", "e"), - (0x1D5F3, "M", "f"), - (0x1D5F4, "M", "g"), - (0x1D5F5, "M", "h"), - (0x1D5F6, "M", "i"), - (0x1D5F7, "M", "j"), - (0x1D5F8, "M", "k"), - (0x1D5F9, "M", "l"), - (0x1D5FA, "M", "m"), - (0x1D5FB, "M", "n"), - (0x1D5FC, "M", "o"), - (0x1D5FD, "M", "p"), - (0x1D5FE, "M", "q"), - (0x1D5FF, "M", "r"), - (0x1D600, "M", "s"), - (0x1D601, "M", "t"), - (0x1D602, "M", "u"), - (0x1D603, "M", "v"), - (0x1D604, "M", "w"), - (0x1D605, "M", "x"), - (0x1D606, "M", "y"), - (0x1D607, "M", "z"), - (0x1D608, "M", "a"), - (0x1D609, "M", "b"), - (0x1D60A, "M", "c"), - (0x1D60B, "M", "d"), - (0x1D60C, "M", "e"), - (0x1D60D, "M", "f"), - (0x1D60E, "M", "g"), - (0x1D60F, "M", "h"), - (0x1D610, "M", "i"), - (0x1D611, "M", "j"), - (0x1D612, "M", "k"), - (0x1D613, "M", "l"), - (0x1D614, "M", "m"), - (0x1D615, "M", "n"), - (0x1D616, "M", "o"), - (0x1D617, "M", "p"), - (0x1D618, "M", "q"), - (0x1D619, "M", "r"), - (0x1D61A, "M", "s"), - (0x1D61B, "M", "t"), - (0x1D61C, "M", "u"), - (0x1D61D, "M", "v"), - (0x1D61E, "M", "w"), - (0x1D61F, "M", "x"), - (0x1D620, "M", "y"), - (0x1D621, "M", "z"), - (0x1D622, "M", "a"), - (0x1D623, "M", "b"), - (0x1D624, "M", "c"), - (0x1D625, "M", "d"), - (0x1D626, "M", "e"), - (0x1D627, "M", "f"), - (0x1D628, "M", "g"), - (0x1D629, "M", "h"), - (0x1D62A, "M", "i"), - (0x1D62B, "M", "j"), - (0x1D62C, "M", "k"), - (0x1D62D, "M", "l"), - (0x1D62E, "M", "m"), - (0x1D62F, "M", "n"), - (0x1D630, "M", "o"), - (0x1D631, "M", "p"), - (0x1D632, "M", "q"), - (0x1D633, "M", "r"), - (0x1D634, "M", "s"), - (0x1D635, "M", "t"), - (0x1D636, "M", "u"), - (0x1D637, "M", "v"), - (0x1D638, "M", "w"), - (0x1D639, "M", "x"), - (0x1D63A, "M", "y"), - (0x1D63B, "M", "z"), - (0x1D63C, "M", "a"), - (0x1D63D, "M", "b"), - (0x1D63E, "M", "c"), - (0x1D63F, "M", "d"), - (0x1D640, "M", "e"), - (0x1D641, "M", "f"), - (0x1D642, "M", "g"), - (0x1D643, "M", "h"), - (0x1D644, "M", "i"), - (0x1D645, "M", "j"), - (0x1D646, "M", "k"), - (0x1D647, "M", "l"), - (0x1D648, "M", "m"), - (0x1D649, "M", "n"), - (0x1D64A, "M", "o"), - (0x1D64B, "M", "p"), - (0x1D64C, "M", "q"), - (0x1D64D, "M", "r"), - (0x1D64E, "M", "s"), - (0x1D64F, "M", "t"), - (0x1D650, "M", "u"), - (0x1D651, "M", "v"), - (0x1D652, "M", "w"), - (0x1D653, "M", "x"), - (0x1D654, "M", "y"), - (0x1D655, "M", "z"), - ] - - -def _seg_68() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D656, "M", "a"), - (0x1D657, "M", "b"), - (0x1D658, "M", "c"), - (0x1D659, "M", "d"), - (0x1D65A, "M", "e"), - (0x1D65B, "M", "f"), - (0x1D65C, "M", "g"), - (0x1D65D, "M", "h"), - (0x1D65E, "M", "i"), - (0x1D65F, "M", "j"), - (0x1D660, "M", "k"), - (0x1D661, "M", "l"), - (0x1D662, "M", "m"), - (0x1D663, "M", "n"), - (0x1D664, "M", "o"), - (0x1D665, "M", "p"), - (0x1D666, "M", "q"), - (0x1D667, "M", "r"), - (0x1D668, "M", "s"), - (0x1D669, "M", "t"), - (0x1D66A, "M", "u"), - (0x1D66B, "M", "v"), - (0x1D66C, "M", "w"), - (0x1D66D, "M", "x"), - (0x1D66E, "M", "y"), - (0x1D66F, "M", "z"), - (0x1D670, "M", "a"), - (0x1D671, "M", "b"), - (0x1D672, "M", "c"), - (0x1D673, "M", "d"), - (0x1D674, "M", "e"), - (0x1D675, "M", "f"), - (0x1D676, "M", "g"), - (0x1D677, "M", "h"), - (0x1D678, "M", "i"), - (0x1D679, "M", "j"), - (0x1D67A, "M", "k"), - (0x1D67B, "M", "l"), - (0x1D67C, "M", "m"), - (0x1D67D, "M", "n"), - (0x1D67E, "M", "o"), - (0x1D67F, "M", "p"), - (0x1D680, "M", "q"), - (0x1D681, "M", "r"), - (0x1D682, "M", "s"), - (0x1D683, "M", "t"), - (0x1D684, "M", "u"), - (0x1D685, "M", "v"), - (0x1D686, "M", "w"), - (0x1D687, "M", "x"), - (0x1D688, "M", "y"), - (0x1D689, "M", "z"), - (0x1D68A, "M", "a"), - (0x1D68B, "M", "b"), - (0x1D68C, "M", "c"), - (0x1D68D, "M", "d"), - (0x1D68E, "M", "e"), - (0x1D68F, "M", "f"), - (0x1D690, "M", "g"), - (0x1D691, "M", "h"), - (0x1D692, "M", "i"), - (0x1D693, "M", "j"), - (0x1D694, "M", "k"), - (0x1D695, "M", "l"), - (0x1D696, "M", "m"), - (0x1D697, "M", "n"), - (0x1D698, "M", "o"), - (0x1D699, "M", "p"), - (0x1D69A, "M", "q"), - (0x1D69B, "M", "r"), - (0x1D69C, "M", "s"), - (0x1D69D, "M", "t"), - (0x1D69E, "M", "u"), - (0x1D69F, "M", "v"), - (0x1D6A0, "M", "w"), - (0x1D6A1, "M", "x"), - (0x1D6A2, "M", "y"), - (0x1D6A3, "M", "z"), - (0x1D6A4, "M", "ı"), - (0x1D6A5, "M", "ȷ"), - (0x1D6A6, "X"), - (0x1D6A8, "M", "α"), - (0x1D6A9, "M", "β"), - (0x1D6AA, "M", "γ"), - (0x1D6AB, "M", "δ"), - (0x1D6AC, "M", "ε"), - (0x1D6AD, "M", "ζ"), - (0x1D6AE, "M", "η"), - (0x1D6AF, "M", "θ"), - (0x1D6B0, "M", "ι"), - (0x1D6B1, "M", "κ"), - (0x1D6B2, "M", "λ"), - (0x1D6B3, "M", "μ"), - (0x1D6B4, "M", "ν"), - (0x1D6B5, "M", "ξ"), - (0x1D6B6, "M", "ο"), - (0x1D6B7, "M", "π"), - (0x1D6B8, "M", "ρ"), - (0x1D6B9, "M", "θ"), - (0x1D6BA, "M", "σ"), - ] - - -def _seg_69() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D6BB, "M", "τ"), - (0x1D6BC, "M", "υ"), - (0x1D6BD, "M", "φ"), - (0x1D6BE, "M", "χ"), - (0x1D6BF, "M", "ψ"), - (0x1D6C0, "M", "ω"), - (0x1D6C1, "M", "∇"), - (0x1D6C2, "M", "α"), - (0x1D6C3, "M", "β"), - (0x1D6C4, "M", "γ"), - (0x1D6C5, "M", "δ"), - (0x1D6C6, "M", "ε"), - (0x1D6C7, "M", "ζ"), - (0x1D6C8, "M", "η"), - (0x1D6C9, "M", "θ"), - (0x1D6CA, "M", "ι"), - (0x1D6CB, "M", "κ"), - (0x1D6CC, "M", "λ"), - (0x1D6CD, "M", "μ"), - (0x1D6CE, "M", "ν"), - (0x1D6CF, "M", "ξ"), - (0x1D6D0, "M", "ο"), - (0x1D6D1, "M", "π"), - (0x1D6D2, "M", "ρ"), - (0x1D6D3, "M", "σ"), - (0x1D6D5, "M", "τ"), - (0x1D6D6, "M", "υ"), - (0x1D6D7, "M", "φ"), - (0x1D6D8, "M", "χ"), - (0x1D6D9, "M", "ψ"), - (0x1D6DA, "M", "ω"), - (0x1D6DB, "M", "∂"), - (0x1D6DC, "M", "ε"), - (0x1D6DD, "M", "θ"), - (0x1D6DE, "M", "κ"), - (0x1D6DF, "M", "φ"), - (0x1D6E0, "M", "ρ"), - (0x1D6E1, "M", "π"), - (0x1D6E2, "M", "α"), - (0x1D6E3, "M", "β"), - (0x1D6E4, "M", "γ"), - (0x1D6E5, "M", "δ"), - (0x1D6E6, "M", "ε"), - (0x1D6E7, "M", "ζ"), - (0x1D6E8, "M", "η"), - (0x1D6E9, "M", "θ"), - (0x1D6EA, "M", "ι"), - (0x1D6EB, "M", "κ"), - (0x1D6EC, "M", "λ"), - (0x1D6ED, "M", "μ"), - (0x1D6EE, "M", "ν"), - (0x1D6EF, "M", "ξ"), - (0x1D6F0, "M", "ο"), - (0x1D6F1, "M", "π"), - (0x1D6F2, "M", "ρ"), - (0x1D6F3, "M", "θ"), - (0x1D6F4, "M", "σ"), - (0x1D6F5, "M", "τ"), - (0x1D6F6, "M", "υ"), - (0x1D6F7, "M", "φ"), - (0x1D6F8, "M", "χ"), - (0x1D6F9, "M", "ψ"), - (0x1D6FA, "M", "ω"), - (0x1D6FB, "M", "∇"), - (0x1D6FC, "M", "α"), - (0x1D6FD, "M", "β"), - (0x1D6FE, "M", "γ"), - (0x1D6FF, "M", "δ"), - (0x1D700, "M", "ε"), - (0x1D701, "M", "ζ"), - (0x1D702, "M", "η"), - (0x1D703, "M", "θ"), - (0x1D704, "M", "ι"), - (0x1D705, "M", "κ"), - (0x1D706, "M", "λ"), - (0x1D707, "M", "μ"), - (0x1D708, "M", "ν"), - (0x1D709, "M", "ξ"), - (0x1D70A, "M", "ο"), - (0x1D70B, "M", "π"), - (0x1D70C, "M", "ρ"), - (0x1D70D, "M", "σ"), - (0x1D70F, "M", "τ"), - (0x1D710, "M", "υ"), - (0x1D711, "M", "φ"), - (0x1D712, "M", "χ"), - (0x1D713, "M", "ψ"), - (0x1D714, "M", "ω"), - (0x1D715, "M", "∂"), - (0x1D716, "M", "ε"), - (0x1D717, "M", "θ"), - (0x1D718, "M", "κ"), - (0x1D719, "M", "φ"), - (0x1D71A, "M", "ρ"), - (0x1D71B, "M", "π"), - (0x1D71C, "M", "α"), - (0x1D71D, "M", "β"), - (0x1D71E, "M", "γ"), - (0x1D71F, "M", "δ"), - (0x1D720, "M", "ε"), - ] - - -def _seg_70() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D721, "M", "ζ"), - (0x1D722, "M", "η"), - (0x1D723, "M", "θ"), - (0x1D724, "M", "ι"), - (0x1D725, "M", "κ"), - (0x1D726, "M", "λ"), - (0x1D727, "M", "μ"), - (0x1D728, "M", "ν"), - (0x1D729, "M", "ξ"), - (0x1D72A, "M", "ο"), - (0x1D72B, "M", "π"), - (0x1D72C, "M", "ρ"), - (0x1D72D, "M", "θ"), - (0x1D72E, "M", "σ"), - (0x1D72F, "M", "τ"), - (0x1D730, "M", "υ"), - (0x1D731, "M", "φ"), - (0x1D732, "M", "χ"), - (0x1D733, "M", "ψ"), - (0x1D734, "M", "ω"), - (0x1D735, "M", "∇"), - (0x1D736, "M", "α"), - (0x1D737, "M", "β"), - (0x1D738, "M", "γ"), - (0x1D739, "M", "δ"), - (0x1D73A, "M", "ε"), - (0x1D73B, "M", "ζ"), - (0x1D73C, "M", "η"), - (0x1D73D, "M", "θ"), - (0x1D73E, "M", "ι"), - (0x1D73F, "M", "κ"), - (0x1D740, "M", "λ"), - (0x1D741, "M", "μ"), - (0x1D742, "M", "ν"), - (0x1D743, "M", "ξ"), - (0x1D744, "M", "ο"), - (0x1D745, "M", "π"), - (0x1D746, "M", "ρ"), - (0x1D747, "M", "σ"), - (0x1D749, "M", "τ"), - (0x1D74A, "M", "υ"), - (0x1D74B, "M", "φ"), - (0x1D74C, "M", "χ"), - (0x1D74D, "M", "ψ"), - (0x1D74E, "M", "ω"), - (0x1D74F, "M", "∂"), - (0x1D750, "M", "ε"), - (0x1D751, "M", "θ"), - (0x1D752, "M", "κ"), - (0x1D753, "M", "φ"), - (0x1D754, "M", "ρ"), - (0x1D755, "M", "π"), - (0x1D756, "M", "α"), - (0x1D757, "M", "β"), - (0x1D758, "M", "γ"), - (0x1D759, "M", "δ"), - (0x1D75A, "M", "ε"), - (0x1D75B, "M", "ζ"), - (0x1D75C, "M", "η"), - (0x1D75D, "M", "θ"), - (0x1D75E, "M", "ι"), - (0x1D75F, "M", "κ"), - (0x1D760, "M", "λ"), - (0x1D761, "M", "μ"), - (0x1D762, "M", "ν"), - (0x1D763, "M", "ξ"), - (0x1D764, "M", "ο"), - (0x1D765, "M", "π"), - (0x1D766, "M", "ρ"), - (0x1D767, "M", "θ"), - (0x1D768, "M", "σ"), - (0x1D769, "M", "τ"), - (0x1D76A, "M", "υ"), - (0x1D76B, "M", "φ"), - (0x1D76C, "M", "χ"), - (0x1D76D, "M", "ψ"), - (0x1D76E, "M", "ω"), - (0x1D76F, "M", "∇"), - (0x1D770, "M", "α"), - (0x1D771, "M", "β"), - (0x1D772, "M", "γ"), - (0x1D773, "M", "δ"), - (0x1D774, "M", "ε"), - (0x1D775, "M", "ζ"), - (0x1D776, "M", "η"), - (0x1D777, "M", "θ"), - (0x1D778, "M", "ι"), - (0x1D779, "M", "κ"), - (0x1D77A, "M", "λ"), - (0x1D77B, "M", "μ"), - (0x1D77C, "M", "ν"), - (0x1D77D, "M", "ξ"), - (0x1D77E, "M", "ο"), - (0x1D77F, "M", "π"), - (0x1D780, "M", "ρ"), - (0x1D781, "M", "σ"), - (0x1D783, "M", "τ"), - (0x1D784, "M", "υ"), - (0x1D785, "M", "φ"), - (0x1D786, "M", "χ"), - ] - - -def _seg_71() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D787, "M", "ψ"), - (0x1D788, "M", "ω"), - (0x1D789, "M", "∂"), - (0x1D78A, "M", "ε"), - (0x1D78B, "M", "θ"), - (0x1D78C, "M", "κ"), - (0x1D78D, "M", "φ"), - (0x1D78E, "M", "ρ"), - (0x1D78F, "M", "π"), - (0x1D790, "M", "α"), - (0x1D791, "M", "β"), - (0x1D792, "M", "γ"), - (0x1D793, "M", "δ"), - (0x1D794, "M", "ε"), - (0x1D795, "M", "ζ"), - (0x1D796, "M", "η"), - (0x1D797, "M", "θ"), - (0x1D798, "M", "ι"), - (0x1D799, "M", "κ"), - (0x1D79A, "M", "λ"), - (0x1D79B, "M", "μ"), - (0x1D79C, "M", "ν"), - (0x1D79D, "M", "ξ"), - (0x1D79E, "M", "ο"), - (0x1D79F, "M", "π"), - (0x1D7A0, "M", "ρ"), - (0x1D7A1, "M", "θ"), - (0x1D7A2, "M", "σ"), - (0x1D7A3, "M", "τ"), - (0x1D7A4, "M", "υ"), - (0x1D7A5, "M", "φ"), - (0x1D7A6, "M", "χ"), - (0x1D7A7, "M", "ψ"), - (0x1D7A8, "M", "ω"), - (0x1D7A9, "M", "∇"), - (0x1D7AA, "M", "α"), - (0x1D7AB, "M", "β"), - (0x1D7AC, "M", "γ"), - (0x1D7AD, "M", "δ"), - (0x1D7AE, "M", "ε"), - (0x1D7AF, "M", "ζ"), - (0x1D7B0, "M", "η"), - (0x1D7B1, "M", "θ"), - (0x1D7B2, "M", "ι"), - (0x1D7B3, "M", "κ"), - (0x1D7B4, "M", "λ"), - (0x1D7B5, "M", "μ"), - (0x1D7B6, "M", "ν"), - (0x1D7B7, "M", "ξ"), - (0x1D7B8, "M", "ο"), - (0x1D7B9, "M", "π"), - (0x1D7BA, "M", "ρ"), - (0x1D7BB, "M", "σ"), - (0x1D7BD, "M", "τ"), - (0x1D7BE, "M", "υ"), - (0x1D7BF, "M", "φ"), - (0x1D7C0, "M", "χ"), - (0x1D7C1, "M", "ψ"), - (0x1D7C2, "M", "ω"), - (0x1D7C3, "M", "∂"), - (0x1D7C4, "M", "ε"), - (0x1D7C5, "M", "θ"), - (0x1D7C6, "M", "κ"), - (0x1D7C7, "M", "φ"), - (0x1D7C8, "M", "ρ"), - (0x1D7C9, "M", "π"), - (0x1D7CA, "M", "ϝ"), - (0x1D7CC, "X"), - (0x1D7CE, "M", "0"), - (0x1D7CF, "M", "1"), - (0x1D7D0, "M", "2"), - (0x1D7D1, "M", "3"), - (0x1D7D2, "M", "4"), - (0x1D7D3, "M", "5"), - (0x1D7D4, "M", "6"), - (0x1D7D5, "M", "7"), - (0x1D7D6, "M", "8"), - (0x1D7D7, "M", "9"), - (0x1D7D8, "M", "0"), - (0x1D7D9, "M", "1"), - (0x1D7DA, "M", "2"), - (0x1D7DB, "M", "3"), - (0x1D7DC, "M", "4"), - (0x1D7DD, "M", "5"), - (0x1D7DE, "M", "6"), - (0x1D7DF, "M", "7"), - (0x1D7E0, "M", "8"), - (0x1D7E1, "M", "9"), - (0x1D7E2, "M", "0"), - (0x1D7E3, "M", "1"), - (0x1D7E4, "M", "2"), - (0x1D7E5, "M", "3"), - (0x1D7E6, "M", "4"), - (0x1D7E7, "M", "5"), - (0x1D7E8, "M", "6"), - (0x1D7E9, "M", "7"), - (0x1D7EA, "M", "8"), - (0x1D7EB, "M", "9"), - (0x1D7EC, "M", "0"), - (0x1D7ED, "M", "1"), - ] - - -def _seg_72() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1D7EE, "M", "2"), - (0x1D7EF, "M", "3"), - (0x1D7F0, "M", "4"), - (0x1D7F1, "M", "5"), - (0x1D7F2, "M", "6"), - (0x1D7F3, "M", "7"), - (0x1D7F4, "M", "8"), - (0x1D7F5, "M", "9"), - (0x1D7F6, "M", "0"), - (0x1D7F7, "M", "1"), - (0x1D7F8, "M", "2"), - (0x1D7F9, "M", "3"), - (0x1D7FA, "M", "4"), - (0x1D7FB, "M", "5"), - (0x1D7FC, "M", "6"), - (0x1D7FD, "M", "7"), - (0x1D7FE, "M", "8"), - (0x1D7FF, "M", "9"), - (0x1D800, "V"), - (0x1DA8C, "X"), - (0x1DA9B, "V"), - (0x1DAA0, "X"), - (0x1DAA1, "V"), - (0x1DAB0, "X"), - (0x1DF00, "V"), - (0x1DF1F, "X"), - (0x1DF25, "V"), - (0x1DF2B, "X"), - (0x1E000, "V"), - (0x1E007, "X"), - (0x1E008, "V"), - (0x1E019, "X"), - (0x1E01B, "V"), - (0x1E022, "X"), - (0x1E023, "V"), - (0x1E025, "X"), - (0x1E026, "V"), - (0x1E02B, "X"), - (0x1E030, "M", "а"), - (0x1E031, "M", "б"), - (0x1E032, "M", "в"), - (0x1E033, "M", "г"), - (0x1E034, "M", "д"), - (0x1E035, "M", "е"), - (0x1E036, "M", "ж"), - (0x1E037, "M", "з"), - (0x1E038, "M", "и"), - (0x1E039, "M", "к"), - (0x1E03A, "M", "л"), - (0x1E03B, "M", "м"), - (0x1E03C, "M", "о"), - (0x1E03D, "M", "п"), - (0x1E03E, "M", "р"), - (0x1E03F, "M", "с"), - (0x1E040, "M", "т"), - (0x1E041, "M", "у"), - (0x1E042, "M", "ф"), - (0x1E043, "M", "х"), - (0x1E044, "M", "ц"), - (0x1E045, "M", "ч"), - (0x1E046, "M", "ш"), - (0x1E047, "M", "ы"), - (0x1E048, "M", "э"), - (0x1E049, "M", "ю"), - (0x1E04A, "M", "ꚉ"), - (0x1E04B, "M", "ә"), - (0x1E04C, "M", "і"), - (0x1E04D, "M", "ј"), - (0x1E04E, "M", "ө"), - (0x1E04F, "M", "ү"), - (0x1E050, "M", "ӏ"), - (0x1E051, "M", "а"), - (0x1E052, "M", "б"), - (0x1E053, "M", "в"), - (0x1E054, "M", "г"), - (0x1E055, "M", "д"), - (0x1E056, "M", "е"), - (0x1E057, "M", "ж"), - (0x1E058, "M", "з"), - (0x1E059, "M", "и"), - (0x1E05A, "M", "к"), - (0x1E05B, "M", "л"), - (0x1E05C, "M", "о"), - (0x1E05D, "M", "п"), - (0x1E05E, "M", "с"), - (0x1E05F, "M", "у"), - (0x1E060, "M", "ф"), - (0x1E061, "M", "х"), - (0x1E062, "M", "ц"), - (0x1E063, "M", "ч"), - (0x1E064, "M", "ш"), - (0x1E065, "M", "ъ"), - (0x1E066, "M", "ы"), - (0x1E067, "M", "ґ"), - (0x1E068, "M", "і"), - (0x1E069, "M", "ѕ"), - (0x1E06A, "M", "џ"), - (0x1E06B, "M", "ҫ"), - (0x1E06C, "M", "ꙑ"), - (0x1E06D, "M", "ұ"), - ] - - -def _seg_73() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1E06E, "X"), - (0x1E08F, "V"), - (0x1E090, "X"), - (0x1E100, "V"), - (0x1E12D, "X"), - (0x1E130, "V"), - (0x1E13E, "X"), - (0x1E140, "V"), - (0x1E14A, "X"), - (0x1E14E, "V"), - (0x1E150, "X"), - (0x1E290, "V"), - (0x1E2AF, "X"), - (0x1E2C0, "V"), - (0x1E2FA, "X"), - (0x1E2FF, "V"), - (0x1E300, "X"), - (0x1E4D0, "V"), - (0x1E4FA, "X"), - (0x1E5D0, "V"), - (0x1E5FB, "X"), - (0x1E5FF, "V"), - (0x1E600, "X"), - (0x1E7E0, "V"), - (0x1E7E7, "X"), - (0x1E7E8, "V"), - (0x1E7EC, "X"), - (0x1E7ED, "V"), - (0x1E7EF, "X"), - (0x1E7F0, "V"), - (0x1E7FF, "X"), - (0x1E800, "V"), - (0x1E8C5, "X"), - (0x1E8C7, "V"), - (0x1E8D7, "X"), - (0x1E900, "M", "𞤢"), - (0x1E901, "M", "𞤣"), - (0x1E902, "M", "𞤤"), - (0x1E903, "M", "𞤥"), - (0x1E904, "M", "𞤦"), - (0x1E905, "M", "𞤧"), - (0x1E906, "M", "𞤨"), - (0x1E907, "M", "𞤩"), - (0x1E908, "M", "𞤪"), - (0x1E909, "M", "𞤫"), - (0x1E90A, "M", "𞤬"), - (0x1E90B, "M", "𞤭"), - (0x1E90C, "M", "𞤮"), - (0x1E90D, "M", "𞤯"), - (0x1E90E, "M", "𞤰"), - (0x1E90F, "M", "𞤱"), - (0x1E910, "M", "𞤲"), - (0x1E911, "M", "𞤳"), - (0x1E912, "M", "𞤴"), - (0x1E913, "M", "𞤵"), - (0x1E914, "M", "𞤶"), - (0x1E915, "M", "𞤷"), - (0x1E916, "M", "𞤸"), - (0x1E917, "M", "𞤹"), - (0x1E918, "M", "𞤺"), - (0x1E919, "M", "𞤻"), - (0x1E91A, "M", "𞤼"), - (0x1E91B, "M", "𞤽"), - (0x1E91C, "M", "𞤾"), - (0x1E91D, "M", "𞤿"), - (0x1E91E, "M", "𞥀"), - (0x1E91F, "M", "𞥁"), - (0x1E920, "M", "𞥂"), - (0x1E921, "M", "𞥃"), - (0x1E922, "V"), - (0x1E94C, "X"), - (0x1E950, "V"), - (0x1E95A, "X"), - (0x1E95E, "V"), - (0x1E960, "X"), - (0x1EC71, "V"), - (0x1ECB5, "X"), - (0x1ED01, "V"), - (0x1ED3E, "X"), - (0x1EE00, "M", "ا"), - (0x1EE01, "M", "ب"), - (0x1EE02, "M", "ج"), - (0x1EE03, "M", "د"), - (0x1EE04, "X"), - (0x1EE05, "M", "و"), - (0x1EE06, "M", "ز"), - (0x1EE07, "M", "ح"), - (0x1EE08, "M", "ط"), - (0x1EE09, "M", "ي"), - (0x1EE0A, "M", "ك"), - (0x1EE0B, "M", "ل"), - (0x1EE0C, "M", "م"), - (0x1EE0D, "M", "ن"), - (0x1EE0E, "M", "س"), - (0x1EE0F, "M", "ع"), - (0x1EE10, "M", "ف"), - (0x1EE11, "M", "ص"), - (0x1EE12, "M", "ق"), - (0x1EE13, "M", "ر"), - (0x1EE14, "M", "ش"), - ] - - -def _seg_74() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1EE15, "M", "ت"), - (0x1EE16, "M", "ث"), - (0x1EE17, "M", "خ"), - (0x1EE18, "M", "ذ"), - (0x1EE19, "M", "ض"), - (0x1EE1A, "M", "ظ"), - (0x1EE1B, "M", "غ"), - (0x1EE1C, "M", "ٮ"), - (0x1EE1D, "M", "ں"), - (0x1EE1E, "M", "ڡ"), - (0x1EE1F, "M", "ٯ"), - (0x1EE20, "X"), - (0x1EE21, "M", "ب"), - (0x1EE22, "M", "ج"), - (0x1EE23, "X"), - (0x1EE24, "M", "ه"), - (0x1EE25, "X"), - (0x1EE27, "M", "ح"), - (0x1EE28, "X"), - (0x1EE29, "M", "ي"), - (0x1EE2A, "M", "ك"), - (0x1EE2B, "M", "ل"), - (0x1EE2C, "M", "م"), - (0x1EE2D, "M", "ن"), - (0x1EE2E, "M", "س"), - (0x1EE2F, "M", "ع"), - (0x1EE30, "M", "ف"), - (0x1EE31, "M", "ص"), - (0x1EE32, "M", "ق"), - (0x1EE33, "X"), - (0x1EE34, "M", "ش"), - (0x1EE35, "M", "ت"), - (0x1EE36, "M", "ث"), - (0x1EE37, "M", "خ"), - (0x1EE38, "X"), - (0x1EE39, "M", "ض"), - (0x1EE3A, "X"), - (0x1EE3B, "M", "غ"), - (0x1EE3C, "X"), - (0x1EE42, "M", "ج"), - (0x1EE43, "X"), - (0x1EE47, "M", "ح"), - (0x1EE48, "X"), - (0x1EE49, "M", "ي"), - (0x1EE4A, "X"), - (0x1EE4B, "M", "ل"), - (0x1EE4C, "X"), - (0x1EE4D, "M", "ن"), - (0x1EE4E, "M", "س"), - (0x1EE4F, "M", "ع"), - (0x1EE50, "X"), - (0x1EE51, "M", "ص"), - (0x1EE52, "M", "ق"), - (0x1EE53, "X"), - (0x1EE54, "M", "ش"), - (0x1EE55, "X"), - (0x1EE57, "M", "خ"), - (0x1EE58, "X"), - (0x1EE59, "M", "ض"), - (0x1EE5A, "X"), - (0x1EE5B, "M", "غ"), - (0x1EE5C, "X"), - (0x1EE5D, "M", "ں"), - (0x1EE5E, "X"), - (0x1EE5F, "M", "ٯ"), - (0x1EE60, "X"), - (0x1EE61, "M", "ب"), - (0x1EE62, "M", "ج"), - (0x1EE63, "X"), - (0x1EE64, "M", "ه"), - (0x1EE65, "X"), - (0x1EE67, "M", "ح"), - (0x1EE68, "M", "ط"), - (0x1EE69, "M", "ي"), - (0x1EE6A, "M", "ك"), - (0x1EE6B, "X"), - (0x1EE6C, "M", "م"), - (0x1EE6D, "M", "ن"), - (0x1EE6E, "M", "س"), - (0x1EE6F, "M", "ع"), - (0x1EE70, "M", "ف"), - (0x1EE71, "M", "ص"), - (0x1EE72, "M", "ق"), - (0x1EE73, "X"), - (0x1EE74, "M", "ش"), - (0x1EE75, "M", "ت"), - (0x1EE76, "M", "ث"), - (0x1EE77, "M", "خ"), - (0x1EE78, "X"), - (0x1EE79, "M", "ض"), - (0x1EE7A, "M", "ظ"), - (0x1EE7B, "M", "غ"), - (0x1EE7C, "M", "ٮ"), - (0x1EE7D, "X"), - (0x1EE7E, "M", "ڡ"), - (0x1EE7F, "X"), - (0x1EE80, "M", "ا"), - (0x1EE81, "M", "ب"), - (0x1EE82, "M", "ج"), - (0x1EE83, "M", "د"), - ] - - -def _seg_75() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1EE84, "M", "ه"), - (0x1EE85, "M", "و"), - (0x1EE86, "M", "ز"), - (0x1EE87, "M", "ح"), - (0x1EE88, "M", "ط"), - (0x1EE89, "M", "ي"), - (0x1EE8A, "X"), - (0x1EE8B, "M", "ل"), - (0x1EE8C, "M", "م"), - (0x1EE8D, "M", "ن"), - (0x1EE8E, "M", "س"), - (0x1EE8F, "M", "ع"), - (0x1EE90, "M", "ف"), - (0x1EE91, "M", "ص"), - (0x1EE92, "M", "ق"), - (0x1EE93, "M", "ر"), - (0x1EE94, "M", "ش"), - (0x1EE95, "M", "ت"), - (0x1EE96, "M", "ث"), - (0x1EE97, "M", "خ"), - (0x1EE98, "M", "ذ"), - (0x1EE99, "M", "ض"), - (0x1EE9A, "M", "ظ"), - (0x1EE9B, "M", "غ"), - (0x1EE9C, "X"), - (0x1EEA1, "M", "ب"), - (0x1EEA2, "M", "ج"), - (0x1EEA3, "M", "د"), - (0x1EEA4, "X"), - (0x1EEA5, "M", "و"), - (0x1EEA6, "M", "ز"), - (0x1EEA7, "M", "ح"), - (0x1EEA8, "M", "ط"), - (0x1EEA9, "M", "ي"), - (0x1EEAA, "X"), - (0x1EEAB, "M", "ل"), - (0x1EEAC, "M", "م"), - (0x1EEAD, "M", "ن"), - (0x1EEAE, "M", "س"), - (0x1EEAF, "M", "ع"), - (0x1EEB0, "M", "ف"), - (0x1EEB1, "M", "ص"), - (0x1EEB2, "M", "ق"), - (0x1EEB3, "M", "ر"), - (0x1EEB4, "M", "ش"), - (0x1EEB5, "M", "ت"), - (0x1EEB6, "M", "ث"), - (0x1EEB7, "M", "خ"), - (0x1EEB8, "M", "ذ"), - (0x1EEB9, "M", "ض"), - (0x1EEBA, "M", "ظ"), - (0x1EEBB, "M", "غ"), - (0x1EEBC, "X"), - (0x1EEF0, "V"), - (0x1EEF2, "X"), - (0x1F000, "V"), - (0x1F02C, "X"), - (0x1F030, "V"), - (0x1F094, "X"), - (0x1F0A0, "V"), - (0x1F0AF, "X"), - (0x1F0B1, "V"), - (0x1F0C0, "X"), - (0x1F0C1, "V"), - (0x1F0D0, "X"), - (0x1F0D1, "V"), - (0x1F0F6, "X"), - (0x1F101, "M", "0,"), - (0x1F102, "M", "1,"), - (0x1F103, "M", "2,"), - (0x1F104, "M", "3,"), - (0x1F105, "M", "4,"), - (0x1F106, "M", "5,"), - (0x1F107, "M", "6,"), - (0x1F108, "M", "7,"), - (0x1F109, "M", "8,"), - (0x1F10A, "M", "9,"), - (0x1F10B, "V"), - (0x1F110, "M", "(a)"), - (0x1F111, "M", "(b)"), - (0x1F112, "M", "(c)"), - (0x1F113, "M", "(d)"), - (0x1F114, "M", "(e)"), - (0x1F115, "M", "(f)"), - (0x1F116, "M", "(g)"), - (0x1F117, "M", "(h)"), - (0x1F118, "M", "(i)"), - (0x1F119, "M", "(j)"), - (0x1F11A, "M", "(k)"), - (0x1F11B, "M", "(l)"), - (0x1F11C, "M", "(m)"), - (0x1F11D, "M", "(n)"), - (0x1F11E, "M", "(o)"), - (0x1F11F, "M", "(p)"), - (0x1F120, "M", "(q)"), - (0x1F121, "M", "(r)"), - (0x1F122, "M", "(s)"), - (0x1F123, "M", "(t)"), - (0x1F124, "M", "(u)"), - (0x1F125, "M", "(v)"), - ] - - -def _seg_76() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1F126, "M", "(w)"), - (0x1F127, "M", "(x)"), - (0x1F128, "M", "(y)"), - (0x1F129, "M", "(z)"), - (0x1F12A, "M", "〔s〕"), - (0x1F12B, "M", "c"), - (0x1F12C, "M", "r"), - (0x1F12D, "M", "cd"), - (0x1F12E, "M", "wz"), - (0x1F12F, "V"), - (0x1F130, "M", "a"), - (0x1F131, "M", "b"), - (0x1F132, "M", "c"), - (0x1F133, "M", "d"), - (0x1F134, "M", "e"), - (0x1F135, "M", "f"), - (0x1F136, "M", "g"), - (0x1F137, "M", "h"), - (0x1F138, "M", "i"), - (0x1F139, "M", "j"), - (0x1F13A, "M", "k"), - (0x1F13B, "M", "l"), - (0x1F13C, "M", "m"), - (0x1F13D, "M", "n"), - (0x1F13E, "M", "o"), - (0x1F13F, "M", "p"), - (0x1F140, "M", "q"), - (0x1F141, "M", "r"), - (0x1F142, "M", "s"), - (0x1F143, "M", "t"), - (0x1F144, "M", "u"), - (0x1F145, "M", "v"), - (0x1F146, "M", "w"), - (0x1F147, "M", "x"), - (0x1F148, "M", "y"), - (0x1F149, "M", "z"), - (0x1F14A, "M", "hv"), - (0x1F14B, "M", "mv"), - (0x1F14C, "M", "sd"), - (0x1F14D, "M", "ss"), - (0x1F14E, "M", "ppv"), - (0x1F14F, "M", "wc"), - (0x1F150, "V"), - (0x1F16A, "M", "mc"), - (0x1F16B, "M", "md"), - (0x1F16C, "M", "mr"), - (0x1F16D, "V"), - (0x1F190, "M", "dj"), - (0x1F191, "V"), - (0x1F1AE, "X"), - (0x1F1E6, "V"), - (0x1F200, "M", "ほか"), - (0x1F201, "M", "ココ"), - (0x1F202, "M", "サ"), - (0x1F203, "X"), - (0x1F210, "M", "手"), - (0x1F211, "M", "字"), - (0x1F212, "M", "双"), - (0x1F213, "M", "デ"), - (0x1F214, "M", "二"), - (0x1F215, "M", "多"), - (0x1F216, "M", "解"), - (0x1F217, "M", "天"), - (0x1F218, "M", "交"), - (0x1F219, "M", "映"), - (0x1F21A, "M", "無"), - (0x1F21B, "M", "料"), - (0x1F21C, "M", "前"), - (0x1F21D, "M", "後"), - (0x1F21E, "M", "再"), - (0x1F21F, "M", "新"), - (0x1F220, "M", "初"), - (0x1F221, "M", "終"), - (0x1F222, "M", "生"), - (0x1F223, "M", "販"), - (0x1F224, "M", "声"), - (0x1F225, "M", "吹"), - (0x1F226, "M", "演"), - (0x1F227, "M", "投"), - (0x1F228, "M", "捕"), - (0x1F229, "M", "一"), - (0x1F22A, "M", "三"), - (0x1F22B, "M", "遊"), - (0x1F22C, "M", "左"), - (0x1F22D, "M", "中"), - (0x1F22E, "M", "右"), - (0x1F22F, "M", "指"), - (0x1F230, "M", "走"), - (0x1F231, "M", "打"), - (0x1F232, "M", "禁"), - (0x1F233, "M", "空"), - (0x1F234, "M", "合"), - (0x1F235, "M", "満"), - (0x1F236, "M", "有"), - (0x1F237, "M", "月"), - (0x1F238, "M", "申"), - (0x1F239, "M", "割"), - (0x1F23A, "M", "営"), - (0x1F23B, "M", "配"), - (0x1F23C, "X"), - ] - - -def _seg_77() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x1F240, "M", "〔本〕"), - (0x1F241, "M", "〔三〕"), - (0x1F242, "M", "〔二〕"), - (0x1F243, "M", "〔安〕"), - (0x1F244, "M", "〔点〕"), - (0x1F245, "M", "〔打〕"), - (0x1F246, "M", "〔盗〕"), - (0x1F247, "M", "〔勝〕"), - (0x1F248, "M", "〔敗〕"), - (0x1F249, "X"), - (0x1F250, "M", "得"), - (0x1F251, "M", "可"), - (0x1F252, "X"), - (0x1F260, "V"), - (0x1F266, "X"), - (0x1F300, "V"), - (0x1F6D8, "X"), - (0x1F6DC, "V"), - (0x1F6ED, "X"), - (0x1F6F0, "V"), - (0x1F6FD, "X"), - (0x1F700, "V"), - (0x1F777, "X"), - (0x1F77B, "V"), - (0x1F7DA, "X"), - (0x1F7E0, "V"), - (0x1F7EC, "X"), - (0x1F7F0, "V"), - (0x1F7F1, "X"), - (0x1F800, "V"), - (0x1F80C, "X"), - (0x1F810, "V"), - (0x1F848, "X"), - (0x1F850, "V"), - (0x1F85A, "X"), - (0x1F860, "V"), - (0x1F888, "X"), - (0x1F890, "V"), - (0x1F8AE, "X"), - (0x1F8B0, "V"), - (0x1F8BC, "X"), - (0x1F8C0, "V"), - (0x1F8C2, "X"), - (0x1F900, "V"), - (0x1FA54, "X"), - (0x1FA60, "V"), - (0x1FA6E, "X"), - (0x1FA70, "V"), - (0x1FA7D, "X"), - (0x1FA80, "V"), - (0x1FA8A, "X"), - (0x1FA8F, "V"), - (0x1FAC7, "X"), - (0x1FACE, "V"), - (0x1FADD, "X"), - (0x1FADF, "V"), - (0x1FAEA, "X"), - (0x1FAF0, "V"), - (0x1FAF9, "X"), - (0x1FB00, "V"), - (0x1FB93, "X"), - (0x1FB94, "V"), - (0x1FBF0, "M", "0"), - (0x1FBF1, "M", "1"), - (0x1FBF2, "M", "2"), - (0x1FBF3, "M", "3"), - (0x1FBF4, "M", "4"), - (0x1FBF5, "M", "5"), - (0x1FBF6, "M", "6"), - (0x1FBF7, "M", "7"), - (0x1FBF8, "M", "8"), - (0x1FBF9, "M", "9"), - (0x1FBFA, "X"), - (0x20000, "V"), - (0x2A6E0, "X"), - (0x2A700, "V"), - (0x2B73A, "X"), - (0x2B740, "V"), - (0x2B81E, "X"), - (0x2B820, "V"), - (0x2CEA2, "X"), - (0x2CEB0, "V"), - (0x2EBE1, "X"), - (0x2EBF0, "V"), - (0x2EE5E, "X"), - (0x2F800, "M", "丽"), - (0x2F801, "M", "丸"), - (0x2F802, "M", "乁"), - (0x2F803, "M", "𠄢"), - (0x2F804, "M", "你"), - (0x2F805, "M", "侮"), - (0x2F806, "M", "侻"), - (0x2F807, "M", "倂"), - (0x2F808, "M", "偺"), - (0x2F809, "M", "備"), - (0x2F80A, "M", "僧"), - (0x2F80B, "M", "像"), - (0x2F80C, "M", "㒞"), - (0x2F80D, "M", "𠘺"), - (0x2F80E, "M", "免"), - ] - - -def _seg_78() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F80F, "M", "兔"), - (0x2F810, "M", "兤"), - (0x2F811, "M", "具"), - (0x2F812, "M", "𠔜"), - (0x2F813, "M", "㒹"), - (0x2F814, "M", "內"), - (0x2F815, "M", "再"), - (0x2F816, "M", "𠕋"), - (0x2F817, "M", "冗"), - (0x2F818, "M", "冤"), - (0x2F819, "M", "仌"), - (0x2F81A, "M", "冬"), - (0x2F81B, "M", "况"), - (0x2F81C, "M", "𩇟"), - (0x2F81D, "M", "凵"), - (0x2F81E, "M", "刃"), - (0x2F81F, "M", "㓟"), - (0x2F820, "M", "刻"), - (0x2F821, "M", "剆"), - (0x2F822, "M", "割"), - (0x2F823, "M", "剷"), - (0x2F824, "M", "㔕"), - (0x2F825, "M", "勇"), - (0x2F826, "M", "勉"), - (0x2F827, "M", "勤"), - (0x2F828, "M", "勺"), - (0x2F829, "M", "包"), - (0x2F82A, "M", "匆"), - (0x2F82B, "M", "北"), - (0x2F82C, "M", "卉"), - (0x2F82D, "M", "卑"), - (0x2F82E, "M", "博"), - (0x2F82F, "M", "即"), - (0x2F830, "M", "卽"), - (0x2F831, "M", "卿"), - (0x2F834, "M", "𠨬"), - (0x2F835, "M", "灰"), - (0x2F836, "M", "及"), - (0x2F837, "M", "叟"), - (0x2F838, "M", "𠭣"), - (0x2F839, "M", "叫"), - (0x2F83A, "M", "叱"), - (0x2F83B, "M", "吆"), - (0x2F83C, "M", "咞"), - (0x2F83D, "M", "吸"), - (0x2F83E, "M", "呈"), - (0x2F83F, "M", "周"), - (0x2F840, "M", "咢"), - (0x2F841, "M", "哶"), - (0x2F842, "M", "唐"), - (0x2F843, "M", "啓"), - (0x2F844, "M", "啣"), - (0x2F845, "M", "善"), - (0x2F847, "M", "喙"), - (0x2F848, "M", "喫"), - (0x2F849, "M", "喳"), - (0x2F84A, "M", "嗂"), - (0x2F84B, "M", "圖"), - (0x2F84C, "M", "嘆"), - (0x2F84D, "M", "圗"), - (0x2F84E, "M", "噑"), - (0x2F84F, "M", "噴"), - (0x2F850, "M", "切"), - (0x2F851, "M", "壮"), - (0x2F852, "M", "城"), - (0x2F853, "M", "埴"), - (0x2F854, "M", "堍"), - (0x2F855, "M", "型"), - (0x2F856, "M", "堲"), - (0x2F857, "M", "報"), - (0x2F858, "M", "墬"), - (0x2F859, "M", "𡓤"), - (0x2F85A, "M", "売"), - (0x2F85B, "M", "壷"), - (0x2F85C, "M", "夆"), - (0x2F85D, "M", "多"), - (0x2F85E, "M", "夢"), - (0x2F85F, "M", "奢"), - (0x2F860, "M", "𡚨"), - (0x2F861, "M", "𡛪"), - (0x2F862, "M", "姬"), - (0x2F863, "M", "娛"), - (0x2F864, "M", "娧"), - (0x2F865, "M", "姘"), - (0x2F866, "M", "婦"), - (0x2F867, "M", "㛮"), - (0x2F868, "M", "㛼"), - (0x2F869, "M", "嬈"), - (0x2F86A, "M", "嬾"), - (0x2F86C, "M", "𡧈"), - (0x2F86D, "M", "寃"), - (0x2F86E, "M", "寘"), - (0x2F86F, "M", "寧"), - (0x2F870, "M", "寳"), - (0x2F871, "M", "𡬘"), - (0x2F872, "M", "寿"), - (0x2F873, "M", "将"), - (0x2F874, "M", "当"), - (0x2F875, "M", "尢"), - (0x2F876, "M", "㞁"), - ] - - -def _seg_79() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F877, "M", "屠"), - (0x2F878, "M", "屮"), - (0x2F879, "M", "峀"), - (0x2F87A, "M", "岍"), - (0x2F87B, "M", "𡷤"), - (0x2F87C, "M", "嵃"), - (0x2F87D, "M", "𡷦"), - (0x2F87E, "M", "嵮"), - (0x2F87F, "M", "嵫"), - (0x2F880, "M", "嵼"), - (0x2F881, "M", "巡"), - (0x2F882, "M", "巢"), - (0x2F883, "M", "㠯"), - (0x2F884, "M", "巽"), - (0x2F885, "M", "帨"), - (0x2F886, "M", "帽"), - (0x2F887, "M", "幩"), - (0x2F888, "M", "㡢"), - (0x2F889, "M", "𢆃"), - (0x2F88A, "M", "㡼"), - (0x2F88B, "M", "庰"), - (0x2F88C, "M", "庳"), - (0x2F88D, "M", "庶"), - (0x2F88E, "M", "廊"), - (0x2F88F, "M", "𪎒"), - (0x2F890, "M", "廾"), - (0x2F891, "M", "𢌱"), - (0x2F893, "M", "舁"), - (0x2F894, "M", "弢"), - (0x2F896, "M", "㣇"), - (0x2F897, "M", "𣊸"), - (0x2F898, "M", "𦇚"), - (0x2F899, "M", "形"), - (0x2F89A, "M", "彫"), - (0x2F89B, "M", "㣣"), - (0x2F89C, "M", "徚"), - (0x2F89D, "M", "忍"), - (0x2F89E, "M", "志"), - (0x2F89F, "M", "忹"), - (0x2F8A0, "M", "悁"), - (0x2F8A1, "M", "㤺"), - (0x2F8A2, "M", "㤜"), - (0x2F8A3, "M", "悔"), - (0x2F8A4, "M", "𢛔"), - (0x2F8A5, "M", "惇"), - (0x2F8A6, "M", "慈"), - (0x2F8A7, "M", "慌"), - (0x2F8A8, "M", "慎"), - (0x2F8A9, "M", "慌"), - (0x2F8AA, "M", "慺"), - (0x2F8AB, "M", "憎"), - (0x2F8AC, "M", "憲"), - (0x2F8AD, "M", "憤"), - (0x2F8AE, "M", "憯"), - (0x2F8AF, "M", "懞"), - (0x2F8B0, "M", "懲"), - (0x2F8B1, "M", "懶"), - (0x2F8B2, "M", "成"), - (0x2F8B3, "M", "戛"), - (0x2F8B4, "M", "扝"), - (0x2F8B5, "M", "抱"), - (0x2F8B6, "M", "拔"), - (0x2F8B7, "M", "捐"), - (0x2F8B8, "M", "𢬌"), - (0x2F8B9, "M", "挽"), - (0x2F8BA, "M", "拼"), - (0x2F8BB, "M", "捨"), - (0x2F8BC, "M", "掃"), - (0x2F8BD, "M", "揤"), - (0x2F8BE, "M", "𢯱"), - (0x2F8BF, "M", "搢"), - (0x2F8C0, "M", "揅"), - (0x2F8C1, "M", "掩"), - (0x2F8C2, "M", "㨮"), - (0x2F8C3, "M", "摩"), - (0x2F8C4, "M", "摾"), - (0x2F8C5, "M", "撝"), - (0x2F8C6, "M", "摷"), - (0x2F8C7, "M", "㩬"), - (0x2F8C8, "M", "敏"), - (0x2F8C9, "M", "敬"), - (0x2F8CA, "M", "𣀊"), - (0x2F8CB, "M", "旣"), - (0x2F8CC, "M", "書"), - (0x2F8CD, "M", "晉"), - (0x2F8CE, "M", "㬙"), - (0x2F8CF, "M", "暑"), - (0x2F8D0, "M", "㬈"), - (0x2F8D1, "M", "㫤"), - (0x2F8D2, "M", "冒"), - (0x2F8D3, "M", "冕"), - (0x2F8D4, "M", "最"), - (0x2F8D5, "M", "暜"), - (0x2F8D6, "M", "肭"), - (0x2F8D7, "M", "䏙"), - (0x2F8D8, "M", "朗"), - (0x2F8D9, "M", "望"), - (0x2F8DA, "M", "朡"), - (0x2F8DB, "M", "杞"), - (0x2F8DC, "M", "杓"), - ] - - -def _seg_80() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F8DD, "M", "𣏃"), - (0x2F8DE, "M", "㭉"), - (0x2F8DF, "M", "柺"), - (0x2F8E0, "M", "枅"), - (0x2F8E1, "M", "桒"), - (0x2F8E2, "M", "梅"), - (0x2F8E3, "M", "𣑭"), - (0x2F8E4, "M", "梎"), - (0x2F8E5, "M", "栟"), - (0x2F8E6, "M", "椔"), - (0x2F8E7, "M", "㮝"), - (0x2F8E8, "M", "楂"), - (0x2F8E9, "M", "榣"), - (0x2F8EA, "M", "槪"), - (0x2F8EB, "M", "檨"), - (0x2F8EC, "M", "𣚣"), - (0x2F8ED, "M", "櫛"), - (0x2F8EE, "M", "㰘"), - (0x2F8EF, "M", "次"), - (0x2F8F0, "M", "𣢧"), - (0x2F8F1, "M", "歔"), - (0x2F8F2, "M", "㱎"), - (0x2F8F3, "M", "歲"), - (0x2F8F4, "M", "殟"), - (0x2F8F5, "M", "殺"), - (0x2F8F6, "M", "殻"), - (0x2F8F7, "M", "𣪍"), - (0x2F8F8, "M", "𡴋"), - (0x2F8F9, "M", "𣫺"), - (0x2F8FA, "M", "汎"), - (0x2F8FB, "M", "𣲼"), - (0x2F8FC, "M", "沿"), - (0x2F8FD, "M", "泍"), - (0x2F8FE, "M", "汧"), - (0x2F8FF, "M", "洖"), - (0x2F900, "M", "派"), - (0x2F901, "M", "海"), - (0x2F902, "M", "流"), - (0x2F903, "M", "浩"), - (0x2F904, "M", "浸"), - (0x2F905, "M", "涅"), - (0x2F906, "M", "𣴞"), - (0x2F907, "M", "洴"), - (0x2F908, "M", "港"), - (0x2F909, "M", "湮"), - (0x2F90A, "M", "㴳"), - (0x2F90B, "M", "滋"), - (0x2F90C, "M", "滇"), - (0x2F90D, "M", "𣻑"), - (0x2F90E, "M", "淹"), - (0x2F90F, "M", "潮"), - (0x2F910, "M", "𣽞"), - (0x2F911, "M", "𣾎"), - (0x2F912, "M", "濆"), - (0x2F913, "M", "瀹"), - (0x2F914, "M", "瀞"), - (0x2F915, "M", "瀛"), - (0x2F916, "M", "㶖"), - (0x2F917, "M", "灊"), - (0x2F918, "M", "災"), - (0x2F919, "M", "灷"), - (0x2F91A, "M", "炭"), - (0x2F91B, "M", "𠔥"), - (0x2F91C, "M", "煅"), - (0x2F91D, "M", "𤉣"), - (0x2F91E, "M", "熜"), - (0x2F91F, "M", "𤎫"), - (0x2F920, "M", "爨"), - (0x2F921, "M", "爵"), - (0x2F922, "M", "牐"), - (0x2F923, "M", "𤘈"), - (0x2F924, "M", "犀"), - (0x2F925, "M", "犕"), - (0x2F926, "M", "𤜵"), - (0x2F927, "M", "𤠔"), - (0x2F928, "M", "獺"), - (0x2F929, "M", "王"), - (0x2F92A, "M", "㺬"), - (0x2F92B, "M", "玥"), - (0x2F92C, "M", "㺸"), - (0x2F92E, "M", "瑇"), - (0x2F92F, "M", "瑜"), - (0x2F930, "M", "瑱"), - (0x2F931, "M", "璅"), - (0x2F932, "M", "瓊"), - (0x2F933, "M", "㼛"), - (0x2F934, "M", "甤"), - (0x2F935, "M", "𤰶"), - (0x2F936, "M", "甾"), - (0x2F937, "M", "𤲒"), - (0x2F938, "M", "異"), - (0x2F939, "M", "𢆟"), - (0x2F93A, "M", "瘐"), - (0x2F93B, "M", "𤾡"), - (0x2F93C, "M", "𤾸"), - (0x2F93D, "M", "𥁄"), - (0x2F93E, "M", "㿼"), - (0x2F93F, "M", "䀈"), - (0x2F940, "M", "直"), - (0x2F941, "M", "𥃳"), - ] - - -def _seg_81() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F942, "M", "𥃲"), - (0x2F943, "M", "𥄙"), - (0x2F944, "M", "𥄳"), - (0x2F945, "M", "眞"), - (0x2F946, "M", "真"), - (0x2F948, "M", "睊"), - (0x2F949, "M", "䀹"), - (0x2F94A, "M", "瞋"), - (0x2F94B, "M", "䁆"), - (0x2F94C, "M", "䂖"), - (0x2F94D, "M", "𥐝"), - (0x2F94E, "M", "硎"), - (0x2F94F, "M", "碌"), - (0x2F950, "M", "磌"), - (0x2F951, "M", "䃣"), - (0x2F952, "M", "𥘦"), - (0x2F953, "M", "祖"), - (0x2F954, "M", "𥚚"), - (0x2F955, "M", "𥛅"), - (0x2F956, "M", "福"), - (0x2F957, "M", "秫"), - (0x2F958, "M", "䄯"), - (0x2F959, "M", "穀"), - (0x2F95A, "M", "穊"), - (0x2F95B, "M", "穏"), - (0x2F95C, "M", "𥥼"), - (0x2F95D, "M", "𥪧"), - (0x2F95F, "M", "竮"), - (0x2F960, "M", "䈂"), - (0x2F961, "M", "𥮫"), - (0x2F962, "M", "篆"), - (0x2F963, "M", "築"), - (0x2F964, "M", "䈧"), - (0x2F965, "M", "𥲀"), - (0x2F966, "M", "糒"), - (0x2F967, "M", "䊠"), - (0x2F968, "M", "糨"), - (0x2F969, "M", "糣"), - (0x2F96A, "M", "紀"), - (0x2F96B, "M", "𥾆"), - (0x2F96C, "M", "絣"), - (0x2F96D, "M", "䌁"), - (0x2F96E, "M", "緇"), - (0x2F96F, "M", "縂"), - (0x2F970, "M", "繅"), - (0x2F971, "M", "䌴"), - (0x2F972, "M", "𦈨"), - (0x2F973, "M", "𦉇"), - (0x2F974, "M", "䍙"), - (0x2F975, "M", "𦋙"), - (0x2F976, "M", "罺"), - (0x2F977, "M", "𦌾"), - (0x2F978, "M", "羕"), - (0x2F979, "M", "翺"), - (0x2F97A, "M", "者"), - (0x2F97B, "M", "𦓚"), - (0x2F97C, "M", "𦔣"), - (0x2F97D, "M", "聠"), - (0x2F97E, "M", "𦖨"), - (0x2F97F, "M", "聰"), - (0x2F980, "M", "𣍟"), - (0x2F981, "M", "䏕"), - (0x2F982, "M", "育"), - (0x2F983, "M", "脃"), - (0x2F984, "M", "䐋"), - (0x2F985, "M", "脾"), - (0x2F986, "M", "媵"), - (0x2F987, "M", "𦞧"), - (0x2F988, "M", "𦞵"), - (0x2F989, "M", "𣎓"), - (0x2F98A, "M", "𣎜"), - (0x2F98B, "M", "舁"), - (0x2F98C, "M", "舄"), - (0x2F98D, "M", "辞"), - (0x2F98E, "M", "䑫"), - (0x2F98F, "M", "芑"), - (0x2F990, "M", "芋"), - (0x2F991, "M", "芝"), - (0x2F992, "M", "劳"), - (0x2F993, "M", "花"), - (0x2F994, "M", "芳"), - (0x2F995, "M", "芽"), - (0x2F996, "M", "苦"), - (0x2F997, "M", "𦬼"), - (0x2F998, "M", "若"), - (0x2F999, "M", "茝"), - (0x2F99A, "M", "荣"), - (0x2F99B, "M", "莭"), - (0x2F99C, "M", "茣"), - (0x2F99D, "M", "莽"), - (0x2F99E, "M", "菧"), - (0x2F99F, "M", "著"), - (0x2F9A0, "M", "荓"), - (0x2F9A1, "M", "菊"), - (0x2F9A2, "M", "菌"), - (0x2F9A3, "M", "菜"), - (0x2F9A4, "M", "𦰶"), - (0x2F9A5, "M", "𦵫"), - (0x2F9A6, "M", "𦳕"), - (0x2F9A7, "M", "䔫"), - ] - - -def _seg_82() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2F9A8, "M", "蓱"), - (0x2F9A9, "M", "蓳"), - (0x2F9AA, "M", "蔖"), - (0x2F9AB, "M", "𧏊"), - (0x2F9AC, "M", "蕤"), - (0x2F9AD, "M", "𦼬"), - (0x2F9AE, "M", "䕝"), - (0x2F9AF, "M", "䕡"), - (0x2F9B0, "M", "𦾱"), - (0x2F9B1, "M", "𧃒"), - (0x2F9B2, "M", "䕫"), - (0x2F9B3, "M", "虐"), - (0x2F9B4, "M", "虜"), - (0x2F9B5, "M", "虧"), - (0x2F9B6, "M", "虩"), - (0x2F9B7, "M", "蚩"), - (0x2F9B8, "M", "蚈"), - (0x2F9B9, "M", "蜎"), - (0x2F9BA, "M", "蛢"), - (0x2F9BB, "M", "蝹"), - (0x2F9BC, "M", "蜨"), - (0x2F9BD, "M", "蝫"), - (0x2F9BE, "M", "螆"), - (0x2F9BF, "M", "䗗"), - (0x2F9C0, "M", "蟡"), - (0x2F9C1, "M", "蠁"), - (0x2F9C2, "M", "䗹"), - (0x2F9C3, "M", "衠"), - (0x2F9C4, "M", "衣"), - (0x2F9C5, "M", "𧙧"), - (0x2F9C6, "M", "裗"), - (0x2F9C7, "M", "裞"), - (0x2F9C8, "M", "䘵"), - (0x2F9C9, "M", "裺"), - (0x2F9CA, "M", "㒻"), - (0x2F9CB, "M", "𧢮"), - (0x2F9CC, "M", "𧥦"), - (0x2F9CD, "M", "䚾"), - (0x2F9CE, "M", "䛇"), - (0x2F9CF, "M", "誠"), - (0x2F9D0, "M", "諭"), - (0x2F9D1, "M", "變"), - (0x2F9D2, "M", "豕"), - (0x2F9D3, "M", "𧲨"), - (0x2F9D4, "M", "貫"), - (0x2F9D5, "M", "賁"), - (0x2F9D6, "M", "贛"), - (0x2F9D7, "M", "起"), - (0x2F9D8, "M", "𧼯"), - (0x2F9D9, "M", "𠠄"), - (0x2F9DA, "M", "跋"), - (0x2F9DB, "M", "趼"), - (0x2F9DC, "M", "跰"), - (0x2F9DD, "M", "𠣞"), - (0x2F9DE, "M", "軔"), - (0x2F9DF, "M", "輸"), - (0x2F9E0, "M", "𨗒"), - (0x2F9E1, "M", "𨗭"), - (0x2F9E2, "M", "邔"), - (0x2F9E3, "M", "郱"), - (0x2F9E4, "M", "鄑"), - (0x2F9E5, "M", "𨜮"), - (0x2F9E6, "M", "鄛"), - (0x2F9E7, "M", "鈸"), - (0x2F9E8, "M", "鋗"), - (0x2F9E9, "M", "鋘"), - (0x2F9EA, "M", "鉼"), - (0x2F9EB, "M", "鏹"), - (0x2F9EC, "M", "鐕"), - (0x2F9ED, "M", "𨯺"), - (0x2F9EE, "M", "開"), - (0x2F9EF, "M", "䦕"), - (0x2F9F0, "M", "閷"), - (0x2F9F1, "M", "𨵷"), - (0x2F9F2, "M", "䧦"), - (0x2F9F3, "M", "雃"), - (0x2F9F4, "M", "嶲"), - (0x2F9F5, "M", "霣"), - (0x2F9F6, "M", "𩅅"), - (0x2F9F7, "M", "𩈚"), - (0x2F9F8, "M", "䩮"), - (0x2F9F9, "M", "䩶"), - (0x2F9FA, "M", "韠"), - (0x2F9FB, "M", "𩐊"), - (0x2F9FC, "M", "䪲"), - (0x2F9FD, "M", "𩒖"), - (0x2F9FE, "M", "頋"), - (0x2FA00, "M", "頩"), - (0x2FA01, "M", "𩖶"), - (0x2FA02, "M", "飢"), - (0x2FA03, "M", "䬳"), - (0x2FA04, "M", "餩"), - (0x2FA05, "M", "馧"), - (0x2FA06, "M", "駂"), - (0x2FA07, "M", "駾"), - (0x2FA08, "M", "䯎"), - (0x2FA09, "M", "𩬰"), - (0x2FA0A, "M", "鬒"), - (0x2FA0B, "M", "鱀"), - (0x2FA0C, "M", "鳽"), - ] - - -def _seg_83() -> List[Union[Tuple[int, str], Tuple[int, str, str]]]: - return [ - (0x2FA0D, "M", "䳎"), - (0x2FA0E, "M", "䳭"), - (0x2FA0F, "M", "鵧"), - (0x2FA10, "M", "𪃎"), - (0x2FA11, "M", "䳸"), - (0x2FA12, "M", "𪄅"), - (0x2FA13, "M", "𪈎"), - (0x2FA14, "M", "𪊑"), - (0x2FA15, "M", "麻"), - (0x2FA16, "M", "䵖"), - (0x2FA17, "M", "黹"), - (0x2FA18, "M", "黾"), - (0x2FA19, "M", "鼅"), - (0x2FA1A, "M", "鼏"), - (0x2FA1B, "M", "鼖"), - (0x2FA1C, "M", "鼻"), - (0x2FA1D, "M", "𪘀"), - (0x2FA1E, "X"), - (0x30000, "V"), - (0x3134B, "X"), - (0x31350, "V"), - (0x323B0, "X"), - (0xE0100, "I"), - (0xE01F0, "X"), - ] - - -uts46data = tuple( - _seg_0() - + _seg_1() - + _seg_2() - + _seg_3() - + _seg_4() - + _seg_5() - + _seg_6() - + _seg_7() - + _seg_8() - + _seg_9() - + _seg_10() - + _seg_11() - + _seg_12() - + _seg_13() - + _seg_14() - + _seg_15() - + _seg_16() - + _seg_17() - + _seg_18() - + _seg_19() - + _seg_20() - + _seg_21() - + _seg_22() - + _seg_23() - + _seg_24() - + _seg_25() - + _seg_26() - + _seg_27() - + _seg_28() - + _seg_29() - + _seg_30() - + _seg_31() - + _seg_32() - + _seg_33() - + _seg_34() - + _seg_35() - + _seg_36() - + _seg_37() - + _seg_38() - + _seg_39() - + _seg_40() - + _seg_41() - + _seg_42() - + _seg_43() - + _seg_44() - + _seg_45() - + _seg_46() - + _seg_47() - + _seg_48() - + _seg_49() - + _seg_50() - + _seg_51() - + _seg_52() - + _seg_53() - + _seg_54() - + _seg_55() - + _seg_56() - + _seg_57() - + _seg_58() - + _seg_59() - + _seg_60() - + _seg_61() - + _seg_62() - + _seg_63() - + _seg_64() - + _seg_65() - + _seg_66() - + _seg_67() - + _seg_68() - + _seg_69() - + _seg_70() - + _seg_71() - + _seg_72() - + _seg_73() - + _seg_74() - + _seg_75() - + _seg_76() - + _seg_77() - + _seg_78() - + _seg_79() - + _seg_80() - + _seg_81() - + _seg_82() - + _seg_83() -) # type: Tuple[Union[Tuple[int, str], Tuple[int, str, str]], ...] diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/AUTHORS.txt b/.env/Lib/site-packages/pip-24.2.dist-info/AUTHORS.txt deleted file mode 100644 index dda2ac3..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/AUTHORS.txt +++ /dev/null @@ -1,796 +0,0 @@ -@Switch01 -A_Rog -Aakanksha Agrawal -Abhinav Sagar -ABHYUDAY PRATAP SINGH -abs51295 -AceGentile -Adam Chainz -Adam Tse -Adam Wentz -admin -Adolfo Ochagavía -Adrien Morison -Agus -ahayrapetyan -Ahilya -AinsworthK -Akash Srivastava -Alan Yee -Albert Tugushev -Albert-Guan -albertg -Alberto Sottile -Aleks Bunin -Ales Erjavec -Alethea Flowers -Alex Gaynor -Alex Grönholm -Alex Hedges -Alex Loosley -Alex Morega -Alex Stachowiak -Alexander Shtyrov -Alexandre Conrad -Alexey Popravka -Aleš Erjavec -Alli -Ami Fischman -Ananya Maiti -Anatoly Techtonik -Anders Kaseorg -Andre Aguiar -Andreas Lutro -Andrei Geacar -Andrew Gaul -Andrew Shymanel -Andrey Bienkowski -Andrey Bulgakov -Andrés Delfino -Andy Freeland -Andy Kluger -Ani Hayrapetyan -Aniruddha Basak -Anish Tambe -Anrs Hu -Anthony Sottile -Antoine Musso -Anton Ovchinnikov -Anton Patrushev -Antonio Alvarado Hernandez -Antony Lee -Antti Kaihola -Anubhav Patel -Anudit Nagar -Anuj Godase -AQNOUCH Mohammed -AraHaan -arena -arenasys -Arindam Choudhury -Armin Ronacher -Arnon Yaari -Artem -Arun Babu Neelicattu -Ashley Manton -Ashwin Ramaswami -atse -Atsushi Odagiri -Avinash Karhana -Avner Cohen -Awit (Ah-Wit) Ghirmai -Baptiste Mispelon -Barney Gale -barneygale -Bartek Ogryczak -Bastian Venthur -Ben Bodenmiller -Ben Darnell -Ben Hoyt -Ben Mares -Ben Rosser -Bence Nagy -Benjamin Peterson -Benjamin VanEvery -Benoit Pierre -Berker Peksag -Bernard -Bernard Tyers -Bernardo B. Marques -Bernhard M. Wiedemann -Bertil Hatt -Bhavam Vidyarthi -Blazej Michalik -Bogdan Opanchuk -BorisZZZ -Brad Erickson -Bradley Ayers -Branch Vincent -Brandon L. Reiss -Brandt Bucher -Brannon Dorsey -Brett Randall -Brett Rosen -Brian Cristante -Brian Rosner -briantracy -BrownTruck -Bruno Oliveira -Bruno Renié -Bruno S -Bstrdsmkr -Buck Golemon -burrows -Bussonnier Matthias -bwoodsend -c22 -Caleb Martinez -Calvin Smith -Carl Meyer -Carlos Liam -Carol Willing -Carter Thayer -Cass -Chandrasekhar Atina -Charlie Marsh -Chih-Hsuan Yen -Chris Brinker -Chris Hunt -Chris Jerdonek -Chris Kuehl -Chris Markiewicz -Chris McDonough -Chris Pawley -Chris Pryer -Chris Wolfe -Christian Clauss -Christian Heimes -Christian Oudard -Christoph Reiter -Christopher Hunt -Christopher Snyder -chrysle -cjc7373 -Clark Boylan -Claudio Jolowicz -Clay McClure -Cody -Cody Soyland -Colin Watson -Collin Anderson -Connor Osborn -Cooper Lees -Cooper Ry Lees -Cory Benfield -Cory Wright -Craig Kerstiens -Cristian Sorinel -Cristina -Cristina Muñoz -ctg123 -Curtis Doty -cytolentino -Daan De Meyer -Dale -Damian -Damian Quiroga -Damian Shaw -Dan Black -Dan Savilonis -Dan Sully -Dane Hillard -daniel -Daniel Collins -Daniel Hahler -Daniel Holth -Daniel Jost -Daniel Katz -Daniel Shaulov -Daniele Esposti -Daniele Nicolodi -Daniele Procida -Daniil Konovalenko -Danny Hermes -Danny McClanahan -Darren Kavanagh -Dav Clark -Dave Abrahams -Dave Jones -David Aguilar -David Black -David Bordeynik -David Caro -David D Lowe -David Evans -David Hewitt -David Linke -David Poggi -David Poznik -David Pursehouse -David Runge -David Tucker -David Wales -Davidovich -ddelange -Deepak Sharma -Deepyaman Datta -Denise Yu -dependabot[bot] -derwolfe -Desetude -Devesh Kumar Singh -devsagul -Diego Caraballo -Diego Ramirez -DiegoCaraballo -Dimitri Merejkowsky -Dimitri Papadopoulos -Dirk Stolle -Dmitry Gladkov -Dmitry Volodin -Domen Kožar -Dominic Davis-Foster -Donald Stufft -Dongweiming -doron zarhi -Dos Moonen -Douglas Thor -DrFeathers -Dustin Ingram -Dustin Rodrigues -Dwayne Bailey -Ed Morley -Edgar Ramírez -Edgar Ramírez Mondragón -Ee Durbin -Efflam Lemaillet -efflamlemaillet -Eitan Adler -ekristina -elainechan -Eli Schwartz -Elisha Hollander -Ellen Marie Dash -Emil Burzo -Emil Styrke -Emmanuel Arias -Endoh Takanao -enoch -Erdinc Mutlu -Eric Cousineau -Eric Gillingham -Eric Hanchrow -Eric Hopper -Erik M. Bray -Erik Rose -Erwin Janssen -Eugene Vereshchagin -everdimension -Federico -Felipe Peter -Felix Yan -fiber-space -Filip Kokosiński -Filipe Laíns -Finn Womack -finnagin -Flavio Amurrio -Florian Briand -Florian Rathgeber -Francesco -Francesco Montesano -Fredrik Orderud -Frost Ming -Gabriel Curio -Gabriel de Perthuis -Garry Polley -gavin -gdanielson -Geoffrey Sneddon -George Song -Georgi Valkov -Georgy Pchelkin -ghost -Giftlin Rajaiah -gizmoguy1 -gkdoc -Godefroid Chapelle -Gopinath M -GOTO Hayato -gousaiyang -gpiks -Greg Roodt -Greg Ward -Guilherme Espada -Guillaume Seguin -gutsytechster -Guy Rozendorn -Guy Tuval -gzpan123 -Hanjun Kim -Hari Charan -Harsh Vardhan -harupy -Harutaka Kawamura -hauntsaninja -Henrich Hartzer -Henry Schreiner -Herbert Pfennig -Holly Stotelmyer -Honnix -Hsiaoming Yang -Hugo Lopes Tavares -Hugo van Kemenade -Hugues Bruant -Hynek Schlawack -Ian Bicking -Ian Cordasco -Ian Lee -Ian Stapleton Cordasco -Ian Wienand -Igor Kuzmitshov -Igor Sobreira -Ikko Ashimine -Ilan Schnell -Illia Volochii -Ilya Baryshev -Inada Naoki -Ionel Cristian Mărieș -Ionel Maries Cristian -Itamar Turner-Trauring -Ivan Pozdeev -J. Nick Koston -Jacob Kim -Jacob Walls -Jaime Sanz -jakirkham -Jakub Kuczys -Jakub Stasiak -Jakub Vysoky -Jakub Wilk -James Cleveland -James Curtin -James Firth -James Gerity -James Polley -Jan Pokorný -Jannis Leidel -Jarek Potiuk -jarondl -Jason Curtis -Jason R. Coombs -JasonMo -JasonMo1 -Jay Graves -Jean Abou Samra -Jean-Christophe Fillion-Robin -Jeff Barber -Jeff Dairiki -Jeff Widman -Jelmer Vernooij -jenix21 -Jeremy Fleischman -Jeremy Stanley -Jeremy Zafran -Jesse Rittner -Jiashuo Li -Jim Fisher -Jim Garrison -Jinzhe Zeng -Jiun Bae -Jivan Amara -Joe Bylund -Joe Michelini -John Paton -John Sirois -John T. Wodder II -John-Scott Atlakson -johnthagen -Jon Banafato -Jon Dufresne -Jon Parise -Jonas Nockert -Jonathan Herbert -Joonatan Partanen -Joost Molenaar -Jorge Niedbalski -Joseph Bylund -Joseph Long -Josh Bronson -Josh Cannon -Josh Hansen -Josh Schneier -Joshua -Juan Luis Cano Rodríguez -Juanjo Bazán -Judah Rand -Julian Berman -Julian Gethmann -Julien Demoor -Jussi Kukkonen -jwg4 -Jyrki Pulliainen -Kai Chen -Kai Mueller -Kamal Bin Mustafa -kasium -kaustav haldar -keanemind -Keith Maxwell -Kelsey Hightower -Kenneth Belitzky -Kenneth Reitz -Kevin Burke -Kevin Carter -Kevin Frommelt -Kevin R Patterson -Kexuan Sun -Kit Randel -Klaas van Schelven -KOLANICH -konstin -kpinc -Krishna Oza -Kumar McMillan -Kuntal Majumder -Kurt McKee -Kyle Persohn -lakshmanaram -Laszlo Kiss-Kollar -Laurent Bristiel -Laurent LAPORTE -Laurie O -Laurie Opperman -layday -Leon Sasson -Lev Givon -Lincoln de Sousa -Lipis -lorddavidiii -Loren Carvalho -Lucas Cimon -Ludovic Gasc -Luis Medel -Lukas Geiger -Lukas Juhrich -Luke Macken -Luo Jiebin -luojiebin -luz.paz -László Kiss Kollár -M00nL1ght -Marc Abramowitz -Marc Tamlyn -Marcus Smith -Mariatta -Mark Kohler -Mark McLoughlin -Mark Williams -Markus Hametner -Martey Dodoo -Martin Fischer -Martin Häcker -Martin Pavlasek -Masaki -Masklinn -Matej Stuchlik -Mathew Jennings -Mathieu Bridon -Mathieu Kniewallner -Matt Bacchi -Matt Good -Matt Maker -Matt Robenolt -Matt Wozniski -matthew -Matthew Einhorn -Matthew Feickert -Matthew Gilliard -Matthew Hughes -Matthew Iversen -Matthew Treinish -Matthew Trumbell -Matthew Willson -Matthias Bussonnier -mattip -Maurits van Rees -Max W Chase -Maxim Kurnikov -Maxime Rouyrre -mayeut -mbaluna -mdebi -memoselyk -meowmeowcat -Michael -Michael Aquilina -Michael E. Karpeles -Michael Klich -Michael Mintz -Michael Williamson -michaelpacer -Michał Górny -Mickaël Schoentgen -Miguel Araujo Perez -Mihir Singh -Mike -Mike Hendricks -Min RK -MinRK -Miro Hrončok -Monica Baluna -montefra -Monty Taylor -morotti -mrKazzila -Muha Ajjan -Nadav Wexler -Nahuel Ambrosini -Nate Coraor -Nate Prewitt -Nathan Houghton -Nathaniel J. Smith -Nehal J Wani -Neil Botelho -Nguyễn Gia Phong -Nicholas Serra -Nick Coghlan -Nick Stenning -Nick Timkovich -Nicolas Bock -Nicole Harris -Nikhil Benesch -Nikhil Ladha -Nikita Chepanov -Nikolay Korolev -Nipunn Koorapati -Nitesh Sharma -Niyas Sait -Noah -Noah Gorny -Nowell Strite -NtaleGrey -nvdv -OBITORASU -Ofek Lev -ofrinevo -Oliver Freund -Oliver Jeeves -Oliver Mannion -Oliver Tonnhofer -Olivier Girardot -Olivier Grisel -Ollie Rutherfurd -OMOTO Kenji -Omry Yadan -onlinejudge95 -Oren Held -Oscar Benjamin -Oz N Tiram -Pachwenko -Patrick Dubroy -Patrick Jenkins -Patrick Lawson -patricktokeeffe -Patrik Kopkan -Paul Ganssle -Paul Kehrer -Paul Moore -Paul Nasrat -Paul Oswald -Paul van der Linden -Paulus Schoutsen -Pavel Safronov -Pavithra Eswaramoorthy -Pawel Jasinski -Paweł Szramowski -Pekka Klärck -Peter Gessler -Peter Lisák -Peter Shen -Peter Waller -Petr Viktorin -petr-tik -Phaneendra Chiruvella -Phil Elson -Phil Freo -Phil Pennock -Phil Whelan -Philip Jägenstedt -Philip Molloy -Philippe Ombredanne -Pi Delport -Pierre-Yves Rofes -Pieter Degroote -pip -Prabakaran Kumaresshan -Prabhjyotsing Surjit Singh Sodhi -Prabhu Marappan -Pradyun Gedam -Prashant Sharma -Pratik Mallya -pre-commit-ci[bot] -Preet Thakkar -Preston Holmes -Przemek Wrzos -Pulkit Goyal -q0w -Qiangning Hong -Qiming Xu -Quentin Lee -Quentin Pradet -R. David Murray -Rafael Caricio -Ralf Schmitt -Ran Benita -Razzi Abuissa -rdb -Reece Dunham -Remi Rampin -Rene Dudfield -Riccardo Magliocchetti -Riccardo Schirone -Richard Jones -Richard Si -Ricky Ng-Adam -Rishi -rmorotti -RobberPhex -Robert Collins -Robert McGibbon -Robert Pollak -Robert T. McGibbon -robin elisha robinson -Roey Berman -Rohan Jain -Roman Bogorodskiy -Roman Donchenko -Romuald Brunet -ronaudinho -Ronny Pfannschmidt -Rory McCann -Ross Brattain -Roy Wellington Ⅳ -Ruairidh MacLeod -Russell Keith-Magee -Ryan Shepherd -Ryan Wooden -ryneeverett -S. Guliaev -Sachi King -Salvatore Rinchiera -sandeepkiran-js -Sander Van Balen -Savio Jomton -schlamar -Scott Kitterman -Sean -seanj -Sebastian Jordan -Sebastian Schaetz -Segev Finer -SeongSoo Cho -Sergey Vasilyev -Seth Michael Larson -Seth Woodworth -Shahar Epstein -Shantanu -shenxianpeng -shireenrao -Shivansh-007 -Shixian Sheng -Shlomi Fish -Shovan Maity -Simeon Visser -Simon Cross -Simon Pichugin -sinoroc -sinscary -snook92 -socketubs -Sorin Sbarnea -Srinivas Nyayapati -Stavros Korokithakis -Stefan Scherfke -Stefano Rivera -Stephan Erb -Stephen Rosen -stepshal -Steve (Gadget) Barnes -Steve Barnes -Steve Dower -Steve Kowalik -Steven Myint -Steven Silvester -stonebig -studioj -Stéphane Bidoul -Stéphane Bidoul (ACSONE) -Stéphane Klein -Sumana Harihareswara -Surbhi Sharma -Sviatoslav Sydorenko -Sviatoslav Sydorenko (Святослав Сидоренко) -Swat009 -Sylvain -Takayuki SHIMIZUKAWA -Taneli Hukkinen -tbeswick -Thiago -Thijs Triemstra -Thomas Fenzl -Thomas Grainger -Thomas Guettler -Thomas Johansson -Thomas Kluyver -Thomas Smith -Thomas VINCENT -Tim D. Smith -Tim Gates -Tim Harder -Tim Heap -tim smith -tinruufu -Tobias Hermann -Tom Forbes -Tom Freudenheim -Tom V -Tomas Hrnciar -Tomas Orsava -Tomer Chachamu -Tommi Enenkel | AnB -Tomáš Hrnčiar -Tony Beswick -Tony Narlock -Tony Zhaocheng Tan -TonyBeswick -toonarmycaptain -Toshio Kuratomi -toxinu -Travis Swicegood -Tushar Sadhwani -Tzu-ping Chung -Valentin Haenel -Victor Stinner -victorvpaulo -Vikram - Google -Viktor Szépe -Ville Skyttä -Vinay Sajip -Vincent Philippon -Vinicyus Macedo -Vipul Kumar -Vitaly Babiy -Vladimir Fokow -Vladimir Rutsky -W. Trevor King -Wil Tan -Wilfred Hughes -William Edwards -William ML Leslie -William T Olson -William Woodruff -Wilson Mo -wim glenn -Winson Luk -Wolfgang Maier -Wu Zhenyu -XAMES3 -Xavier Fernandez -Xianpeng Shen -xoviat -xtreak -YAMAMOTO Takashi -Yen Chi Hsuan -Yeray Diaz Diaz -Yoval P -Yu Jian -Yuan Jing Vincent Yan -Yusuke Hayashi -Zearin -Zhiping Deng -ziebam -Zvezdan Petkovic -Łukasz Langa -Роман Донченко -Семён Марьясин diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/INSTALLER b/.env/Lib/site-packages/pip-24.2.dist-info/INSTALLER deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/INSTALLER +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/LICENSE.txt b/.env/Lib/site-packages/pip-24.2.dist-info/LICENSE.txt deleted file mode 100644 index 8e7b65e..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/LICENSE.txt +++ /dev/null @@ -1,20 +0,0 @@ -Copyright (c) 2008-present The pip developers (see AUTHORS.txt file) - -Permission is hereby granted, free of charge, to any person obtaining -a copy of this software and associated documentation files (the -"Software"), to deal in the Software without restriction, including -without limitation the rights to use, copy, modify, merge, publish, -distribute, sublicense, and/or sell copies of the Software, and to -permit persons to whom the Software is furnished to do so, subject to -the following conditions: - -The above copyright notice and this permission notice shall be -included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/METADATA b/.env/Lib/site-packages/pip-24.2.dist-info/METADATA deleted file mode 100644 index 6141107..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/METADATA +++ /dev/null @@ -1,89 +0,0 @@ -Metadata-Version: 2.1 -Name: pip -Version: 24.2 -Summary: The PyPA recommended tool for installing Python packages. -Author-email: The pip developers -License: MIT -Project-URL: Homepage, https://pip.pypa.io/ -Project-URL: Documentation, https://pip.pypa.io -Project-URL: Source, https://github.com/pypa/pip -Project-URL: Changelog, https://pip.pypa.io/en/stable/news/ -Classifier: Development Status :: 5 - Production/Stable -Classifier: Intended Audience :: Developers -Classifier: License :: OSI Approved :: MIT License -Classifier: Topic :: Software Development :: Build Tools -Classifier: Programming Language :: Python -Classifier: Programming Language :: Python :: 3 -Classifier: Programming Language :: Python :: 3 :: Only -Classifier: Programming Language :: Python :: 3.8 -Classifier: Programming Language :: Python :: 3.9 -Classifier: Programming Language :: Python :: 3.10 -Classifier: Programming Language :: Python :: 3.11 -Classifier: Programming Language :: Python :: 3.12 -Classifier: Programming Language :: Python :: Implementation :: CPython -Classifier: Programming Language :: Python :: Implementation :: PyPy -Requires-Python: >=3.8 -Description-Content-Type: text/x-rst -License-File: LICENSE.txt -License-File: AUTHORS.txt - -pip - The Python Package Installer -================================== - -.. |pypi-version| image:: https://img.shields.io/pypi/v/pip.svg - :target: https://pypi.org/project/pip/ - :alt: PyPI - -.. |python-versions| image:: https://img.shields.io/pypi/pyversions/pip - :target: https://pypi.org/project/pip - :alt: PyPI - Python Version - -.. |docs-badge| image:: https://readthedocs.org/projects/pip/badge/?version=latest - :target: https://pip.pypa.io/en/latest - :alt: Documentation - -|pypi-version| |python-versions| |docs-badge| - -pip is the `package installer`_ for Python. You can use pip to install packages from the `Python Package Index`_ and other indexes. - -Please take a look at our documentation for how to install and use pip: - -* `Installation`_ -* `Usage`_ - -We release updates regularly, with a new version every 3 months. Find more details in our documentation: - -* `Release notes`_ -* `Release process`_ - -If you find bugs, need help, or want to talk to the developers, please use our mailing lists or chat rooms: - -* `Issue tracking`_ -* `Discourse channel`_ -* `User IRC`_ - -If you want to get involved head over to GitHub to get the source code, look at our development documentation and feel free to jump on the developer mailing lists and chat rooms: - -* `GitHub page`_ -* `Development documentation`_ -* `Development IRC`_ - -Code of Conduct ---------------- - -Everyone interacting in the pip project's codebases, issue trackers, chat -rooms, and mailing lists is expected to follow the `PSF Code of Conduct`_. - -.. _package installer: https://packaging.python.org/guides/tool-recommendations/ -.. _Python Package Index: https://pypi.org -.. _Installation: https://pip.pypa.io/en/stable/installation/ -.. _Usage: https://pip.pypa.io/en/stable/ -.. _Release notes: https://pip.pypa.io/en/stable/news.html -.. _Release process: https://pip.pypa.io/en/latest/development/release-process/ -.. _GitHub page: https://github.com/pypa/pip -.. _Development documentation: https://pip.pypa.io/en/latest/development -.. _Issue tracking: https://github.com/pypa/pip/issues -.. _Discourse channel: https://discuss.python.org/c/packaging -.. _User IRC: https://kiwiirc.com/nextclient/#ircs://irc.libera.chat:+6697/pypa -.. _Development IRC: https://kiwiirc.com/nextclient/#ircs://irc.libera.chat:+6697/pypa-dev -.. _PSF Code of Conduct: https://github.com/pypa/.github/blob/main/CODE_OF_CONDUCT.md diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/RECORD b/.env/Lib/site-packages/pip-24.2.dist-info/RECORD deleted file mode 100644 index a182b2a..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/RECORD +++ /dev/null @@ -1,853 +0,0 @@ -../../Scripts/pip.exe,sha256=bwcDqlhC_hrwBjBOeVSkVZL714U52k23I5GiEtn7I1M,108425 -../../Scripts/pip3.12.exe,sha256=bwcDqlhC_hrwBjBOeVSkVZL714U52k23I5GiEtn7I1M,108425 -../../Scripts/pip3.exe,sha256=bwcDqlhC_hrwBjBOeVSkVZL714U52k23I5GiEtn7I1M,108425 -pip-24.2.dist-info/AUTHORS.txt,sha256=KDa8Pd3GDeKSogF6yFW0l9A9eMneLDOFrcIDqkL8G8s,10868 -pip-24.2.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -pip-24.2.dist-info/LICENSE.txt,sha256=Y0MApmnUmurmWxLGxIySTFGkzfPR_whtw0VtyLyqIQQ,1093 -pip-24.2.dist-info/METADATA,sha256=PhzCxQxIhsnZ871cPUe3Hew9PhhpgflLbfqU3WizZqM,3624 -pip-24.2.dist-info/RECORD,, -pip-24.2.dist-info/REQUESTED,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip-24.2.dist-info/WHEEL,sha256=Wyh-_nZ0DJYolHNn1_hMa4lM7uDedD_RGVwbmTjyItk,91 -pip-24.2.dist-info/entry_points.txt,sha256=eeIjuzfnfR2PrhbjnbzFU6MnSS70kZLxwaHHq6M-bD0,87 -pip-24.2.dist-info/top_level.txt,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4 -pip/__init__.py,sha256=EQxEGXUQIu-9fNJxVEK74ufx_fTk_HpYV9lAbw-WWbs,355 -pip/__main__.py,sha256=WzbhHXTbSE6gBY19mNN9m4s5o_365LOvTYSgqgbdBhE,854 -pip/__pip-runner__.py,sha256=cPPWuJ6NK_k-GzfvlejLFgwzmYUROmpAR6QC3Q-vkXQ,1450 -pip/__pycache__/__init__.cpython-312.pyc,, -pip/__pycache__/__main__.cpython-312.pyc,, -pip/__pycache__/__pip-runner__.cpython-312.pyc,, -pip/_internal/__init__.py,sha256=MfcoOluDZ8QMCFYal04IqOJ9q6m2V7a0aOsnI-WOxUo,513 -pip/_internal/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/__pycache__/build_env.cpython-312.pyc,, -pip/_internal/__pycache__/cache.cpython-312.pyc,, -pip/_internal/__pycache__/configuration.cpython-312.pyc,, -pip/_internal/__pycache__/exceptions.cpython-312.pyc,, -pip/_internal/__pycache__/main.cpython-312.pyc,, -pip/_internal/__pycache__/pyproject.cpython-312.pyc,, -pip/_internal/__pycache__/self_outdated_check.cpython-312.pyc,, -pip/_internal/__pycache__/wheel_builder.cpython-312.pyc,, -pip/_internal/build_env.py,sha256=QiusW8QEaj387y0hdRqVbuelHSHGYcT7WzVckbmMhR0,10420 -pip/_internal/cache.py,sha256=Jb698p5PNigRtpW5o26wQNkkUv4MnQ94mc471wL63A0,10369 -pip/_internal/cli/__init__.py,sha256=FkHBgpxxb-_gd6r1FjnNhfMOzAUYyXoXKJ6abijfcFU,132 -pip/_internal/cli/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/cli/__pycache__/autocompletion.cpython-312.pyc,, -pip/_internal/cli/__pycache__/base_command.cpython-312.pyc,, -pip/_internal/cli/__pycache__/cmdoptions.cpython-312.pyc,, -pip/_internal/cli/__pycache__/command_context.cpython-312.pyc,, -pip/_internal/cli/__pycache__/index_command.cpython-312.pyc,, -pip/_internal/cli/__pycache__/main.cpython-312.pyc,, -pip/_internal/cli/__pycache__/main_parser.cpython-312.pyc,, -pip/_internal/cli/__pycache__/parser.cpython-312.pyc,, -pip/_internal/cli/__pycache__/progress_bars.cpython-312.pyc,, -pip/_internal/cli/__pycache__/req_command.cpython-312.pyc,, -pip/_internal/cli/__pycache__/spinners.cpython-312.pyc,, -pip/_internal/cli/__pycache__/status_codes.cpython-312.pyc,, -pip/_internal/cli/autocompletion.py,sha256=Lli3Mr6aDNu7ZkJJFFvwD2-hFxNI6Avz8OwMyS5TVrs,6865 -pip/_internal/cli/base_command.py,sha256=F8nUcSM-Y-MQljJUe724-yxmc5viFXHyM_zH70NmIh4,8289 -pip/_internal/cli/cmdoptions.py,sha256=mDqBr0d0hoztbRJs-PWtcKpqNAc7khU6ZpoesZKocT8,30110 -pip/_internal/cli/command_context.py,sha256=RHgIPwtObh5KhMrd3YZTkl8zbVG-6Okml7YbFX4Ehg0,774 -pip/_internal/cli/index_command.py,sha256=YIJ84cfYcbDBACnB8eoDgqjYJU6GpiWP2Rh7Ij-Xyak,5633 -pip/_internal/cli/main.py,sha256=BDZef-bWe9g9Jpr4OVs4dDf-845HJsKw835T7AqEnAc,2817 -pip/_internal/cli/main_parser.py,sha256=laDpsuBDl6kyfywp9eMMA9s84jfH2TJJn-vmL0GG90w,4338 -pip/_internal/cli/parser.py,sha256=QAkY6s8N-AD7w5D2PQm2Y8C2MIJSv7iuAeNjOMvDBUA,10811 -pip/_internal/cli/progress_bars.py,sha256=0FAf7eN67KnIv_gZQhTWSnKXXUzQko1ftGXEoLe5Yec,2713 -pip/_internal/cli/req_command.py,sha256=DqeFhmUMs6o6Ev8qawAcOoYNdAZsfyKS0MZI5jsJYwQ,12250 -pip/_internal/cli/spinners.py,sha256=hIJ83GerdFgFCdobIA23Jggetegl_uC4Sp586nzFbPE,5118 -pip/_internal/cli/status_codes.py,sha256=sEFHUaUJbqv8iArL3HAtcztWZmGOFX01hTesSytDEh0,116 -pip/_internal/commands/__init__.py,sha256=5oRO9O3dM2vGuh0bFw4HOVletryrz5HHMmmPWwJrH9U,3882 -pip/_internal/commands/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/commands/__pycache__/cache.cpython-312.pyc,, -pip/_internal/commands/__pycache__/check.cpython-312.pyc,, -pip/_internal/commands/__pycache__/completion.cpython-312.pyc,, -pip/_internal/commands/__pycache__/configuration.cpython-312.pyc,, -pip/_internal/commands/__pycache__/debug.cpython-312.pyc,, -pip/_internal/commands/__pycache__/download.cpython-312.pyc,, -pip/_internal/commands/__pycache__/freeze.cpython-312.pyc,, -pip/_internal/commands/__pycache__/hash.cpython-312.pyc,, -pip/_internal/commands/__pycache__/help.cpython-312.pyc,, -pip/_internal/commands/__pycache__/index.cpython-312.pyc,, -pip/_internal/commands/__pycache__/inspect.cpython-312.pyc,, -pip/_internal/commands/__pycache__/install.cpython-312.pyc,, -pip/_internal/commands/__pycache__/list.cpython-312.pyc,, -pip/_internal/commands/__pycache__/search.cpython-312.pyc,, -pip/_internal/commands/__pycache__/show.cpython-312.pyc,, -pip/_internal/commands/__pycache__/uninstall.cpython-312.pyc,, -pip/_internal/commands/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/commands/cache.py,sha256=xg76_ZFEBC6zoQ3gXLRfMZJft4z2a0RwH4GEFZC6nnU,7944 -pip/_internal/commands/check.py,sha256=Hr_4eiMd9cgVDgEvjtIdw915NmL7ROIWW8enkr8slPQ,2268 -pip/_internal/commands/completion.py,sha256=HT4lD0bgsflHq2IDgYfiEdp7IGGtE7s6MgI3xn0VQEw,4287 -pip/_internal/commands/configuration.py,sha256=n98enwp6y0b5G6fiRQjaZo43FlJKYve_daMhN-4BRNc,9766 -pip/_internal/commands/debug.py,sha256=DNDRgE9YsKrbYzU0s3VKi8rHtKF4X13CJ_br_8PUXO0,6797 -pip/_internal/commands/download.py,sha256=0qB0nys6ZEPsog451lDsjL5Bx7Z97t-B80oFZKhpzKM,5273 -pip/_internal/commands/freeze.py,sha256=2Vt72BYTSm9rzue6d8dNzt8idxWK4Db6Hd-anq7GQ80,3203 -pip/_internal/commands/hash.py,sha256=EVVOuvGtoPEdFi8SNnmdqlCQrhCxV-kJsdwtdcCnXGQ,1703 -pip/_internal/commands/help.py,sha256=gcc6QDkcgHMOuAn5UxaZwAStsRBrnGSn_yxjS57JIoM,1132 -pip/_internal/commands/index.py,sha256=RAXxmJwFhVb5S1BYzb5ifX3sn9Na8v2CCVYwSMP8pao,4731 -pip/_internal/commands/inspect.py,sha256=PGrY9TRTRCM3y5Ml8Bdk8DEOXquWRfscr4DRo1LOTPc,3189 -pip/_internal/commands/install.py,sha256=iqesiLIZc6Op9uihMQFYRhAA2DQRZUxbM4z1BwXoFls,29428 -pip/_internal/commands/list.py,sha256=RgaIV4kN-eMSpgUAXc-6bjnURzl0v3cRE11xr54O9Cg,12771 -pip/_internal/commands/search.py,sha256=hSGtIHg26LRe468Ly7oZ6gfd9KbTxBRZAAtJc9Um6S4,5628 -pip/_internal/commands/show.py,sha256=IG9L5uo8w6UA4tI_IlmaxLCoNKPa5JNJCljj3NWs0OE,7507 -pip/_internal/commands/uninstall.py,sha256=7pOR7enK76gimyxQbzxcG1OsyLXL3DvX939xmM8Fvtg,3892 -pip/_internal/commands/wheel.py,sha256=eJRhr_qoNNxWAkkdJCNiQM7CXd4E1_YyQhsqJnBPGGg,6414 -pip/_internal/configuration.py,sha256=XkAiBS0hpzsM-LF0Qu5hvPWO_Bs67-oQKRYFBuMbESs,14006 -pip/_internal/distributions/__init__.py,sha256=Hq6kt6gXBgjNit5hTTWLAzeCNOKoB-N0pGYSqehrli8,858 -pip/_internal/distributions/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/distributions/__pycache__/base.cpython-312.pyc,, -pip/_internal/distributions/__pycache__/installed.cpython-312.pyc,, -pip/_internal/distributions/__pycache__/sdist.cpython-312.pyc,, -pip/_internal/distributions/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/distributions/base.py,sha256=QeB9qvKXDIjLdPBDE5fMgpfGqMMCr-govnuoQnGuiF8,1783 -pip/_internal/distributions/installed.py,sha256=QinHFbWAQ8oE0pbD8MFZWkwlnfU1QYTccA1vnhrlYOU,842 -pip/_internal/distributions/sdist.py,sha256=PlcP4a6-R6c98XnOM-b6Lkb3rsvh9iG4ok8shaanrzs,6751 -pip/_internal/distributions/wheel.py,sha256=THBYfnv7VVt8mYhMYUtH13S1E7FDwtDyDfmUcl8ai0E,1317 -pip/_internal/exceptions.py,sha256=6qcW3QgmFVlRxlZvDSLUhSzKJ7_Tedo-lyqWA6NfdAU,25371 -pip/_internal/index/__init__.py,sha256=vpt-JeTZefh8a-FC22ZeBSXFVbuBcXSGiILhQZJaNpQ,30 -pip/_internal/index/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/index/__pycache__/collector.cpython-312.pyc,, -pip/_internal/index/__pycache__/package_finder.cpython-312.pyc,, -pip/_internal/index/__pycache__/sources.cpython-312.pyc,, -pip/_internal/index/collector.py,sha256=RdPO0JLAlmyBWPAWYHPyRoGjz3GNAeTngCNkbGey_mE,16265 -pip/_internal/index/package_finder.py,sha256=yRC4xsyudwKnNoU6IXvNoyqYo5ScT7lB6Wa-z2eh7cs,37666 -pip/_internal/index/sources.py,sha256=dJegiR9f86kslaAHcv9-R5L_XBf5Rzm_FkyPteDuPxI,8688 -pip/_internal/locations/__init__.py,sha256=UaAxeZ_f93FyouuFf4p7SXYF-4WstXuEvd3LbmPCAno,14925 -pip/_internal/locations/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/locations/__pycache__/_distutils.cpython-312.pyc,, -pip/_internal/locations/__pycache__/_sysconfig.cpython-312.pyc,, -pip/_internal/locations/__pycache__/base.cpython-312.pyc,, -pip/_internal/locations/_distutils.py,sha256=H9ZHK_35rdDV1Qsmi4QeaBULjFT4Mbu6QuoVGkJ6QHI,6009 -pip/_internal/locations/_sysconfig.py,sha256=IGzds60qsFneRogC-oeBaY7bEh3lPt_v47kMJChQXsU,7724 -pip/_internal/locations/base.py,sha256=RQiPi1d4FVM2Bxk04dQhXZ2PqkeljEL2fZZ9SYqIQ78,2556 -pip/_internal/main.py,sha256=r-UnUe8HLo5XFJz8inTcOOTiu_sxNhgHb6VwlGUllOI,340 -pip/_internal/metadata/__init__.py,sha256=9pU3W3s-6HtjFuYhWcLTYVmSaziklPv7k2x8p7X1GmA,4339 -pip/_internal/metadata/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/metadata/__pycache__/_json.cpython-312.pyc,, -pip/_internal/metadata/__pycache__/base.cpython-312.pyc,, -pip/_internal/metadata/__pycache__/pkg_resources.cpython-312.pyc,, -pip/_internal/metadata/_json.py,sha256=P0cAJrH_mtmMZvlZ16ZXm_-izA4lpr5wy08laICuiaA,2644 -pip/_internal/metadata/base.py,sha256=ft0K5XNgI4ETqZnRv2-CtvgYiMOMAeGMAzxT-f6VLJA,25298 -pip/_internal/metadata/importlib/__init__.py,sha256=jUUidoxnHcfITHHaAWG1G2i5fdBYklv_uJcjo2x7VYE,135 -pip/_internal/metadata/importlib/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/metadata/importlib/__pycache__/_compat.cpython-312.pyc,, -pip/_internal/metadata/importlib/__pycache__/_dists.cpython-312.pyc,, -pip/_internal/metadata/importlib/__pycache__/_envs.cpython-312.pyc,, -pip/_internal/metadata/importlib/_compat.py,sha256=c6av8sP8BBjAZuFSJow1iWfygUXNM3xRTCn5nqw6B9M,2796 -pip/_internal/metadata/importlib/_dists.py,sha256=anh0mLI-FYRPUhAdipd0Va3YJJc6HelCKQ0bFhY10a0,8017 -pip/_internal/metadata/importlib/_envs.py,sha256=JHjNfnk9RsjrcQw8dLBqdfBglOKSepEe9aq03B4nRpU,7431 -pip/_internal/metadata/pkg_resources.py,sha256=U07ETAINSGeSRBfWUG93E4tZZbaW_f7PGzEqZN0hulc,10542 -pip/_internal/models/__init__.py,sha256=3DHUd_qxpPozfzouoqa9g9ts1Czr5qaHfFxbnxriepM,63 -pip/_internal/models/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/models/__pycache__/candidate.cpython-312.pyc,, -pip/_internal/models/__pycache__/direct_url.cpython-312.pyc,, -pip/_internal/models/__pycache__/format_control.cpython-312.pyc,, -pip/_internal/models/__pycache__/index.cpython-312.pyc,, -pip/_internal/models/__pycache__/installation_report.cpython-312.pyc,, -pip/_internal/models/__pycache__/link.cpython-312.pyc,, -pip/_internal/models/__pycache__/scheme.cpython-312.pyc,, -pip/_internal/models/__pycache__/search_scope.cpython-312.pyc,, -pip/_internal/models/__pycache__/selection_prefs.cpython-312.pyc,, -pip/_internal/models/__pycache__/target_python.cpython-312.pyc,, -pip/_internal/models/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/models/candidate.py,sha256=zzgFRuw_kWPjKpGw7LC0ZUMD2CQ2EberUIYs8izjdCA,753 -pip/_internal/models/direct_url.py,sha256=uBtY2HHd3TO9cKQJWh0ThvE5FRr-MWRYChRU4IG9HZE,6578 -pip/_internal/models/format_control.py,sha256=wtsQqSK9HaUiNxQEuB-C62eVimw6G4_VQFxV9-_KDBE,2486 -pip/_internal/models/index.py,sha256=tYnL8oxGi4aSNWur0mG8DAP7rC6yuha_MwJO8xw0crI,1030 -pip/_internal/models/installation_report.py,sha256=zRVZoaz-2vsrezj_H3hLOhMZCK9c7TbzWgC-jOalD00,2818 -pip/_internal/models/link.py,sha256=jHax9O-9zlSzEwjBCDkx0OXjKXwBDwOuPwn-PsR8dCs,21034 -pip/_internal/models/scheme.py,sha256=PakmHJM3e8OOWSZFtfz1Az7f1meONJnkGuQxFlt3wBE,575 -pip/_internal/models/search_scope.py,sha256=67NEnsYY84784S-MM7ekQuo9KXLH-7MzFntXjapvAo0,4531 -pip/_internal/models/selection_prefs.py,sha256=qaFfDs3ciqoXPg6xx45N1jPLqccLJw4N0s4P0PyHTQ8,2015 -pip/_internal/models/target_python.py,sha256=2XaH2rZ5ZF-K5wcJbEMGEl7SqrTToDDNkrtQ2v_v_-Q,4271 -pip/_internal/models/wheel.py,sha256=Odc1NVWL5N-i6A3vFa50BfNvCRlGvGa4som60FQM198,3601 -pip/_internal/network/__init__.py,sha256=jf6Tt5nV_7zkARBrKojIXItgejvoegVJVKUbhAa5Ioc,50 -pip/_internal/network/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/network/__pycache__/auth.cpython-312.pyc,, -pip/_internal/network/__pycache__/cache.cpython-312.pyc,, -pip/_internal/network/__pycache__/download.cpython-312.pyc,, -pip/_internal/network/__pycache__/lazy_wheel.cpython-312.pyc,, -pip/_internal/network/__pycache__/session.cpython-312.pyc,, -pip/_internal/network/__pycache__/utils.cpython-312.pyc,, -pip/_internal/network/__pycache__/xmlrpc.cpython-312.pyc,, -pip/_internal/network/auth.py,sha256=D4gASjUrqoDFlSt6gQ767KAAjv6PUyJU0puDlhXNVRE,20809 -pip/_internal/network/cache.py,sha256=48A971qCzKNFvkb57uGEk7-0xaqPS0HWj2711QNTxkU,3935 -pip/_internal/network/download.py,sha256=FLOP29dPYECBiAi7eEjvAbNkyzaKNqbyjOT2m8HPW8U,6048 -pip/_internal/network/lazy_wheel.py,sha256=2PXVduYZPCPZkkQFe1J1GbfHJWeCU--FXonGyIfw9eU,7638 -pip/_internal/network/session.py,sha256=XmanBKjVwPFmh1iJ58q6TDh9xabH37gREuQJ_feuZGA,18741 -pip/_internal/network/utils.py,sha256=Inaxel-NxBu4PQWkjyErdnfewsFCcgHph7dzR1-FboY,4088 -pip/_internal/network/xmlrpc.py,sha256=sAxzOacJ-N1NXGPvap9jC3zuYWSnnv3GXtgR2-E2APA,1838 -pip/_internal/operations/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/operations/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/operations/__pycache__/check.cpython-312.pyc,, -pip/_internal/operations/__pycache__/freeze.cpython-312.pyc,, -pip/_internal/operations/__pycache__/prepare.cpython-312.pyc,, -pip/_internal/operations/build/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/operations/build/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/build_tracker.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/metadata.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/metadata_editable.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/metadata_legacy.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/wheel_editable.cpython-312.pyc,, -pip/_internal/operations/build/__pycache__/wheel_legacy.cpython-312.pyc,, -pip/_internal/operations/build/build_tracker.py,sha256=-ARW_TcjHCOX7D2NUOGntB4Fgc6b4aolsXkAK6BWL7w,4774 -pip/_internal/operations/build/metadata.py,sha256=9S0CUD8U3QqZeXp-Zyt8HxwU90lE4QrnYDgrqZDzBnc,1422 -pip/_internal/operations/build/metadata_editable.py,sha256=VLL7LvntKE8qxdhUdEJhcotFzUsOSI8NNS043xULKew,1474 -pip/_internal/operations/build/metadata_legacy.py,sha256=8i6i1QZX9m_lKPStEFsHKM0MT4a-CD408JOw99daLmo,2190 -pip/_internal/operations/build/wheel.py,sha256=sT12FBLAxDC6wyrDorh8kvcZ1jG5qInCRWzzP-UkJiQ,1075 -pip/_internal/operations/build/wheel_editable.py,sha256=yOtoH6zpAkoKYEUtr8FhzrYnkNHQaQBjWQ2HYae1MQg,1417 -pip/_internal/operations/build/wheel_legacy.py,sha256=K-6kNhmj-1xDF45ny1yheMerF0ui4EoQCLzEoHh6-tc,3045 -pip/_internal/operations/check.py,sha256=L24vRL8VWbyywdoeAhM89WCd8zLTnjIbULlKelUgIec,5912 -pip/_internal/operations/freeze.py,sha256=V59yEyCSz_YhZuhH09-6aV_zvYBMrS_IxFFNqn2QzlA,9864 -pip/_internal/operations/install/__init__.py,sha256=mX7hyD2GNBO2mFGokDQ30r_GXv7Y_PLdtxcUv144e-s,51 -pip/_internal/operations/install/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/operations/install/__pycache__/editable_legacy.cpython-312.pyc,, -pip/_internal/operations/install/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/operations/install/editable_legacy.py,sha256=PoEsNEPGbIZ2yQphPsmYTKLOCMs4gv5OcCdzW124NcA,1283 -pip/_internal/operations/install/wheel.py,sha256=X5Iz9yUg5LlK5VNQ9g2ikc6dcRu8EPi_SUi5iuEDRgo,27615 -pip/_internal/operations/prepare.py,sha256=joWJwPkuqGscQgVNImLK71e9hRapwKvRCM8HclysmvU,28118 -pip/_internal/pyproject.py,sha256=rw4fwlptDp1hZgYoplwbAGwWA32sWQkp7ysf8Ju6iXc,7287 -pip/_internal/req/__init__.py,sha256=HxBFtZy_BbCclLgr26waMtpzYdO5T3vxePvpGAXSt5s,2653 -pip/_internal/req/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/req/__pycache__/constructors.cpython-312.pyc,, -pip/_internal/req/__pycache__/req_file.cpython-312.pyc,, -pip/_internal/req/__pycache__/req_install.cpython-312.pyc,, -pip/_internal/req/__pycache__/req_set.cpython-312.pyc,, -pip/_internal/req/__pycache__/req_uninstall.cpython-312.pyc,, -pip/_internal/req/constructors.py,sha256=qXNZtUqhsXpHxkRaIQhp20_Kz6I88MDKM8SQR9fckIc,18424 -pip/_internal/req/req_file.py,sha256=hnC9Oz-trqGQpuDnCVWqwpJkAvtbCsk7-5k0EWVQhlQ,17687 -pip/_internal/req/req_install.py,sha256=yhT98NGDoAEk03jznTJnYCznzhiMEEA2ocgsUG_dcNU,35788 -pip/_internal/req/req_set.py,sha256=j3esG0s6SzoVReX9rWn4rpYNtyET_fwxbwJPRimvRxo,2858 -pip/_internal/req/req_uninstall.py,sha256=qzDIxJo-OETWqGais7tSMCDcWbATYABT-Tid3ityF0s,23853 -pip/_internal/resolution/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/resolution/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/resolution/__pycache__/base.cpython-312.pyc,, -pip/_internal/resolution/base.py,sha256=qlmh325SBVfvG6Me9gc5Nsh5sdwHBwzHBq6aEXtKsLA,583 -pip/_internal/resolution/legacy/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/resolution/legacy/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/resolution/legacy/__pycache__/resolver.cpython-312.pyc,, -pip/_internal/resolution/legacy/resolver.py,sha256=3HZiJBRd1FTN6jQpI4qRO8-TbLYeIbUTS6PFvXnXs2w,24068 -pip/_internal/resolution/resolvelib/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/resolution/resolvelib/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/base.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/candidates.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/factory.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/found_candidates.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/provider.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/reporter.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/requirements.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/__pycache__/resolver.cpython-312.pyc,, -pip/_internal/resolution/resolvelib/base.py,sha256=DCf669FsqyQY5uqXeePDHQY1e4QO-pBzWH8O0s9-K94,5023 -pip/_internal/resolution/resolvelib/candidates.py,sha256=07CBc85ya3J19XqdvUsLQwtVIxiTYq9km9hbTRh0jb0,19823 -pip/_internal/resolution/resolvelib/factory.py,sha256=mTTq_nG1F9Eq3VnlYPH6Ap-mydcS-mxC5y5L-CLLp80,32459 -pip/_internal/resolution/resolvelib/found_candidates.py,sha256=9hrTyQqFvl9I7Tji79F1AxHv39Qh1rkJ_7deSHSMfQc,6383 -pip/_internal/resolution/resolvelib/provider.py,sha256=bcsFnYvlmtB80cwVdW1fIwgol8ZNr1f1VHyRTkz47SM,9935 -pip/_internal/resolution/resolvelib/reporter.py,sha256=00JtoXEkTlw0-rl_sl54d71avwOsJHt9GGHcrj5Sza0,3168 -pip/_internal/resolution/resolvelib/requirements.py,sha256=7JG4Z72e5Yk4vU0S5ulGvbqTy4FMQGYhY5zQhX9zTtY,8065 -pip/_internal/resolution/resolvelib/resolver.py,sha256=nLJOsVMEVi2gQUVJoUFKMZAeu2f7GRMjGMvNSWyz0Bc,12592 -pip/_internal/self_outdated_check.py,sha256=pkjQixuWyQ1vrVxZAaYD6SSHgXuFUnHZybXEWTkh0S0,8145 -pip/_internal/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_internal/utils/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/utils/__pycache__/_jaraco_text.cpython-312.pyc,, -pip/_internal/utils/__pycache__/_log.cpython-312.pyc,, -pip/_internal/utils/__pycache__/appdirs.cpython-312.pyc,, -pip/_internal/utils/__pycache__/compat.cpython-312.pyc,, -pip/_internal/utils/__pycache__/compatibility_tags.cpython-312.pyc,, -pip/_internal/utils/__pycache__/datetime.cpython-312.pyc,, -pip/_internal/utils/__pycache__/deprecation.cpython-312.pyc,, -pip/_internal/utils/__pycache__/direct_url_helpers.cpython-312.pyc,, -pip/_internal/utils/__pycache__/egg_link.cpython-312.pyc,, -pip/_internal/utils/__pycache__/encoding.cpython-312.pyc,, -pip/_internal/utils/__pycache__/entrypoints.cpython-312.pyc,, -pip/_internal/utils/__pycache__/filesystem.cpython-312.pyc,, -pip/_internal/utils/__pycache__/filetypes.cpython-312.pyc,, -pip/_internal/utils/__pycache__/glibc.cpython-312.pyc,, -pip/_internal/utils/__pycache__/hashes.cpython-312.pyc,, -pip/_internal/utils/__pycache__/logging.cpython-312.pyc,, -pip/_internal/utils/__pycache__/misc.cpython-312.pyc,, -pip/_internal/utils/__pycache__/packaging.cpython-312.pyc,, -pip/_internal/utils/__pycache__/retry.cpython-312.pyc,, -pip/_internal/utils/__pycache__/setuptools_build.cpython-312.pyc,, -pip/_internal/utils/__pycache__/subprocess.cpython-312.pyc,, -pip/_internal/utils/__pycache__/temp_dir.cpython-312.pyc,, -pip/_internal/utils/__pycache__/unpacking.cpython-312.pyc,, -pip/_internal/utils/__pycache__/urls.cpython-312.pyc,, -pip/_internal/utils/__pycache__/virtualenv.cpython-312.pyc,, -pip/_internal/utils/__pycache__/wheel.cpython-312.pyc,, -pip/_internal/utils/_jaraco_text.py,sha256=M15uUPIh5NpP1tdUGBxRau6q1ZAEtI8-XyLEETscFfE,3350 -pip/_internal/utils/_log.py,sha256=-jHLOE_THaZz5BFcCnoSL9EYAtJ0nXem49s9of4jvKw,1015 -pip/_internal/utils/appdirs.py,sha256=swgcTKOm3daLeXTW6v5BUS2Ti2RvEnGRQYH_yDXklAo,1665 -pip/_internal/utils/compat.py,sha256=ckkFveBiYQjRWjkNsajt_oWPS57tJvE8XxoC4OIYgCY,2399 -pip/_internal/utils/compatibility_tags.py,sha256=ydin8QG8BHqYRsPY4OL6cmb44CbqXl1T0xxS97VhHkk,5377 -pip/_internal/utils/datetime.py,sha256=m21Y3wAtQc-ji6Veb6k_M5g6A0ZyFI4egchTdnwh-pQ,242 -pip/_internal/utils/deprecation.py,sha256=k7Qg_UBAaaTdyq82YVARA6D7RmcGTXGv7fnfcgigj4Q,3707 -pip/_internal/utils/direct_url_helpers.py,sha256=r2MRtkVDACv9AGqYODBUC9CjwgtsUU1s68hmgfCJMtA,3196 -pip/_internal/utils/egg_link.py,sha256=0FePZoUYKv4RGQ2t6x7w5Z427wbA_Uo3WZnAkrgsuqo,2463 -pip/_internal/utils/encoding.py,sha256=qqsXDtiwMIjXMEiIVSaOjwH5YmirCaK-dIzb6-XJsL0,1169 -pip/_internal/utils/entrypoints.py,sha256=YlhLTRl2oHBAuqhc-zmL7USS67TPWVHImjeAQHreZTQ,3064 -pip/_internal/utils/filesystem.py,sha256=ajvA-q4ocliW9kPp8Yquh-4vssXbu-UKbo5FV9V4X64,4950 -pip/_internal/utils/filetypes.py,sha256=i8XAQ0eFCog26Fw9yV0Yb1ygAqKYB1w9Cz9n0fj8gZU,716 -pip/_internal/utils/glibc.py,sha256=vUkWq_1pJuzcYNcGKLlQmABoUiisK8noYY1yc8Wq4w4,3734 -pip/_internal/utils/hashes.py,sha256=XGGLL0AG8-RhWnyz87xF6MFZ--BKadHU35D47eApCKI,4972 -pip/_internal/utils/logging.py,sha256=7BFKB1uFjdxD5crM-GtwA5T2qjbQ2LPD-gJDuJeDNTg,11606 -pip/_internal/utils/misc.py,sha256=HR_V97vNTHNzwq01JrnTZtsLLkWAOJ9_EeYfHJZSgDY,23745 -pip/_internal/utils/packaging.py,sha256=iI3LH43lVNR4hWBOqF6lFsZq4aycb2j0UcHlmDmcqUg,2109 -pip/_internal/utils/retry.py,sha256=mhFbykXjhTnZfgzeuy-vl9c8nECnYn_CMtwNJX2tYzQ,1392 -pip/_internal/utils/setuptools_build.py,sha256=ouXpud-jeS8xPyTPsXJ-m34NPvK5os45otAzdSV_IJE,4435 -pip/_internal/utils/subprocess.py,sha256=EsvqSRiSMHF98T8Txmu6NLU3U--MpTTQjtNgKP0P--M,8988 -pip/_internal/utils/temp_dir.py,sha256=5qOXe8M4JeY6vaFQM867d5zkp1bSwMZ-KT5jymmP0Zg,9310 -pip/_internal/utils/unpacking.py,sha256=eyDkSsk4nW8ZfiSjNzJduCznpHyaGHVv3ak_LMGsiEM,11951 -pip/_internal/utils/urls.py,sha256=qceSOZb5lbNDrHNsv7_S4L4Ytszja5NwPKUMnZHbYnM,1599 -pip/_internal/utils/virtualenv.py,sha256=S6f7csYorRpiD6cvn3jISZYc3I8PJC43H5iMFpRAEDU,3456 -pip/_internal/utils/wheel.py,sha256=b442jkydFHjXzDy6cMR7MpzWBJ1Q82hR5F33cmcHV3g,4494 -pip/_internal/vcs/__init__.py,sha256=UAqvzpbi0VbZo3Ub6skEeZAw-ooIZR-zX_WpCbxyCoU,596 -pip/_internal/vcs/__pycache__/__init__.cpython-312.pyc,, -pip/_internal/vcs/__pycache__/bazaar.cpython-312.pyc,, -pip/_internal/vcs/__pycache__/git.cpython-312.pyc,, -pip/_internal/vcs/__pycache__/mercurial.cpython-312.pyc,, -pip/_internal/vcs/__pycache__/subversion.cpython-312.pyc,, -pip/_internal/vcs/__pycache__/versioncontrol.cpython-312.pyc,, -pip/_internal/vcs/bazaar.py,sha256=EKStcQaKpNu0NK4p5Q10Oc4xb3DUxFw024XrJy40bFQ,3528 -pip/_internal/vcs/git.py,sha256=3tpc9LQA_J4IVW5r5NvWaaSeDzcmJOrSFZN0J8vIKfU,18177 -pip/_internal/vcs/mercurial.py,sha256=oULOhzJ2Uie-06d1omkL-_Gc6meGaUkyogvqG9ZCyPs,5249 -pip/_internal/vcs/subversion.py,sha256=ddTugHBqHzV3ebKlU5QXHPN4gUqlyXbOx8q8NgXKvs8,11735 -pip/_internal/vcs/versioncontrol.py,sha256=cvf_-hnTAjQLXJ3d17FMNhQfcO1AcKWUF10tfrYyP-c,22440 -pip/_internal/wheel_builder.py,sha256=DL3A8LKeRj_ACp11WS5wSgASgPFqeyAeXJKdXfmaWXU,11799 -pip/_vendor/__init__.py,sha256=JYuAXvClhInxIrA2FTp5p-uuWVL7WV6-vEpTs46-Qh4,4873 -pip/_vendor/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/__pycache__/typing_extensions.cpython-312.pyc,, -pip/_vendor/cachecontrol/__init__.py,sha256=GiYoagwPEiJ_xR_lbwWGaoCiPtF_rz4isjfjdDAgHU4,676 -pip/_vendor/cachecontrol/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/_cmd.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/adapter.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/cache.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/controller.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/filewrapper.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/heuristics.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/serialize.cpython-312.pyc,, -pip/_vendor/cachecontrol/__pycache__/wrapper.cpython-312.pyc,, -pip/_vendor/cachecontrol/_cmd.py,sha256=iist2EpzJvDVIhMAxXq8iFnTBsiZAd6iplxfmNboNyk,1737 -pip/_vendor/cachecontrol/adapter.py,sha256=fByO_Pd_EOemjWbuocvBWdN85xT0q_TBm2lxS6vD4fk,6355 -pip/_vendor/cachecontrol/cache.py,sha256=OTQj72tUf8C1uEgczdl3Gc8vkldSzsTITKtDGKMx4z8,1952 -pip/_vendor/cachecontrol/caches/__init__.py,sha256=dtrrroK5BnADR1GWjCZ19aZ0tFsMfvFBtLQQU1sp_ag,303 -pip/_vendor/cachecontrol/caches/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/cachecontrol/caches/__pycache__/file_cache.cpython-312.pyc,, -pip/_vendor/cachecontrol/caches/__pycache__/redis_cache.cpython-312.pyc,, -pip/_vendor/cachecontrol/caches/file_cache.py,sha256=9AlmmTJc6cslb6k5z_6q0sGPHVrMj8zv-uWy-simmfE,5406 -pip/_vendor/cachecontrol/caches/redis_cache.py,sha256=9rmqwtYu_ljVkW6_oLqbC7EaX_a8YT_yLuna-eS0dgo,1386 -pip/_vendor/cachecontrol/controller.py,sha256=o-ejGJlBmpKK8QQLyTPJj0t7siU8XVHXuV8MCybCxQ8,18575 -pip/_vendor/cachecontrol/filewrapper.py,sha256=STttGmIPBvZzt2b51dUOwoWX5crcMCpKZOisM3f5BNc,4292 -pip/_vendor/cachecontrol/heuristics.py,sha256=IYe4QmHERWsMvtxNrp920WeaIsaTTyqLB14DSheSbtY,4834 -pip/_vendor/cachecontrol/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/cachecontrol/serialize.py,sha256=HQd2IllQ05HzPkVLMXTF2uX5mjEQjDBkxCqUJUODpZk,5163 -pip/_vendor/cachecontrol/wrapper.py,sha256=hsGc7g8QGQTT-4f8tgz3AM5qwScg6FO0BSdLSRdEvpU,1417 -pip/_vendor/certifi/__init__.py,sha256=LHXz7E80YJYBzCBv6ZyidQ5-ciYSkSebpY2E5OM0l7o,94 -pip/_vendor/certifi/__main__.py,sha256=1k3Cr95vCxxGRGDljrW3wMdpZdL3Nhf0u1n-k2qdsCY,255 -pip/_vendor/certifi/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/certifi/__pycache__/__main__.cpython-312.pyc,, -pip/_vendor/certifi/__pycache__/core.cpython-312.pyc,, -pip/_vendor/certifi/cacert.pem,sha256=SIupYGAr8HzGP073rsEIaS_sQYIPwzKKjj894DgUmu4,291528 -pip/_vendor/certifi/core.py,sha256=2SRT5rIcQChFDbe37BQa-kULxAgJ8qN6l1jfqTp4HIs,4486 -pip/_vendor/certifi/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/distlib/__init__.py,sha256=hJKF7FHoqbmGckncDuEINWo_OYkDNiHODtYXSMcvjcc,625 -pip/_vendor/distlib/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/compat.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/database.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/index.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/locators.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/manifest.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/markers.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/metadata.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/resources.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/scripts.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/util.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/version.cpython-312.pyc,, -pip/_vendor/distlib/__pycache__/wheel.cpython-312.pyc,, -pip/_vendor/distlib/compat.py,sha256=Un-uIBvy02w-D267OG4VEhuddqWgKj9nNkxVltAb75w,41487 -pip/_vendor/distlib/database.py,sha256=0V9Qvs0Vrxa2F_-hLWitIyVyRifJ0pCxyOI-kEOBwsA,51965 -pip/_vendor/distlib/index.py,sha256=lTbw268rRhj8dw1sib3VZ_0EhSGgoJO3FKJzSFMOaeA,20797 -pip/_vendor/distlib/locators.py,sha256=o1r_M86_bRLafSpetmyfX8KRtFu-_Q58abvQrnOSnbA,51767 -pip/_vendor/distlib/manifest.py,sha256=3qfmAmVwxRqU1o23AlfXrQGZzh6g_GGzTAP_Hb9C5zQ,14168 -pip/_vendor/distlib/markers.py,sha256=n3DfOh1yvZ_8EW7atMyoYeZFXjYla0Nz0itQlojCd0A,5268 -pip/_vendor/distlib/metadata.py,sha256=pB9WZ9mBfmQxc9OVIldLS5CjOoQRvKAvUwwQyKwKQtQ,39693 -pip/_vendor/distlib/resources.py,sha256=LwbPksc0A1JMbi6XnuPdMBUn83X7BPuFNWqPGEKI698,10820 -pip/_vendor/distlib/scripts.py,sha256=8_gP9J7_tlNRicnWmPX4ZiDlP5wTwJKDeeg-8_qXUZU,18780 -pip/_vendor/distlib/t32.exe,sha256=a0GV5kCoWsMutvliiCKmIgV98eRZ33wXoS-XrqvJQVs,97792 -pip/_vendor/distlib/t64-arm.exe,sha256=68TAa32V504xVBnufojh0PcenpR3U4wAqTqf-MZqbPw,182784 -pip/_vendor/distlib/t64.exe,sha256=gaYY8hy4fbkHYTTnA4i26ct8IQZzkBG2pRdy0iyuBrc,108032 -pip/_vendor/distlib/util.py,sha256=XSznxEi_i3T20UJuaVc0qXHz5ksGUCW1khYlBprN_QE,67530 -pip/_vendor/distlib/version.py,sha256=9pXkduchve_aN7JG6iL9VTYV_kqNSGoc2Dwl8JuySnQ,23747 -pip/_vendor/distlib/w32.exe,sha256=R4csx3-OGM9kL4aPIzQKRo5TfmRSHZo6QWyLhDhNBks,91648 -pip/_vendor/distlib/w64-arm.exe,sha256=xdyYhKj0WDcVUOCb05blQYvzdYIKMbmJn2SZvzkcey4,168448 -pip/_vendor/distlib/w64.exe,sha256=ejGf-rojoBfXseGLpya6bFTFPWRG21X5KvU8J5iU-K0,101888 -pip/_vendor/distlib/wheel.py,sha256=FVQCve8u-L0QYk5-YTZc7s4WmNQdvjRWTK08KXzZVX4,43958 -pip/_vendor/distro/__init__.py,sha256=2fHjF-SfgPvjyNZ1iHh_wjqWdR_Yo5ODHwZC0jLBPhc,981 -pip/_vendor/distro/__main__.py,sha256=bu9d3TifoKciZFcqRBuygV3GSuThnVD_m2IK4cz96Vs,64 -pip/_vendor/distro/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/distro/__pycache__/__main__.cpython-312.pyc,, -pip/_vendor/distro/__pycache__/distro.cpython-312.pyc,, -pip/_vendor/distro/distro.py,sha256=XqbefacAhDT4zr_trnbA15eY8vdK4GTghgmvUGrEM_4,49430 -pip/_vendor/distro/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/idna/__init__.py,sha256=KJQN1eQBr8iIK5SKrJ47lXvxG0BJ7Lm38W4zT0v_8lk,849 -pip/_vendor/idna/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/codec.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/compat.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/core.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/idnadata.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/intranges.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/package_data.cpython-312.pyc,, -pip/_vendor/idna/__pycache__/uts46data.cpython-312.pyc,, -pip/_vendor/idna/codec.py,sha256=PS6m-XmdST7Wj7J7ulRMakPDt5EBJyYrT3CPtjh-7t4,3426 -pip/_vendor/idna/compat.py,sha256=0_sOEUMT4CVw9doD3vyRhX80X19PwqFoUBs7gWsFME4,321 -pip/_vendor/idna/core.py,sha256=lyhpoe2vulEaB_65xhXmoKgO-xUqFDvcwxu5hpNNO4E,12663 -pip/_vendor/idna/idnadata.py,sha256=dqRwytzkjIHMBa2R1lYvHDwACenZPt8eGVu1Y8UBE-E,78320 -pip/_vendor/idna/intranges.py,sha256=YBr4fRYuWH7kTKS2tXlFjM24ZF1Pdvcir-aywniInqg,1881 -pip/_vendor/idna/package_data.py,sha256=Tkt0KnIeyIlnHddOaz9WSkkislNgokJAuE-p5GorMqo,21 -pip/_vendor/idna/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/idna/uts46data.py,sha256=1KuksWqLuccPXm2uyRVkhfiFLNIhM_H2m4azCcnOqEU,206503 -pip/_vendor/msgpack/__init__.py,sha256=gsMP7JTECZNUSjvOyIbdhNOkpB9Z8BcGwabVGY2UcdQ,1077 -pip/_vendor/msgpack/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/msgpack/__pycache__/exceptions.cpython-312.pyc,, -pip/_vendor/msgpack/__pycache__/ext.cpython-312.pyc,, -pip/_vendor/msgpack/__pycache__/fallback.cpython-312.pyc,, -pip/_vendor/msgpack/exceptions.py,sha256=dCTWei8dpkrMsQDcjQk74ATl9HsIBH0ybt8zOPNqMYc,1081 -pip/_vendor/msgpack/ext.py,sha256=fKp00BqDLjUtZnPd70Llr138zk8JsCuSpJkkZ5S4dt8,5629 -pip/_vendor/msgpack/fallback.py,sha256=wdUWJkWX2gzfRW9BBCTOuIE1Wvrf5PtBtR8ZtY7G_EE,33175 -pip/_vendor/packaging/__init__.py,sha256=dtw2bNmWCQ9WnMoK3bk_elL1svSlikXtLpZhCFIB9SE,496 -pip/_vendor/packaging/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_elffile.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_manylinux.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_musllinux.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_parser.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_structures.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/_tokenizer.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/markers.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/metadata.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/requirements.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/specifiers.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/tags.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/utils.cpython-312.pyc,, -pip/_vendor/packaging/__pycache__/version.cpython-312.pyc,, -pip/_vendor/packaging/_elffile.py,sha256=_LcJW4YNKywYsl4169B2ukKRqwxjxst_8H0FRVQKlz8,3282 -pip/_vendor/packaging/_manylinux.py,sha256=Xo4V0PZz8sbuVCbTni0t1CR0AHeir_7ib4lTmV8scD4,9586 -pip/_vendor/packaging/_musllinux.py,sha256=p9ZqNYiOItGee8KcZFeHF_YcdhVwGHdK6r-8lgixvGQ,2694 -pip/_vendor/packaging/_parser.py,sha256=s_TvTvDNK0NrM2QB3VKThdWFM4Nc0P6JnkObkl3MjpM,10236 -pip/_vendor/packaging/_structures.py,sha256=q3eVNmbWJGG_S0Dit_S3Ao8qQqz_5PYTXFAKBZe5yr4,1431 -pip/_vendor/packaging/_tokenizer.py,sha256=J6v5H7Jzvb-g81xp_2QACKwO7LxHQA6ikryMU7zXwN8,5273 -pip/_vendor/packaging/markers.py,sha256=dWKSqn5Sp-jDmOG-W3GfLHKjwhf1IsznbT71VlBoB5M,10671 -pip/_vendor/packaging/metadata.py,sha256=KINuSkJ12u-SyoKNTy_pHNGAfMUtxNvZ53qA1zAKcKI,32349 -pip/_vendor/packaging/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/packaging/requirements.py,sha256=gYyRSAdbrIyKDY66ugIDUQjRMvxkH2ALioTmX3tnL6o,2947 -pip/_vendor/packaging/specifiers.py,sha256=HfGgfNJRvrzC759gnnoojHyiWs_DYmcw5PEh5jHH-YE,39738 -pip/_vendor/packaging/tags.py,sha256=y8EbheOu9WS7s-MebaXMcHMF-jzsA_C1Lz5XRTiSy4w,18883 -pip/_vendor/packaging/utils.py,sha256=NAdYUwnlAOpkat_RthavX8a07YuVxgGL_vwrx73GSDM,5287 -pip/_vendor/packaging/version.py,sha256=wE4sSVlF-d1H6HFC1vszEe35CwTig_fh4HHIFg95hFE,16210 -pip/_vendor/pkg_resources/__init__.py,sha256=jrhDRbOubP74QuPXxd7U7Po42PH2l-LZ2XfcO7llpZ4,124463 -pip/_vendor/pkg_resources/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/platformdirs/__init__.py,sha256=FTA6LGNm40GwNZt3gG3uLAacWvf2E_2HTmH0rAALGR8,22285 -pip/_vendor/platformdirs/__main__.py,sha256=jBJ8zb7Mpx5ebcqF83xrpO94MaeCpNGHVf9cvDN2JLg,1505 -pip/_vendor/platformdirs/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/__main__.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/android.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/api.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/macos.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/unix.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/version.cpython-312.pyc,, -pip/_vendor/platformdirs/__pycache__/windows.cpython-312.pyc,, -pip/_vendor/platformdirs/android.py,sha256=xZXY9Jd46WOsxT2U6-5HsNtDZ-IQqxcEUrBLl3hYk4o,9016 -pip/_vendor/platformdirs/api.py,sha256=QBYdUac2eC521ek_y53uD1Dcq-lJX8IgSRVd4InC6uc,8996 -pip/_vendor/platformdirs/macos.py,sha256=wftsbsvq6nZ0WORXSiCrZNkRHz_WKuktl0a6mC7MFkI,5580 -pip/_vendor/platformdirs/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/platformdirs/unix.py,sha256=Cci9Wqt35dAMsg6HT9nRGHSBW5obb0pR3AE1JJnsCXg,10643 -pip/_vendor/platformdirs/version.py,sha256=r7F76tZRjgQKzrpx_I0_ZMQOMU-PS7eGnHD7zEK3KB0,411 -pip/_vendor/platformdirs/windows.py,sha256=IFpiohUBwxPtCzlyKwNtxyW4Jk8haa6W8o59mfrDXVo,10125 -pip/_vendor/pygments/__init__.py,sha256=7N1oiaWulw_nCsTY4EEixYLz15pWY5u4uPAFFi-ielU,2983 -pip/_vendor/pygments/__main__.py,sha256=isIhBxLg65nLlXukG4VkMuPfNdd7gFzTZ_R_z3Q8diY,353 -pip/_vendor/pygments/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/__main__.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/cmdline.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/console.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/filter.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/formatter.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/lexer.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/modeline.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/plugin.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/regexopt.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/scanner.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/sphinxext.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/style.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/token.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/unistring.cpython-312.pyc,, -pip/_vendor/pygments/__pycache__/util.cpython-312.pyc,, -pip/_vendor/pygments/cmdline.py,sha256=LIVzmAunlk9sRJJp54O4KRy9GDIN4Wu13v9p9QzfGPM,23656 -pip/_vendor/pygments/console.py,sha256=yhP9UsLAVmWKVQf2446JJewkA7AiXeeTf4Ieg3Oi2fU,1718 -pip/_vendor/pygments/filter.py,sha256=_ADNPCskD8_GmodHi6_LoVgPU3Zh336aBCT5cOeTMs0,1910 -pip/_vendor/pygments/filters/__init__.py,sha256=RdedK2KWKXlKwR7cvkfr3NUj9YiZQgMgilRMFUg2jPA,40392 -pip/_vendor/pygments/filters/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pygments/formatter.py,sha256=jDWBTndlBH2Z5IYZFVDnP0qn1CaTQjTWt7iAGtCnJEg,4390 -pip/_vendor/pygments/formatters/__init__.py,sha256=8No-NUs8rBTSSBJIv4hSEQt2M0cFB4hwAT0snVc2QGE,5385 -pip/_vendor/pygments/formatters/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/_mapping.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/bbcode.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/groff.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/html.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/img.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/irc.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/latex.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/other.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/pangomarkup.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/rtf.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/svg.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/terminal.cpython-312.pyc,, -pip/_vendor/pygments/formatters/__pycache__/terminal256.cpython-312.pyc,, -pip/_vendor/pygments/formatters/_mapping.py,sha256=1Cw37FuQlNacnxRKmtlPX4nyLoX9_ttko5ZwscNUZZ4,4176 -pip/_vendor/pygments/formatters/bbcode.py,sha256=3JQLI45tcrQ_kRUMjuab6C7Hb0XUsbVWqqbSn9cMjkI,3320 -pip/_vendor/pygments/formatters/groff.py,sha256=M39k0PaSSZRnxWjqBSVPkF0mu1-Vr7bm6RsFvs-CNN4,5106 -pip/_vendor/pygments/formatters/html.py,sha256=SE2jc3YCqbMS3rZW9EAmDlAUhdVxJ52gA4dileEvCGU,35669 -pip/_vendor/pygments/formatters/img.py,sha256=MwA4xWPLOwh6j7Yc6oHzjuqSPt0M1fh5r-5BTIIUfsU,23287 -pip/_vendor/pygments/formatters/irc.py,sha256=dp1Z0l_ObJ5NFh9MhqLGg5ptG5hgJqedT2Vkutt9v0M,4981 -pip/_vendor/pygments/formatters/latex.py,sha256=XMmhOCqUKDBQtG5mGJNAFYxApqaC5puo5cMmPfK3944,19306 -pip/_vendor/pygments/formatters/other.py,sha256=56PMJOliin-rAUdnRM0i1wsV1GdUPd_dvQq0_UPfF9c,5034 -pip/_vendor/pygments/formatters/pangomarkup.py,sha256=y16U00aVYYEFpeCfGXlYBSMacG425CbfoG8oKbKegIg,2218 -pip/_vendor/pygments/formatters/rtf.py,sha256=ZT90dmcKyJboIB0mArhL7IhE467GXRN0G7QAUgG03To,11957 -pip/_vendor/pygments/formatters/svg.py,sha256=KKsiophPupHuxm0So-MsbQEWOT54IAiSF7hZPmxtKXE,7174 -pip/_vendor/pygments/formatters/terminal.py,sha256=AojNG4MlKq2L6IsC_VnXHu4AbHCBn9Otog6u45XvxeI,4674 -pip/_vendor/pygments/formatters/terminal256.py,sha256=kGkNUVo3FpwjytIDS0if79EuUoroAprcWt3igrcIqT0,11753 -pip/_vendor/pygments/lexer.py,sha256=TYHDt___gNW4axTl2zvPZff-VQi8fPaIh5OKRcVSjUM,35349 -pip/_vendor/pygments/lexers/__init__.py,sha256=pIlxyQJuu_syh9lE080cq8ceVbEVcKp0osAFU5fawJU,12115 -pip/_vendor/pygments/lexers/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pygments/lexers/__pycache__/_mapping.cpython-312.pyc,, -pip/_vendor/pygments/lexers/__pycache__/python.cpython-312.pyc,, -pip/_vendor/pygments/lexers/_mapping.py,sha256=61-h3zr103m01OS5BUq_AfUiL9YI06Ves9ipQ7k4vr4,76097 -pip/_vendor/pygments/lexers/python.py,sha256=2J_YJrPTr_A6fJY_qKiKv0GpgPwHMrlMSeo59qN3fe4,53687 -pip/_vendor/pygments/modeline.py,sha256=gtRYZBS-CKOCDXHhGZqApboHBaZwGH8gznN3O6nuxj4,1005 -pip/_vendor/pygments/plugin.py,sha256=ioeJ3QeoJ-UQhZpY9JL7vbxsTVuwwM7BCu-Jb8nN0AU,1891 -pip/_vendor/pygments/regexopt.py,sha256=Hky4EB13rIXEHQUNkwmCrYqtIlnXDehNR3MztafZ43w,3072 -pip/_vendor/pygments/scanner.py,sha256=NDy3ofK_fHRFK4hIDvxpamG871aewqcsIb6sgTi7Fhk,3092 -pip/_vendor/pygments/sphinxext.py,sha256=iOptJBcqOGPwMEJ2p70PvwpZPIGdvdZ8dxvq6kzxDgA,7981 -pip/_vendor/pygments/style.py,sha256=rSCZWFpg1_DwFMXDU0nEVmAcBHpuQGf9RxvOPPQvKLQ,6420 -pip/_vendor/pygments/styles/__init__.py,sha256=qUk6_1z5KmT8EdJFZYgESmG6P_HJF_2vVrDD7HSCGYY,2042 -pip/_vendor/pygments/styles/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pygments/styles/__pycache__/_mapping.cpython-312.pyc,, -pip/_vendor/pygments/styles/_mapping.py,sha256=6lovFUE29tz6EsV3XYY4hgozJ7q1JL7cfO3UOlgnS8w,3312 -pip/_vendor/pygments/token.py,sha256=qZwT7LSPy5YBY3JgDjut642CCy7JdQzAfmqD9NmT5j0,6226 -pip/_vendor/pygments/unistring.py,sha256=p5c1i-HhoIhWemy9CUsaN9o39oomYHNxXll0Xfw6tEA,63208 -pip/_vendor/pygments/util.py,sha256=2tj2nS1X9_OpcuSjf8dOET2bDVZhs8cEKd_uT6-Fgg8,10031 -pip/_vendor/pyproject_hooks/__init__.py,sha256=kCehmy0UaBa9oVMD7ZIZrnswfnP3LXZ5lvnNJAL5JBM,491 -pip/_vendor/pyproject_hooks/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pyproject_hooks/__pycache__/_compat.cpython-312.pyc,, -pip/_vendor/pyproject_hooks/__pycache__/_impl.cpython-312.pyc,, -pip/_vendor/pyproject_hooks/_compat.py,sha256=by6evrYnqkisiM-MQcvOKs5bgDMzlOSgZqRHNqf04zE,138 -pip/_vendor/pyproject_hooks/_impl.py,sha256=61GJxzQip0IInhuO69ZI5GbNQ82XEDUB_1Gg5_KtUoc,11920 -pip/_vendor/pyproject_hooks/_in_process/__init__.py,sha256=9gQATptbFkelkIy0OfWFEACzqxXJMQDWCH9rBOAZVwQ,546 -pip/_vendor/pyproject_hooks/_in_process/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/pyproject_hooks/_in_process/__pycache__/_in_process.cpython-312.pyc,, -pip/_vendor/pyproject_hooks/_in_process/_in_process.py,sha256=m2b34c917IW5o-Q_6TYIHlsK9lSUlNiyrITTUH_zwew,10927 -pip/_vendor/requests/__init__.py,sha256=HlB_HzhrzGtfD_aaYUwUh1zWXLZ75_YCLyit75d0Vz8,5057 -pip/_vendor/requests/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/__version__.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/_internal_utils.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/adapters.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/api.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/auth.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/certs.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/compat.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/cookies.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/exceptions.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/help.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/hooks.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/models.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/packages.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/sessions.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/status_codes.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/structures.cpython-312.pyc,, -pip/_vendor/requests/__pycache__/utils.cpython-312.pyc,, -pip/_vendor/requests/__version__.py,sha256=FVfglgZmNQnmYPXpOohDU58F5EUb_-VnSTaAesS187g,435 -pip/_vendor/requests/_internal_utils.py,sha256=nMQymr4hs32TqVo5AbCrmcJEhvPUh7xXlluyqwslLiQ,1495 -pip/_vendor/requests/adapters.py,sha256=J7VeVxKBvawbtlX2DERVo05J9BXTcWYLMHNd1Baa-bk,27607 -pip/_vendor/requests/api.py,sha256=_Zb9Oa7tzVIizTKwFrPjDEY9ejtm_OnSRERnADxGsQs,6449 -pip/_vendor/requests/auth.py,sha256=kF75tqnLctZ9Mf_hm9TZIj4cQWnN5uxRz8oWsx5wmR0,10186 -pip/_vendor/requests/certs.py,sha256=PVPooB0jP5hkZEULSCwC074532UFbR2Ptgu0I5zwmCs,575 -pip/_vendor/requests/compat.py,sha256=Mo9f9xZpefod8Zm-n9_StJcVTmwSukXR2p3IQyyVXvU,1485 -pip/_vendor/requests/cookies.py,sha256=bNi-iqEj4NPZ00-ob-rHvzkvObzN3lEpgw3g6paS3Xw,18590 -pip/_vendor/requests/exceptions.py,sha256=D1wqzYWne1mS2rU43tP9CeN1G7QAy7eqL9o1god6Ejw,4272 -pip/_vendor/requests/help.py,sha256=hRKaf9u0G7fdwrqMHtF3oG16RKktRf6KiwtSq2Fo1_0,3813 -pip/_vendor/requests/hooks.py,sha256=CiuysiHA39V5UfcCBXFIx83IrDpuwfN9RcTUgv28ftQ,733 -pip/_vendor/requests/models.py,sha256=x4K4CmH-lC0l2Kb-iPfMN4dRXxHEcbOaEWBL_i09AwI,35483 -pip/_vendor/requests/packages.py,sha256=_ZQDCJTJ8SP3kVWunSqBsRZNPzj2c1WFVqbdr08pz3U,1057 -pip/_vendor/requests/sessions.py,sha256=ykTI8UWGSltOfH07HKollH7kTBGw4WhiBVaQGmckTw4,30495 -pip/_vendor/requests/status_codes.py,sha256=iJUAeA25baTdw-6PfD0eF4qhpINDJRJI-yaMqxs4LEI,4322 -pip/_vendor/requests/structures.py,sha256=-IbmhVz06S-5aPSZuUthZ6-6D9XOjRuTXHOabY041XM,2912 -pip/_vendor/requests/utils.py,sha256=L79vnFbzJ3SFLKtJwpoWe41Tozi3RlZv94pY1TFIyow,33631 -pip/_vendor/resolvelib/__init__.py,sha256=h509TdEcpb5-44JonaU3ex2TM15GVBLjM9CNCPwnTTs,537 -pip/_vendor/resolvelib/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/resolvelib/__pycache__/providers.cpython-312.pyc,, -pip/_vendor/resolvelib/__pycache__/reporters.cpython-312.pyc,, -pip/_vendor/resolvelib/__pycache__/resolvers.cpython-312.pyc,, -pip/_vendor/resolvelib/__pycache__/structs.cpython-312.pyc,, -pip/_vendor/resolvelib/compat/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/resolvelib/compat/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/resolvelib/compat/__pycache__/collections_abc.cpython-312.pyc,, -pip/_vendor/resolvelib/compat/collections_abc.py,sha256=uy8xUZ-NDEw916tugUXm8HgwCGiMO0f-RcdnpkfXfOs,156 -pip/_vendor/resolvelib/providers.py,sha256=fuuvVrCetu5gsxPB43ERyjfO8aReS3rFQHpDgiItbs4,5871 -pip/_vendor/resolvelib/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/resolvelib/reporters.py,sha256=TSbRmWzTc26w0ggsV1bxVpeWDB8QNIre6twYl7GIZBE,1601 -pip/_vendor/resolvelib/resolvers.py,sha256=G8rsLZSq64g5VmIq-lB7UcIJ1gjAxIQJmTF4REZleQ0,20511 -pip/_vendor/resolvelib/structs.py,sha256=0_1_XO8z_CLhegP3Vpf9VJ3zJcfLm0NOHRM-i0Ykz3o,4963 -pip/_vendor/rich/__init__.py,sha256=dRxjIL-SbFVY0q3IjSMrfgBTHrm1LZDgLOygVBwiYZc,6090 -pip/_vendor/rich/__main__.py,sha256=eO7Cq8JnrgG8zVoeImiAs92q3hXNMIfp0w5lMsO7Q2Y,8477 -pip/_vendor/rich/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/__main__.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_cell_widths.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_emoji_codes.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_emoji_replace.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_export_format.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_extension.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_fileno.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_inspect.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_log_render.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_loop.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_null_file.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_palettes.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_pick.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_ratio.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_spinners.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_stack.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_timer.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_win32_console.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_windows.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_windows_renderer.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/_wrap.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/abc.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/align.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/ansi.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/bar.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/box.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/cells.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/color.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/color_triplet.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/columns.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/console.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/constrain.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/containers.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/control.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/default_styles.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/diagnose.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/emoji.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/errors.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/file_proxy.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/filesize.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/highlighter.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/json.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/jupyter.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/layout.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/live.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/live_render.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/logging.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/markup.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/measure.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/padding.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/pager.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/palette.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/panel.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/pretty.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/progress.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/progress_bar.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/prompt.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/protocol.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/region.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/repr.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/rule.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/scope.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/screen.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/segment.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/spinner.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/status.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/style.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/styled.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/syntax.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/table.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/terminal_theme.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/text.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/theme.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/themes.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/traceback.cpython-312.pyc,, -pip/_vendor/rich/__pycache__/tree.cpython-312.pyc,, -pip/_vendor/rich/_cell_widths.py,sha256=fbmeyetEdHjzE_Vx2l1uK7tnPOhMs2X1lJfO3vsKDpA,10209 -pip/_vendor/rich/_emoji_codes.py,sha256=hu1VL9nbVdppJrVoijVshRlcRRe_v3dju3Mmd2sKZdY,140235 -pip/_vendor/rich/_emoji_replace.py,sha256=n-kcetsEUx2ZUmhQrfeMNc-teeGhpuSQ5F8VPBsyvDo,1064 -pip/_vendor/rich/_export_format.py,sha256=RI08pSrm5tBSzPMvnbTqbD9WIalaOoN5d4M1RTmLq1Y,2128 -pip/_vendor/rich/_extension.py,sha256=Xt47QacCKwYruzjDi-gOBq724JReDj9Cm9xUi5fr-34,265 -pip/_vendor/rich/_fileno.py,sha256=HWZxP5C2ajMbHryvAQZseflVfQoGzsKOHzKGsLD8ynQ,799 -pip/_vendor/rich/_inspect.py,sha256=oZJGw31e64dwXSCmrDnvZbwVb1ZKhWfU8wI3VWohjJk,9695 -pip/_vendor/rich/_log_render.py,sha256=1ByI0PA1ZpxZY3CGJOK54hjlq4X-Bz_boIjIqCd8Kns,3225 -pip/_vendor/rich/_loop.py,sha256=hV_6CLdoPm0va22Wpw4zKqM0RYsz3TZxXj0PoS-9eDQ,1236 -pip/_vendor/rich/_null_file.py,sha256=tGSXk_v-IZmbj1GAzHit8A3kYIQMiCpVsCFfsC-_KJ4,1387 -pip/_vendor/rich/_palettes.py,sha256=cdev1JQKZ0JvlguV9ipHgznTdnvlIzUFDBb0It2PzjI,7063 -pip/_vendor/rich/_pick.py,sha256=evDt8QN4lF5CiwrUIXlOJCntitBCOsI3ZLPEIAVRLJU,423 -pip/_vendor/rich/_ratio.py,sha256=Zt58apszI6hAAcXPpgdWKpu3c31UBWebOeR4mbyptvU,5471 -pip/_vendor/rich/_spinners.py,sha256=U2r1_g_1zSjsjiUdAESc2iAMc3i4ri_S8PYP6kQ5z1I,19919 -pip/_vendor/rich/_stack.py,sha256=-C8OK7rxn3sIUdVwxZBBpeHhIzX0eI-VM3MemYfaXm0,351 -pip/_vendor/rich/_timer.py,sha256=zelxbT6oPFZnNrwWPpc1ktUeAT-Vc4fuFcRZLQGLtMI,417 -pip/_vendor/rich/_win32_console.py,sha256=P0vxI2fcndym1UU1S37XAzQzQnkyY7YqAKmxm24_gug,22820 -pip/_vendor/rich/_windows.py,sha256=aBwaD_S56SbgopIvayVmpk0Y28uwY2C5Bab1wl3Bp-I,1925 -pip/_vendor/rich/_windows_renderer.py,sha256=t74ZL3xuDCP3nmTp9pH1L5LiI2cakJuQRQleHCJerlk,2783 -pip/_vendor/rich/_wrap.py,sha256=FlSsom5EX0LVkA3KWy34yHnCfLtqX-ZIepXKh-70rpc,3404 -pip/_vendor/rich/abc.py,sha256=ON-E-ZqSSheZ88VrKX2M3PXpFbGEUUZPMa_Af0l-4f0,890 -pip/_vendor/rich/align.py,sha256=sCUkisXkQfoq-IQPyBELfJ8l7LihZJX3HbH8K7Cie-M,10368 -pip/_vendor/rich/ansi.py,sha256=iD6532QYqnBm6hADulKjrV8l8kFJ-9fEVooHJHH3hMg,6906 -pip/_vendor/rich/bar.py,sha256=ldbVHOzKJOnflVNuv1xS7g6dLX2E3wMnXkdPbpzJTcs,3263 -pip/_vendor/rich/box.py,sha256=nr5fYIUghB_iUCEq6y0Z3LlCT8gFPDrzN9u2kn7tJl4,10831 -pip/_vendor/rich/cells.py,sha256=aMmGK4BjXhgE6_JF1ZEGmW3O7mKkE8g84vUnj4Et4To,4780 -pip/_vendor/rich/color.py,sha256=bCRATVdRe5IClJ6Hl62de2PKQ_U4i2MZ4ugjUEg7Tao,18223 -pip/_vendor/rich/color_triplet.py,sha256=3lhQkdJbvWPoLDO-AnYImAWmJvV5dlgYNCVZ97ORaN4,1054 -pip/_vendor/rich/columns.py,sha256=HUX0KcMm9dsKNi11fTbiM_h2iDtl8ySCaVcxlalEzq8,7131 -pip/_vendor/rich/console.py,sha256=deFZIubq2M9A2MCsKFAsFQlWDvcOMsGuUA07QkOaHIw,99173 -pip/_vendor/rich/constrain.py,sha256=1VIPuC8AgtKWrcncQrjBdYqA3JVWysu6jZo1rrh7c7Q,1288 -pip/_vendor/rich/containers.py,sha256=c_56TxcedGYqDepHBMTuZdUIijitAQgnox-Qde0Z1qo,5502 -pip/_vendor/rich/control.py,sha256=DSkHTUQLorfSERAKE_oTAEUFefZnZp4bQb4q8rHbKws,6630 -pip/_vendor/rich/default_styles.py,sha256=-Fe318kMVI_IwciK5POpThcO0-9DYJ67TZAN6DlmlmM,8082 -pip/_vendor/rich/diagnose.py,sha256=an6uouwhKPAlvQhYpNNpGq9EJysfMIOvvCbO3oSoR24,972 -pip/_vendor/rich/emoji.py,sha256=omTF9asaAnsM4yLY94eR_9dgRRSm1lHUszX20D1yYCQ,2501 -pip/_vendor/rich/errors.py,sha256=5pP3Kc5d4QJ_c0KFsxrfyhjiPVe7J1zOqSFbFAzcV-Y,642 -pip/_vendor/rich/file_proxy.py,sha256=Tl9THMDZ-Pk5Wm8sI1gGg_U5DhusmxD-FZ0fUbcU0W0,1683 -pip/_vendor/rich/filesize.py,sha256=9fTLAPCAwHmBXdRv7KZU194jSgNrRb6Wx7RIoBgqeKY,2508 -pip/_vendor/rich/highlighter.py,sha256=6ZAjUcNhBRajBCo9umFUclyi2xL0-55JL7S0vYGUJu4,9585 -pip/_vendor/rich/json.py,sha256=vVEoKdawoJRjAFayPwXkMBPLy7RSTs-f44wSQDR2nJ0,5031 -pip/_vendor/rich/jupyter.py,sha256=QyoKoE_8IdCbrtiSHp9TsTSNyTHY0FO5whE7jOTd9UE,3252 -pip/_vendor/rich/layout.py,sha256=ajkSFAtEVv9EFTcFs-w4uZfft7nEXhNzL7ZVdgrT5rI,14004 -pip/_vendor/rich/live.py,sha256=vUcnJV2LMSK3sQNaILbm0-_B8BpAeiHfcQMAMLfpRe0,14271 -pip/_vendor/rich/live_render.py,sha256=zJtB471jGziBtEwxc54x12wEQtH4BuQr1SA8v9kU82w,3666 -pip/_vendor/rich/logging.py,sha256=uB-cB-3Q4bmXDLLpbOWkmFviw-Fde39zyMV6tKJ2WHQ,11903 -pip/_vendor/rich/markup.py,sha256=3euGKP5s41NCQwaSjTnJxus5iZMHjxpIM0W6fCxra38,8451 -pip/_vendor/rich/measure.py,sha256=HmrIJX8sWRTHbgh8MxEay_83VkqNW_70s8aKP5ZcYI8,5305 -pip/_vendor/rich/padding.py,sha256=kTFGsdGe0os7tXLnHKpwTI90CXEvrceeZGCshmJy5zw,4970 -pip/_vendor/rich/pager.py,sha256=SO_ETBFKbg3n_AgOzXm41Sv36YxXAyI3_R-KOY2_uSc,828 -pip/_vendor/rich/palette.py,sha256=lInvR1ODDT2f3UZMfL1grq7dY_pDdKHw4bdUgOGaM4Y,3396 -pip/_vendor/rich/panel.py,sha256=2Fd1V7e1kHxlPFIusoHY5T7-Cs0RpkrihgVG9ZVqJ4g,10705 -pip/_vendor/rich/pretty.py,sha256=5oIHP_CGWnHEnD0zMdW5qfGC5kHqIKn7zH_eC4crULE,35848 -pip/_vendor/rich/progress.py,sha256=P02xi7T2Ua3qq17o83bkshe4c0v_45cg8VyTj6US6Vg,59715 -pip/_vendor/rich/progress_bar.py,sha256=L4jw8E6Qb_x-jhOrLVhkuMaPmiAhFIl8jHQbWFrKuR8,8164 -pip/_vendor/rich/prompt.py,sha256=wdOn2X8XTJKnLnlw6PoMY7xG4iUPp3ezt4O5gqvpV-E,11304 -pip/_vendor/rich/protocol.py,sha256=5hHHDDNHckdk8iWH5zEbi-zuIVSF5hbU2jIo47R7lTE,1391 -pip/_vendor/rich/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/rich/region.py,sha256=rNT9xZrVZTYIXZC0NYn41CJQwYNbR-KecPOxTgQvB8Y,166 -pip/_vendor/rich/repr.py,sha256=5MZJZmONgC6kud-QW-_m1okXwL2aR6u6y-pUcUCJz28,4431 -pip/_vendor/rich/rule.py,sha256=0fNaS_aERa3UMRc3T5WMpN_sumtDxfaor2y3of1ftBk,4602 -pip/_vendor/rich/scope.py,sha256=TMUU8qo17thyqQCPqjDLYpg_UU1k5qVd-WwiJvnJVas,2843 -pip/_vendor/rich/screen.py,sha256=YoeReESUhx74grqb0mSSb9lghhysWmFHYhsbMVQjXO8,1591 -pip/_vendor/rich/segment.py,sha256=hU1ueeXqI6YeFa08K9DAjlF2QLxcJY9pwZx7RsXavlk,24246 -pip/_vendor/rich/spinner.py,sha256=15koCmF0DQeD8-k28Lpt6X_zJQUlzEhgo_6A6uy47lc,4339 -pip/_vendor/rich/status.py,sha256=kkPph3YeAZBo-X-4wPp8gTqZyU466NLwZBA4PZTTewo,4424 -pip/_vendor/rich/style.py,sha256=3hiocH_4N8vwRm3-8yFWzM7tSwjjEven69XqWasSQwM,27073 -pip/_vendor/rich/styled.py,sha256=eZNnzGrI4ki_54pgY3Oj0T-x3lxdXTYh4_ryDB24wBU,1258 -pip/_vendor/rich/syntax.py,sha256=TnZDuOD4DeHFbkaVEAji1gf8qgAlMU9Boe_GksMGCkk,35475 -pip/_vendor/rich/table.py,sha256=nGEvAZHF4dy1vT9h9Gj9O5qhSQO3ODAxJv0RY1vnIB8,39680 -pip/_vendor/rich/terminal_theme.py,sha256=1j5-ufJfnvlAo5Qsi_ACZiXDmwMXzqgmFByObT9-yJY,3370 -pip/_vendor/rich/text.py,sha256=5rQ3zvNrg5UZKNLecbh7fiw9v3HeFulNVtRY_CBDjjE,47312 -pip/_vendor/rich/theme.py,sha256=belFJogzA0W0HysQabKaHOc3RWH2ko3fQAJhoN-AFdo,3777 -pip/_vendor/rich/themes.py,sha256=0xgTLozfabebYtcJtDdC5QkX5IVUEaviqDUJJh4YVFk,102 -pip/_vendor/rich/traceback.py,sha256=CUpxYLjQWIb6vQQ6O72X0hvDV6caryGqU6UweHgOyCY,29601 -pip/_vendor/rich/tree.py,sha256=meAOUU6sYnoBEOX2ILrPLY9k5bWrWNQKkaiEFvHinXM,9167 -pip/_vendor/tomli/__init__.py,sha256=JhUwV66DB1g4Hvt1UQCVMdfCu-IgAV8FXmvDU9onxd4,396 -pip/_vendor/tomli/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/tomli/__pycache__/_parser.cpython-312.pyc,, -pip/_vendor/tomli/__pycache__/_re.cpython-312.pyc,, -pip/_vendor/tomli/__pycache__/_types.cpython-312.pyc,, -pip/_vendor/tomli/_parser.py,sha256=g9-ENaALS-B8dokYpCuzUFalWlog7T-SIYMjLZSWrtM,22633 -pip/_vendor/tomli/_re.py,sha256=dbjg5ChZT23Ka9z9DHOXfdtSpPwUfdgMXnj8NOoly-w,2943 -pip/_vendor/tomli/_types.py,sha256=-GTG2VUqkpxwMqzmVO4F7ybKddIbAnuAHXfmWQcTi3Q,254 -pip/_vendor/tomli/py.typed,sha256=8PjyZ1aVoQpRVvt71muvuq5qE-jTFZkK-GLHkhdebmc,26 -pip/_vendor/truststore/__init__.py,sha256=M-PhuLMIF7gxKXk7tpo2MD7dk6nqG1ae8GXWdNXbMdQ,403 -pip/_vendor/truststore/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/truststore/__pycache__/_api.cpython-312.pyc,, -pip/_vendor/truststore/__pycache__/_macos.cpython-312.pyc,, -pip/_vendor/truststore/__pycache__/_openssl.cpython-312.pyc,, -pip/_vendor/truststore/__pycache__/_ssl_constants.cpython-312.pyc,, -pip/_vendor/truststore/__pycache__/_windows.cpython-312.pyc,, -pip/_vendor/truststore/_api.py,sha256=B9JIHipzBIS8pMP_J50-o1DHVZsvKZQUXTB0HQQ_UPg,10461 -pip/_vendor/truststore/_macos.py,sha256=VJ24avz5aEGYAs_kWvnGjMJtuIP4xJcYa459UQOQC3M,17608 -pip/_vendor/truststore/_openssl.py,sha256=LLUZ7ZGaio-i5dpKKjKCSeSufmn6T8pi9lDcFnvSyq0,2324 -pip/_vendor/truststore/_ssl_constants.py,sha256=NUD4fVKdSD02ri7-db0tnO0VqLP9aHuzmStcW7tAl08,1130 -pip/_vendor/truststore/_windows.py,sha256=eldNViHNHeY5r3fiBoz_JFGD37atXB9S5yaRoPKEGAA,17891 -pip/_vendor/truststore/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/typing_extensions.py,sha256=78hFl0HpDY-ylHUVCnWdU5nTHxUP2-S-3wEZk6CQmLk,134499 -pip/_vendor/urllib3/__init__.py,sha256=iXLcYiJySn0GNbWOOZDDApgBL1JgP44EZ8i1760S8Mc,3333 -pip/_vendor/urllib3/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/_collections.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/_version.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/connection.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/connectionpool.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/exceptions.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/fields.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/filepost.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/poolmanager.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/request.cpython-312.pyc,, -pip/_vendor/urllib3/__pycache__/response.cpython-312.pyc,, -pip/_vendor/urllib3/_collections.py,sha256=pyASJJhW7wdOpqJj9QJA8FyGRfr8E8uUUhqUvhF0728,11372 -pip/_vendor/urllib3/_version.py,sha256=cuJvnSrWxXGYgQ3-ZRoPMw8-qaN5tpw71jnH1t16dLA,64 -pip/_vendor/urllib3/connection.py,sha256=92k9td_y4PEiTIjNufCUa1NzMB3J3w0LEdyokYgXnW8,20300 -pip/_vendor/urllib3/connectionpool.py,sha256=Be6q65SR9laoikg-h_jmc_p8OWtEmwgq_Om_Xtig-2M,40285 -pip/_vendor/urllib3/contrib/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/urllib3/contrib/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/_appengine_environ.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/appengine.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/ntlmpool.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/pyopenssl.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/securetransport.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/__pycache__/socks.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/_appengine_environ.py,sha256=bDbyOEhW2CKLJcQqAKAyrEHN-aklsyHFKq6vF8ZFsmk,957 -pip/_vendor/urllib3/contrib/_securetransport/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/urllib3/contrib/_securetransport/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/_securetransport/__pycache__/bindings.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/_securetransport/__pycache__/low_level.cpython-312.pyc,, -pip/_vendor/urllib3/contrib/_securetransport/bindings.py,sha256=4Xk64qIkPBt09A5q-RIFUuDhNc9mXilVapm7WnYnzRw,17632 -pip/_vendor/urllib3/contrib/_securetransport/low_level.py,sha256=B2JBB2_NRP02xK6DCa1Pa9IuxrPwxzDzZbixQkb7U9M,13922 -pip/_vendor/urllib3/contrib/appengine.py,sha256=VR68eAVE137lxTgjBDwCna5UiBZTOKa01Aj_-5BaCz4,11036 -pip/_vendor/urllib3/contrib/ntlmpool.py,sha256=NlfkW7WMdW8ziqudopjHoW299og1BTWi0IeIibquFwk,4528 -pip/_vendor/urllib3/contrib/pyopenssl.py,sha256=hDJh4MhyY_p-oKlFcYcQaVQRDv6GMmBGuW9yjxyeejM,17081 -pip/_vendor/urllib3/contrib/securetransport.py,sha256=Fef1IIUUFHqpevzXiDPbIGkDKchY2FVKeVeLGR1Qq3g,34446 -pip/_vendor/urllib3/contrib/socks.py,sha256=aRi9eWXo9ZEb95XUxef4Z21CFlnnjbEiAo9HOseoMt4,7097 -pip/_vendor/urllib3/exceptions.py,sha256=0Mnno3KHTNfXRfY7638NufOPkUb6mXOm-Lqj-4x2w8A,8217 -pip/_vendor/urllib3/fields.py,sha256=kvLDCg_JmH1lLjUUEY_FLS8UhY7hBvDPuVETbY8mdrM,8579 -pip/_vendor/urllib3/filepost.py,sha256=5b_qqgRHVlL7uLtdAYBzBh-GHmU5AfJVt_2N0XS3PeY,2440 -pip/_vendor/urllib3/packages/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/urllib3/packages/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/packages/__pycache__/six.cpython-312.pyc,, -pip/_vendor/urllib3/packages/backports/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0 -pip/_vendor/urllib3/packages/backports/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/packages/backports/__pycache__/makefile.cpython-312.pyc,, -pip/_vendor/urllib3/packages/backports/__pycache__/weakref_finalize.cpython-312.pyc,, -pip/_vendor/urllib3/packages/backports/makefile.py,sha256=nbzt3i0agPVP07jqqgjhaYjMmuAi_W5E0EywZivVO8E,1417 -pip/_vendor/urllib3/packages/backports/weakref_finalize.py,sha256=tRCal5OAhNSRyb0DhHp-38AtIlCsRP8BxF3NX-6rqIA,5343 -pip/_vendor/urllib3/packages/six.py,sha256=b9LM0wBXv7E7SrbCjAm4wwN-hrH-iNxv18LgWNMMKPo,34665 -pip/_vendor/urllib3/poolmanager.py,sha256=aWyhXRtNO4JUnCSVVqKTKQd8EXTvUm1VN9pgs2bcONo,19990 -pip/_vendor/urllib3/request.py,sha256=YTWFNr7QIwh7E1W9dde9LM77v2VWTJ5V78XuTTw7D1A,6691 -pip/_vendor/urllib3/response.py,sha256=fmDJAFkG71uFTn-sVSTh2Iw0WmcXQYqkbRjihvwBjU8,30641 -pip/_vendor/urllib3/util/__init__.py,sha256=JEmSmmqqLyaw8P51gUImZh8Gwg9i1zSe-DoqAitn2nc,1155 -pip/_vendor/urllib3/util/__pycache__/__init__.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/connection.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/proxy.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/queue.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/request.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/response.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/retry.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/ssl_.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/ssl_match_hostname.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/ssltransport.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/timeout.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/url.cpython-312.pyc,, -pip/_vendor/urllib3/util/__pycache__/wait.cpython-312.pyc,, -pip/_vendor/urllib3/util/connection.py,sha256=5Lx2B1PW29KxBn2T0xkN1CBgRBa3gGVJBKoQoRogEVk,4901 -pip/_vendor/urllib3/util/proxy.py,sha256=zUvPPCJrp6dOF0N4GAVbOcl6o-4uXKSrGiTkkr5vUS4,1605 -pip/_vendor/urllib3/util/queue.py,sha256=nRgX8_eX-_VkvxoX096QWoz8Ps0QHUAExILCY_7PncM,498 -pip/_vendor/urllib3/util/request.py,sha256=C0OUt2tcU6LRiQJ7YYNP9GvPrSvl7ziIBekQ-5nlBZk,3997 -pip/_vendor/urllib3/util/response.py,sha256=GJpg3Egi9qaJXRwBh5wv-MNuRWan5BIu40oReoxWP28,3510 -pip/_vendor/urllib3/util/retry.py,sha256=Z6WEf518eTOXP5jr5QSQ9gqJI0DVYt3Xs3EKnYaTmus,22013 -pip/_vendor/urllib3/util/ssl_.py,sha256=X4-AqW91aYPhPx6-xbf66yHFQKbqqfC_5Zt4WkLX1Hc,17177 -pip/_vendor/urllib3/util/ssl_match_hostname.py,sha256=Ir4cZVEjmAk8gUAIHWSi7wtOO83UCYABY2xFD1Ql_WA,5758 -pip/_vendor/urllib3/util/ssltransport.py,sha256=NA-u5rMTrDFDFC8QzRKUEKMG0561hOD4qBTr3Z4pv6E,6895 -pip/_vendor/urllib3/util/timeout.py,sha256=cwq4dMk87mJHSBktK1miYJ-85G-3T3RmT20v7SFCpno,10168 -pip/_vendor/urllib3/util/url.py,sha256=lCAE7M5myA8EDdW0sJuyyZhVB9K_j38ljWhHAnFaWoE,14296 -pip/_vendor/urllib3/util/wait.py,sha256=fOX0_faozG2P7iVojQoE1mbydweNyTcm-hXEfFrTtLI,5403 -pip/_vendor/vendor.txt,sha256=PxNaxxkkpBaw5zOTsDpHEY-zEaHjgkDgyrSxOuxg8nw,330 -pip/py.typed,sha256=EBVvvPRTn_eIpz5e5QztSCdrMX7Qwd7VP93RSoIlZ2I,286 diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/REQUESTED b/.env/Lib/site-packages/pip-24.2.dist-info/REQUESTED deleted file mode 100644 index e69de29..0000000 diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/WHEEL b/.env/Lib/site-packages/pip-24.2.dist-info/WHEEL deleted file mode 100644 index ecaf39f..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/WHEEL +++ /dev/null @@ -1,5 +0,0 @@ -Wheel-Version: 1.0 -Generator: setuptools (71.1.0) -Root-Is-Purelib: true -Tag: py3-none-any - diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/entry_points.txt b/.env/Lib/site-packages/pip-24.2.dist-info/entry_points.txt deleted file mode 100644 index 25fcf7e..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/entry_points.txt +++ /dev/null @@ -1,3 +0,0 @@ -[console_scripts] -pip = pip._internal.cli.main:main -pip3 = pip._internal.cli.main:main diff --git a/.env/Lib/site-packages/pip-24.2.dist-info/top_level.txt b/.env/Lib/site-packages/pip-24.2.dist-info/top_level.txt deleted file mode 100644 index a1b589e..0000000 --- a/.env/Lib/site-packages/pip-24.2.dist-info/top_level.txt +++ /dev/null @@ -1 +0,0 @@ -pip diff --git a/.env/Lib/site-packages/pip/__init__.py b/.env/Lib/site-packages/pip/__init__.py deleted file mode 100644 index 640e922..0000000 --- a/.env/Lib/site-packages/pip/__init__.py +++ /dev/null @@ -1,13 +0,0 @@ -from typing import List, Optional - -__version__ = "24.2" - - -def main(args: Optional[List[str]] = None) -> int: - """This is an internal API only meant for use by pip's own console scripts. - - For additional details, see https://github.com/pypa/pip/issues/7498. - """ - from pip._internal.utils.entrypoints import _wrapper - - return _wrapper(args) diff --git a/.env/Lib/site-packages/pip/__main__.py b/.env/Lib/site-packages/pip/__main__.py deleted file mode 100644 index 5991326..0000000 --- a/.env/Lib/site-packages/pip/__main__.py +++ /dev/null @@ -1,24 +0,0 @@ -import os -import sys - -# Remove '' and current working directory from the first entry -# of sys.path, if present to avoid using current directory -# in pip commands check, freeze, install, list and show, -# when invoked as python -m pip -if sys.path[0] in ("", os.getcwd()): - sys.path.pop(0) - -# If we are running from a wheel, add the wheel to sys.path -# This allows the usage python pip-*.whl/pip install pip-*.whl -if __package__ == "": - # __file__ is pip-*.whl/pip/__main__.py - # first dirname call strips of '/__main__.py', second strips off '/pip' - # Resulting path is the name of the wheel itself - # Add that to sys.path so we can import pip - path = os.path.dirname(os.path.dirname(__file__)) - sys.path.insert(0, path) - -if __name__ == "__main__": - from pip._internal.cli.main import main as _main - - sys.exit(_main()) diff --git a/.env/Lib/site-packages/pip/__pip-runner__.py b/.env/Lib/site-packages/pip/__pip-runner__.py deleted file mode 100644 index c633787..0000000 --- a/.env/Lib/site-packages/pip/__pip-runner__.py +++ /dev/null @@ -1,50 +0,0 @@ -"""Execute exactly this copy of pip, within a different environment. - -This file is named as it is, to ensure that this module can't be imported via -an import statement. -""" - -# /!\ This version compatibility check section must be Python 2 compatible. /!\ - -import sys - -# Copied from pyproject.toml -PYTHON_REQUIRES = (3, 8) - - -def version_str(version): # type: ignore - return ".".join(str(v) for v in version) - - -if sys.version_info[:2] < PYTHON_REQUIRES: - raise SystemExit( - "This version of pip does not support python {} (requires >={}).".format( - version_str(sys.version_info[:2]), version_str(PYTHON_REQUIRES) - ) - ) - -# From here on, we can use Python 3 features, but the syntax must remain -# Python 2 compatible. - -import runpy # noqa: E402 -from importlib.machinery import PathFinder # noqa: E402 -from os.path import dirname # noqa: E402 - -PIP_SOURCES_ROOT = dirname(dirname(__file__)) - - -class PipImportRedirectingFinder: - @classmethod - def find_spec(self, fullname, path=None, target=None): # type: ignore - if fullname != "pip": - return None - - spec = PathFinder.find_spec(fullname, [PIP_SOURCES_ROOT], target) - assert spec, (PIP_SOURCES_ROOT, fullname) - return spec - - -sys.meta_path.insert(0, PipImportRedirectingFinder()) - -assert __name__ == "__main__", "Cannot run __pip-runner__.py as a non-main module" -runpy.run_module("pip", run_name="__main__", alter_sys=True) diff --git a/.env/Lib/site-packages/pip/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 2763d42d52a43982c5f7539aaff67eecffe07542..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 685 zcmX|9Pixdb6rV}5*>%@i{Bx-X9lc0E6DUP(k%B0K&_81JvIIglGu<7UOoo}s)|3i8 zhzMT1wSEJ|@8B2kqSsthJbA0EUV8G)+WMIH=DnAf-}}wH`7|@rMIe)pFL{O$`eB0; zX&pP0o8astiYU%ef%~{{`VL0a>ASQ|J?1dXyg8_`N?WY8=~4ehtMAut(D`XW0;lm- zxHe5^T^e4BG~e;&7sC0q#!J;uH@;$~Qo=I$FD`giCzrrLcT64I{W-?SLA!VteK@Y( zK!Lln|7659ft3ovg<%SMkh_l_l2YU~DOf5D*(?>QG$VtWNG`8wQf>>9l|q*}BRW%D z8Xb1K1mppfrj+u3WC>*^<+;92G-G6BjMPg}G~{Mf4Z^G}B3a8c0?deOT`?WqT3Eat zPJxsI>;v3%l5Ler$rN10WvC336P96R06JvCs9Khg)v9Cvag@FPsjn>vAxG=?mg1+H zDIGs9X-QVoY{UfLVzI2_Ri3F5zKwatbPRCgc!dvQ%?-OD(`+jpLfTSHpffi~63V*q zij)i019Tr$p9Qmr{&Z1m@q6$5oBOYqKYMc%NWux$f_CGZT5>T|b}Y&R)AW*L418du zNz%C5sK$j8OcCEBb=ub1_jS~{Q@K@&bQPol zu?-%&B3OIsRmuP0Kj1}M1=%7LJb4S`){`?$mK3Z9-(lW2Z@ziow{PcjE|&%#y-%;$ z0~z3#*hCQbp~KnYJxL$c8^*r zVh=Nc4tF5}0gVx(R?Cp1U~Z7OzgLy-wh#5s%k@HrmD`M5DPRj5=xdE z%C7H**w(!DFuq1T{Rs(W+p@!qs@j@f*Vd@2h6%zPQ)^JsWtADC6lbPO9Zyum39pRX z%+9OK^eE>VdEH=nq2q}65XMf_LrkAC?<4ettK-Vz4P}Klah3ZPu`Vxb`Z_gPohr7k zEHmA)_*s+b)K&7-Y$|2;NO74*X9vDqg$rJbJ}Yng$H`j_i##^yZD*YK7ysRT$Cs{* z5ITg^A*8!mz=_RbtGKn$x*5y`e&Fo*Z~V86ph;}uOEKL7rF_+lg=2o_6KgXw~hb+ diff --git a/.env/Lib/site-packages/pip/__pycache__/__pip-runner__.cpython-312.pyc b/.env/Lib/site-packages/pip/__pycache__/__pip-runner__.cpython-312.pyc deleted file mode 100644 index 83ad85cf0077dcda0c199d292f4b051b88a0521b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2205 zcmaJ>O-vg{6n?Y2{>QNsLm;GSHcFb}3QS2;C5i~CN=ZV}05Nc=MYY=Oj$y&xb#~VT z6C+WC+El7k4ppJIb(~`jtW^RQmXb)ZX`am<0{iKg2%T0^6W^|?wAci(!$;35#9 z`_<4)2Y}_f0&0_MUt+s`(7p#ES~^Wls@o#fXEoE%vp7!#gJaAkt!1c)qb_HuD$16q z*|_W!Z3A~(d6OctY}etlhMLFSimbnGi#O0Mn=?!s-DWj8DC>>2XxXw&>t}4Spoikz z7K?t15khIa95aA`AG1_Xp+Y8+>AhC`$3*3pp$PARn=AK)Mj_4*?U2Mvh;N}&4e=@v zL@mOH5r754*jkq&aWTFmtAxO3`voY#cl<39Cqr@K@Rn`HIfqNU>5y5AkxCSI#{17r zj9b*SCX$9~h}UFgit5@7oyZj?u4#&Cplw!Ds5OzmG*7&*-I%a6n|9@7Wk#OF-}orq zl7za;%;!rzsZQr|X5aj)vxu%4x`f9kdNYP1XRN-&3wspfuv$?40e>w* zDb{)K(%Q{0`q#+n#e0|jwpcEgT3Yc6jxcR#I*z^upv7)3njzFqXwM-g<^|YiwW3;t z&CtHj&whHg%%Cf=&Pu4WBy?8M1Y0Z#^A~E2X#g~MCaMNS_87Kk<#UVpyjzJ2LQ$~7S76s#Bx%kKZ;2EIWGFY}@c86$-aLR%P9Tj}m#iFB zoao5lh%`Dpp6nkOm6F55V{wloSTvJ%{ONoq!_wjiIn0*hwPkaX+Gs^Ti`_$HQhQIc z4VUvL{yPbY7bm`kJFWAm7U5|qwvvAsdSx@#`g`QiM&x)oa{TV)N8J}niFbZFS&0nZ z5gtckC4bMOgD3C%OD#PWf6pCzGZMQKG8ql=T0#>soorL-Xuvk^K$J1 zxC$v5*|M?}Iic2qeE>Cxdi0FWCQ~Qyvzk#Y!=@+tob!5`t2}u3KV!4H1KY-q%*vWB zN%KAZvd(CS#U#E|R;-c9$V$D7?H22?nT+LQSlE8_sYUJtUwbdhYOYN zt5u#01gpFd@@|9R@m3E22|s~1p1|Q>W9?s^xqG&9xU&-LDnay5ub(tO1611wfe4?3 zKMHR^a~Yafu9w?RRoYHJg14XgA}i+}`i_(!Svs1mp!OZvgy^DY1EOV!uC&~3|K9!u IdTTfS2eWf0L;wH) diff --git a/.env/Lib/site-packages/pip/_internal/__init__.py b/.env/Lib/site-packages/pip/_internal/__init__.py deleted file mode 100644 index 1a5b7f8..0000000 --- a/.env/Lib/site-packages/pip/_internal/__init__.py +++ /dev/null @@ -1,18 +0,0 @@ -from typing import List, Optional - -from pip._internal.utils import _log - -# init_logging() must be called before any call to logging.getLogger() -# which happens at import of most modules. -_log.init_logging() - - -def main(args: Optional[List[str]] = None) -> int: - """This is preserved for old console scripts that may still be referencing - it. - - For additional details, see https://github.com/pypa/pip/issues/7498. - """ - from pip._internal.utils.entrypoints import _wrapper - - return _wrapper(args) diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 94a4c1d45ed32d9b87aed94cb856ee2b428408d0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 787 zcmZuvy=xRf6rb6hyKF8Aa>gbWI*QGK?lwY9kTi-_62WNUv2bp7U+%{2$1pP&U5SuF zM6j_lHr7i12mb*Z1A?%Ph*;R2oHnU^v$Crle!t}GD;i*>|tH$B>7k#OiYZPN*Fjmp}BxQ`;rDs zZLy^JtkLyKx8wb^u^p#kt3sQ~yDg0<@+j@liuKZ>zh1FO^>x)$V1_D>sSTfyi_*UF zIb?{GA~Hn!lv-rFbe9}VqBls#ee(A1KxmGiRG>laK*F~Q#fvoIagpmH1+HTyq|w|A zBEz$&tfL`fo!QS$^$BvO5uYXCklM(X7t=nFF# zZu)VN1+tV;AcYKs)VeqBi69SBbC^&Y86N>I3y^BC#(L~cOgO8gFXN_Qm6Uw!iUa`ta1n N5kdQ1{M(AU@L!e3(fa@Z diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/build_env.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/build_env.cpython-312.pyc deleted file mode 100644 index 5d97615c8e51c73d586feaf4a3a4eb1cd68829f0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14472 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

-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 diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/configuration.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/configuration.cpython-312.pyc deleted file mode 100644 index e7a25a5deb242e5a236a88430364199c7e592259..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 17632 zcmbV!dvF`andb~Jcn|;y5~N6q;$tL|A|Z*;Te4+E5+9N*StcD(w#!`GSO_yDL4g1> z1Bxa>%CgI~OJ;p1vXU&RO0t5gylcAby)bvFi*i?Y7kjU2UA6xJCLJ;prB-|Ex?A60 zu<4W&rRwf}Uk?Tk60$ElGP`@adwP1hzsK+U`h)+WwA8`j+W4=37W$9JIPRb6#<*;m z!iLGjaSNQp4R8`KnZo=4&vMg%iRI=2Gs}em0l7JB8MO{rM{NVPQTu>>v}B-U)G^=~ zbq+X3T>~zj#t_1#qh$kSqwWDWySIePM=J&@SlJr(j8+a*va&5)HM(tJ8!Owx)uS~7 zHKVlywLB;DBZrJ>=JJ8EPU+?R6t!pa@AWcjyNcHK6sL+RH{)LV-w(W+AGf z$WT}oC!&fZO0pVMLKtR9oEVlBOnsOJ&i_Wdv4c%c91DaLF-Wf-RHCDfPie_}y*AC< z8JX0Co=`BR*)QPR16RYcCR_@sG0oO1Ph6QClQnza7|kpY)+|@X$HKDLqVoh-8n2ycXT`~^+sb|@}+1%k|i&%*#cu@Qb6rU~P!tZ3Gu zaP%ssE2yEEteNG=^|%cQmO}AbG~wC)vzImN`AdCWotHGB|H75Cn)z(+bD9l_|GCb~ znm|jhSrw3l5(#F=lM$(a)QrCk1<3+8#f{{*vMwC=ZYCe0o|`TbHwI}=)AJc7&hI9f z-_4AwSf)5hkSui^_3h`p)_CXBfoq^KMpLxgDz=JfSdoVUiUfjG#ps}@M#q()EVkpT zLrOHls647LiuG!|{~_NIH+6J;O0uTcYt}5`=mgfyrpROAKv32kYAm3{R3cw5uQZ^S zN;484zTAD{#b>d^>Wi17QdE385FC~xp=w(A5`d z4?4yI!E4|)^+jymet#$eN&`LZr%lXqq;G6ebNeGvWi${D#brNjg3^Z9lDt$Q;<=mL z1E>4eiP;l>adOSslyWvD%!&RLr}t*hy3P6LufF!`Tx``=zwWM_>zwPFd1Xdece-yr zwSgzO;=A|^TA&{IDegM2R7j>N?gyM?{*if_k7f6DiZ^!Sb@Y40o(jgEVvj7n4|!kw zxu@hwQHsi{7>UMcqhbL_4hdyd9t%JQf%y%V4@4vq4nFZ2Ss4zDsp4dGT$}*Uqfaa< zf~LhbouqwnM;jy%wmcx6{*>nEH7TSAHH$hH4#jXsyQP)N#FG9|kOM6{?XxHkK7#e*x(hNRG-v}1@M{<0UTV#YBNCb7Wf)^&s?wdsq|7Mg z@~!WhHvRE){+cH$a_=)rkR};Qjd_eZzy=<$!JIdzj3k;if6W_;tYhAeRT*!{<+&Zp zds}8_r-fLx(Q~BMDCP1gKCitlS8tT2OhfrS>W%tbK4qQ~hD;%IKj?(V-!wPDd4*oh z3^DVNzY!OFD0tlpZJkz5lE|-%98xZ-W{RqsFcydnYr;r06w%z>eZA){ocDKi_Mh$T zd>U#&om4gJm?95`UV&bEWel838b&Fl0o_#08&I=cgPu|q0wfB3k3t{qwJ1lZccsq3 zMt=q#3g<*snkNiF9-PeYrnw{XM1F-zN<*YeXXzTK&rEv;-T2w?${FerhS{{k`m4vDUl!gp-LlQvZn}`jZ=03f0hn1?3rmRL0g&RkW0{B3askHy_?;;(G~Q@eZjL{kCtN_og>lr8@MkN3H&aB?Vu&18ohNHmlrdtO|X9W)14mqlwL{@CyW~ z0WBh)rFE;-N5ywFqIM^na}#VEwgZxgx?SjrcS}1_Zl2*@;RjlvSX;Ex?tH)y_#fEU zC&7|HJ^64R_@@w9IDnl+@fjt0Jp|A!1_;ChfeZrbkbVsW0ayrGqeapOqML#-46x5e zx-uM6#b6jplsW-{##C`EZ}iS*E&z-NVqy$fO%-<;h@Rj$p_*Ny8k-E0#UPU2rx^hZ z#G(p16NuL#`(`^L6oY9Y?#c}^9+6~)9$*6*V8VFYE}#C{Y_fqc5S}Uy0s{sxbGcv% znDqc~La)9Wg~dVzV>p_c@e1SBVgS|#q#}VmKomTR@#KIyNvo;bRrIP+YN4@;tR$Oq zW^7Rk!~**FV=$~nSdEGiphzF<)vMV^W5iq@BVE9* zSQZ!Lw5)+V-py#~%{ee7P^I4xRhSp}@q@hjC6L}@Cxi^0FJn(BM0{RR|87d``%5I9-A*#b< z^odtxF&e?&p!n_SxM`p0EdxRwAA^x$_XGcjVt&8oey;Pps@XDHP^<8d zMy1eTNS2H?GW=9IJeW6*4fY^e)>?_*&$!v|&uZ42OtUKI@xFL``J_rO_|bc%@=ei}`+p1tpBSua1oR^FQ`?_IB|UAQoR;fKMuTHbT7 zG(NUc^*A(eg=fPoRF*-Tm6oL)-1eO_!keY*+jk)=`>ma8KZZU}@{V7%9bG&>bAImf z>{E#wpRp?HX!JO;rNa>i+FfZYca}fT&j=se8-M?KDUI~|&#k5-4wd%){k@K^CgB5% zqpL+IFpvnqF;;pRg|9Pn!Ax8+Ft1h=H*a%HILS})UQ_%ec?Xy%%^8EtO;Jp6Fj<*r z){^<^@iPk4#Y_hJ5V>g8dP81!KlWUGhgZn*(`{NELJ2fNywI)TV|CP#3?~?$cGsG{ zDP?b3weN(Hd29MR)9bGCHCJ28)%LN=3&2#KF@Q46j;A@d(o>r*qlaWBjoZ&P=NU#VoMPs=7r04&f_ugM0yn{Xh4}Hy%!#4v6#|92 zPii?c%jpT48}+9#0<5fWS|*raSN9AQqFG|Yh*P$o4C8*cMLE5yz5%EGl} zJ!fe_iK1?DX**Y5wdQV0xtrG9ttofwin}doZ)1kNX2u5u^X+yrL>O_;pn&z^VfAK- zKj%4_)(JZQN%W^{iZ4pYbv&2T#<_7&o7Ii*+L(U8{U~eyzMdm{Q@r`S476mZD0n=# z;t=1%vfikujt7ScgF$n_P$O@X@)8ow{**kaQ!9B#m}UZnlhjcBtQ~AKMBWpXJ|7#S zixF~ULd3ug9Iq){=WN3*^jDun0^+T#p0VFAYgi1fljA(VePn zPrBMcvo)V{0-C&?|`>%frh9$<{L~p3bDJGhISG9;&oHx5^s1ziRH} z;n-tyDS%1YExU@1%6G`*mU;RSjxIy`Fo{ocZPTTiE^^FCmIYF|4#u-Df(WL~)54VS zT6b z<@#NIirOOINaQ-jpQMxDAL*E#?IDKauet>7xJ4oPf$ddLtsm|t(^=zrD_kY)E=2Fow3YZf8eZ0IveiS?_26$sXsXTG%(K*DOixrv03kXK#%9_%|fBj~=2g_q}FPT5-$ z{VBVbO`$PmZ(JNt41R3y&|Q`JB8hgJsAA)PGtPLlF#BrV{59M`yj8@7`Dki`X;PNs zl&xY9PdYK{O;-_q8_St0Mft5LuNeHMc)(3X9FW9K2ED7!ym@91F(46yqMTC5%iA`6 z0B#F-nH#Ao(xZUe{@9hB;=jS+bWn&lKN|u4el>$C0ldC9e&igC)v$OWbA%x>N4xkZ zA?X_jUxEW4rZv3v(Be^rnd<@Z>iCe)>(W)r5QbndiSZ}XDmj)`hS92(D)ML)j2~rZ zZcJrsWtlpIH?N^%0VfA(+%P4hln~N<1Oq$A=}R*#yuDCGfv4*;W(zz2pExkVb@3z z{RqD14L-_;IXSobNY-38#^Jc4`Yn5iV(>1{m2j~v?@r|%K^`Snf~hEXX89K}3}*V5 zo86mXftxfpa|Z$AC*XL00a}#j-gN0!2W*n8+X6O8HI4zYoGS@aJdW5bQv#ryV9f7z z99B#VH-w2CgtNwX_Cx>aux(sX=!i2E=@y zwBZDSX-*+JCPy?2g$Kx*t*>84jpU31pk#nC%a|HJs9D(TvVMk)UKLb)>}Kiz-a zsCSWM%uhpyHtbw$IK0|$7z*<2+_C#mfltjowdh{jw(8nXFtO%)3iH^%I{#{7e5Iyi z)#XcDZKa<3+qN&ZCMs9AwJuexgJ~|!UrJPbukTAUSH1gppH*;;N4_ja5pMTCd{S8l z*ojA~+t;g~S+A*|8(O@auq;l%aN5?sVKx)@nJYYhNSo2?p?Uzbe1Dg#tH$)c&(r0! ze8AaJ{=jMJsw(-Q+(zY^lCEQx55$VDgO(2tT2L-9wzB#+d$^7fH(xiMG2%;hfh=o`eP48*d|{7c>(vOx|F=y~|M4BnZwIZ+Six4wDUxPz;>73#ZE#DeugaAB>*Ke6CDbSFJ`#KH|exItN35&f&EBeC;0gE?9+=a zclO=hmzYS^?f=+y0KS~s-GH6t)ib4Amyg6ZQx5-$DTh3Z_bZ(?#`3_t$|u(%V33Rq ziWI5iVx)-JH9}Oh=^Em%wS%;(^UkJ=fVf;d2@Dbl;9@Y&fQ#7tZ)H>#`IXJM2XVq& z<3_UQ%ehIs*2avdA}!aBV_vmiGcTxtd_(NbV|Yt37--6wRMUuv8Ofe1=Vqc4;0P_; zrVH<|3}(FqAkC1#y{STh35cfIcj=FCJu*L+Ld+K*=)V>k8za6a{|x29d>^rXS|mz{ z27{+8&;5uvuV*ein+s;^7d#I&^g%?X#L3d2~1DfK?&(9s40C^CVrHgR8L+Y-B;((%}aVvd764#5uGJR zB(IezF3qKXRK}x93i&Suc{RMide}@D{V$F`l zp0)Zzsro}J^+#4}j?TE&oi#tS+$p(Tl5*}!987#O>1=zi>&F-0xwv%lBj<5ac0h1x z6JOEEe_=MmZ*J#GD<3rOS!+CyYCQ1E#zV8cFdLu!_4dX&3)zmf3#aB!B`Z6Y+jN=u z(V?gA>$)$wv-=m%et3Sh>dJam{e$fdcbaZDE!8Zyu55p7&WcAXwk?#-mo5&a%6G4o zccjWYmd2L{m!cn+pZaVYSJ(8JnQJ=CTOAc3oXkbl+9Pl);k$QdDEgc+l`IYxayar0N7s z4iDvZBnWWI#*j(OLzf}v+!p0oNyH3WXm|#d@!DcEOw{khvZ-We(Xmh2t+Ls&WX+jh zw0(GV)&9-(ifwBZZK;a3CCf^MZ^rtdreOgP?3Xp|P&661RkKp%ojD7;gxMiADc8<5 zS36lERcqc;DetL!6CYYuyuBa0`qCDz)=LiAtQN{QGKlk-7V5%;@*R1_4r2_OC@@2c zXr7{!k~July)h@=%C2JONDh!qn{k9;DsG-kncvICK$82dpZ&pX$U5O*Hh5qJW!-BM zC>>^l@FVjS|6Z2H0Q!u{a3fXOVzz)$DqwtMN0}1dvl?^>lLK0i6Af&DepF1K%18OwF*dB3`(|EJjX-LhCL(P3TFrib@ zGwhHu&xEkbuON660Y>3yFo4jXD)@*g)(92?Kxk$fbov>PG|#g*5RXc7k4&jQR~paE^rcOX(z84a@&)I-bMfp_-<)%`{A9X{ zQS{99nbO~VQOQ;9{+zRvdOoQ>zTV(ldNt`geec`J9Z#*-HGN?*Yz%}TRf)O>@N>?D zG12>;yjtCpEbsY4+CnuCRf@KH-%;7QlY4*XzOF{|2c@pA?dA`*TTm`AG_q285l=ko zR410@wMV&ui^*O{((DNZ2G3lO>|g1m+bT8TmNLhEk3=VVkV2QIh~-q{mJ_Hy(F=LU zq#=U>^_T*ai^?2I8ar#tJk8ERx7Z?dG_l1NVG2C&rR>o?L>P21#%xovI==Vp!b|fn z!Ogp7-;=WMVZN805O9gMr6X&5&!qOAS=rmQ;_Cjb*;L}t^=5?TM^d;zZ)OdYi)dj6 z3fvUHq5w9`ogx4S;-?&OwC17jtVW}H93fcXjJJ}j{>=tM2gTQVjR0ftH66MULd%l8 z7|1tPm`#^cCM;Gb6IDZyheg&X(Q&vSWQtqKMd6cUKn1cxn7^Eh@S&)gQJm&6r-q6y z4jnkbxn3-IyQ3(kgaXam2R%ZDhEew5J0XPHD9d>Hvumq4a$lhb2%E`-qLVC@m?1L{ z=Hq8{viSsdvr0WcTTk=_TMGVy+8GDH*6f9up1J0A&-TUc`Hr{tFSo4hI-c?zpXpw2 z+&Ob`PW=ukw7MoTtLl>W?OW|C0RuMUZ{rt8o`OrM05N6x5sjIv7(OyI9jN8;obwG< z4}IvSE2pni{>@r)j7jFYq8x0DN|-C>lDp31xg(%5)1c`u`2`+m=LvF+{jHfDzMnD9 zeFNf<9Knw(5aOjf3J{N#zm0GxQ(S`HriYs{J3$l3TKJUKg}7Qr^qK9~VNWZX6k0Q_ zf_}X~QLFv_3+H?LE}!jz`+@nNXq%P0l(0yC8jUtw-|fG`-R9krf%`z;-R^Th3Wa}@4h-G z;7oJ1>A**o2iEHv6Xr$lcl$mo;kNJny-FJY^<$ll{LgnFd4F38(gIPPB?%7O|F0#9 zf*)P*xr62GrkcLm%toNtSDN8ubs7W!L0`mA>p~2p(gLusvgp?qEFeQy3kY4y zR|YAwroDRXSI`HG2Nj+wVT-+3$ z?h`N2Z_5yzdKJIyniNN)2>Owgp=bm@3JL{D_hp{TeiU9B86wP^Ra5#`Q}AiwM6-`* z?oCdOUii^TSSf#vL^ESKlrwb85DC$X9))*=N`FZSK{q{Q9vU7Y{JQ_a4Iz~8Hc-BU zMy*EQHr;oT7lIF;PQGCKGu&;)wp-jK=Si}Ch@MQceNoNrXt=ZI_MSVw+rFgt#J!%C zy3QH*{Tkm&&4C%$CuNnl!n5JUOYd#_as4~>srqBdvST0Fk3pfe?R|Un?r3t?VW87y z$GU6BN3PRvnclYDwY^<-w`}Rv74dZ1Y*3WP>m1 z@vXRgG!oTZp6gg~HT|C1vaffa>^{X81nQ9TG~;hWMDi%eYXsvUWQhgPu;3A*9z^=G z^A~W0&S}iS1++mp6TdlPYKA#v={-iU0NFG8Nu!F8&&)=~cXj9Us;^j|A=;G+j09kb z_^zqhsm1dtcT3XVvK2GXN3jnhk@~0)%Lb+t!4MGGcyjYaL`r^I9BF5LFgdh9QM(dz zdBd-PV}nN@3Lkp(q41~w7gr{1j=A6TC?kC}HJfffs`e?l>;=Wv=zQK%*sg<4)Sr_6pkX|K~UIypNE z;CXh@{}e40_C)}Qpi%L~L9@s?t4u&KP0$4PFIdx!4@iP4FiLUC#as>z+IRghOE z8*)g$P~9X7uHmuGqF_sC%p@%tNhF-CH0(;8TkHcyU3b}q1k>Z z8bq~lH5v^ogr^i@aTa2&JWmgs;dW5|gzAXp6tbfgT0$M^N>qwcqEd2$k|89T9oh|n z<*`YH)=Xc5D^&VECFILveh!5UTV~uSgn1Qmekr6Rl)t6q1xkoelzT|D(o9HrP!5Ov zen>i6Xcn}3M#s%v)RhvNy83rm!QZ{XeP(i6?4Q+DTF&!{sx(LWa{Ij_Kk@114|{&g z>NfTWmOW{g%~G3aUpjoZV}nC6?GY^PbFU;UiQv-V#PD5rs^-83e&f`hb~!9v{9=Ea zqdd{S!SZx%g~dme$d)Quwp_JAcj?_b(4wX}WQo`YWg90v7SF~3v*l?XM_<;mw5h~W zy?7RmkXr z9BG%!QU~atHdEfbcj@eM^Rl|!{!U-28C-(O2PLI5!MVdT!?TSW7Tl-nt(F!n!O`W) z<=}GdJEv2-p4i|}m-aMT#KdD5q2-=%um9&I%llLNx>C*E8yp(aXLysPb*Tr^%F?=g z+8JY4uN){+#c;+}tnl&~_M?X4kGMZ!b$_R>#Or5xie6?Eose1PJ z&OZ5-pv>ZWR6X{`Nqs5{`@o$LR`p*j))i^N58p@elSTa(h&zgPpr{3!URjX|u)%$! zA^e{M=%7#``%CAO|3W3kkmS)({)UnqTr0$93Ueya39a%ZB|oF&Iwd4TSbU<6_h=>G z;q+DAjq*eLNnJdgQvM#dU>Z&>LyCw3o(J>r!bTm(SN)nR{WVweYtHeH+_^RGT#7sQ zrC{bw_^|`u{u|Es8}8sgaz`I{8rM9&l*f0|{VQA9%=I~S#kS+7@J~&<=e3#YZjbkmY95e2;ma4B(;Tuy^9E(< zavR^ccsR`=OB`7;-+de`&Nrs3%6V&|Dj_ZPELXmBArVdOJhs81?#tc70^h}d&asTG G*Z%|1M%DxX diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/exceptions.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/exceptions.cpython-312.pyc deleted file mode 100644 index 3d631ef0bf5fb21d350e1e36624a458f62c6872a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 35553 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 s1Ie+`k(Qd4N{-0#8f+xEB+W2gHM@0{{R3 diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/main.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/main.cpython-312.pyc deleted file mode 100644 index 3e05350e70f84e753b7e950fac1e833a84e6d34e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 670 zcmX|9&1)1f6i;Smx9he=#8W+J6fX-p15(fy=|%C>R>+Ns}6c|KRrP&XrgtEuIW59 zPFJPOH=9V>5jJfLZeq)f;+{!J$m{en8H&kO;A1}JoCWNVAF_|9r?*%yeDZF4tWnS~ z77*+{C^4+8sB$H;s>CV>ftl6Dp>SjAL?LT|PUpE8fUqzG3uUIuQCE;?7k6ji0Z~Y$ z^uKI{0w;BjHv|HRv2zAjlVqgbcru8ysz^+2WMZ^QG~xs>xxKt{C!P`6fQsnxrn7rs zr7>Ws7gy+HysJwGmRb={C>xBSbhb7XdC_+Koxu9$&oFutAx?Jgt)@FvB&LrlRf!Fm zjiJmLrUlQ|AK=ik`XU+oUR^z!jhbhP#M+}HEhjv1}rXTSY4WBCgg7`6TY diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/pyproject.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/pyproject.cpython-312.pyc deleted file mode 100644 index 2d670375b004cd23e4ff54f29e603fb90d129c37..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 5118 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 R!lj6~;@SPkL!qgH{|21ZFRB0l diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/self_outdated_check.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/self_outdated_check.cpython-312.pyc deleted file mode 100644 index c6b7d35cadfca60d15d924b4d104ba518c320853..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10208 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�)-mb!Trz_U?(G>CB?%Dm`e9Uqy?fbuoXEmCM6TnkRm5( z*a)c%!?6KL^Ayb{y$oYpunh?Nj80>;Qg@gO(13+T5k|mq73~XC#a6**g2%{M1wv(` zvl!Z>Nq|^}AM?hH_(}^4f*r}Yjy#C0x{k3h*exyXYWSV2Z^e4A>Swz@+MSX5(n24j z1>LS1QB~cZWD#F=iwA@Hep#?DQin16sQb=-9YVIVB(M()KL)l}(gW0%){544t2+TeV)))O2!0Y80 zIRi1Nh7w6QcUieU1O}NK1XXe-G#gG#DYEW2T9AhTv>+o@0P{jTZVDn{g%G&zgLr@u z)JDKDl}^DD0g>X)bXx=*EU?7Nph0nedgT}_XvDN55Cx2)*wno&#$9Y1+-^5LFdI$b zblf_{#|Juth-`>Rp>z%!UoqO6Sdi89eW?;pY6=pS>DAE%nDs9>a~3inE)g86tUr+P z2Xdk>S5}j&-~H4sRNEGZ!RV^#%=zncyIY=G`5N2eD^H!o+mQ9_$$0kUYT9zOd!gTJ zTO7&*lE$q6K*oRIfpyh?H0vMC_y-qUF~V1-H=$X z!xju%)Dwdsf6UyZUV+?H*nHtIA^^*h0C|ys|BV?3pn{7MssIH=NsTT5sVyF~ka^ye zU#zEED-bTk--;w%nlBe)@#qV{8b#tR!ORP=N&yQ;N@{+-sI(WXe`OWRe32yp!`JVK zhZtOGHPnqP$)(k7e+7ee-d4geZ$qDR9(FJVSV3CDDZE3ipevj=)(0OU^Y6C&2(g@?^rfaLW)BpH9egb*~vCX0KtP9FyAAxw?x; zDaCk(IRZnB|K=JmgSZBQ-zwjs<0OE994ML=V2oUV)6;FS2YR;>mK$X zW|AWl$hvzJY^7q?+c1@JD=>VKjOn+r7d5Y5v6XruFiCaq);_%o`EtAT*Q>X%LI|F_ zScG2{0WIn+Oh#O#E`$+Db4x}HCj?&uFE%#-_?PfgKZ9(M8Euc)UANg(~}me zbN0Hleb4WA9a!CUFfG>ST)Wb)wsm*K&5JiKrt6Nbx%(i3k`p~Q9XA~5iq@6bnt1q$ zzd7A<1lW6VcwKP*!^L+mrpx!=ORNbmY-+~WgyT;dPz*F2O$)Wfkpt<2gKNUj?_EtG z4)QkQs9yGF#CC9497Eu`)n(o78F%}=?rg_k2L8JT^HySQ_(CM&?sZ?|hm|W=SN9xF zKX+lxcQGwqeC+Zu39?12p-sfJT8s&Wzs+bVF*(HGEWi$%v|CA=fdPZZzb!%VydZWo zT$`nxbk<~1SD8?RY?`ry_L@Ysu`QWv0fX)koG%4hCX0df5&{hkL6%UPo0=RnHwsfo z7U$_@I{qxO+)w|-mbkaXMB^QNACp&7$e-w(7{G<7Rn7ZLZc}SOhIkJF!Y#Qf%kP5`iBtc4(vW>X@iFbbTcl&(}WcQ zDl{DMG0Tv~;0PH%8V)^1*qQ4Oes6L{7cqc3mrN)+KdnXRJ20v{#-*WX6B;*Pz*l7KeZDs>_MqC;skrZ{x~Lc5i=XZ~vP2xpn`ptiL_u zZ_oNWGX9SH{n^grna<8Gl zM(0z99W9Q084;^NF0G5+o8BAVTiWtB9}GPlcyMCL`pxp(Co2LAGa*^JB*#p~||#bZ3zcgGbi zjAvpvGOgPN;cp9KYWH?i7v)poY>IdiiZ+v^Qvl2Upo5kpO#N1L;2Ht82+(9$W3d3e zX(s@9x1qBsc(Kz|BTj%31AX^Sp!lZ*hWExIpho-@}c43o+Kx<{3*flViSKh#_Bru>PFyU9Ce)pe$e#ZuJ*%YW59Ht@^5P@X!Tz`Z zf%H8h#~+bnkI3L7());XKeOTQEY>fHS+PDN)~|4DV)LT)v7>&e zChMrrIO>=CR~@a3w!Dq!%9cm+1d5fB4J`6*8+U=|Xyn#i&s*c;6^VH;dESo#-9=LOW z)gN0uOObc4Zdjo|-`~WQJ$BXQtx$r2P}z{JXvtKxu`jlK%rD CM?JCt diff --git a/.env/Lib/site-packages/pip/_internal/__pycache__/wheel_builder.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/__pycache__/wheel_builder.cpython-312.pyc deleted file mode 100644 index 1e214ce466e94d0f808db085941a61a8324e71df..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13619 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 diff --git a/.env/Lib/site-packages/pip/_internal/build_env.py b/.env/Lib/site-packages/pip/_internal/build_env.py deleted file mode 100644 index be1e0ca..0000000 --- a/.env/Lib/site-packages/pip/_internal/build_env.py +++ /dev/null @@ -1,315 +0,0 @@ -"""Build Environment used for isolation during sdist building -""" - -import logging -import os -import pathlib -import site -import sys -import textwrap -from collections import OrderedDict -from types import TracebackType -from typing import TYPE_CHECKING, Iterable, List, Optional, Set, Tuple, Type, Union - -from pip._vendor.certifi import where -from pip._vendor.packaging.version import Version - -from pip import __file__ as pip_location -from pip._internal.cli.spinners import open_spinner -from pip._internal.locations import get_platlib, get_purelib, get_scheme -from pip._internal.metadata import get_default_environment, get_environment -from pip._internal.utils.logging import VERBOSE -from pip._internal.utils.packaging import get_requirement -from pip._internal.utils.subprocess import call_subprocess -from pip._internal.utils.temp_dir import TempDirectory, tempdir_kinds - -if TYPE_CHECKING: - from pip._internal.index.package_finder import PackageFinder - -logger = logging.getLogger(__name__) - - -def _dedup(a: str, b: str) -> Union[Tuple[str], Tuple[str, str]]: - return (a, b) if a != b else (a,) - - -class _Prefix: - def __init__(self, path: str) -> None: - self.path = path - self.setup = False - scheme = get_scheme("", prefix=path) - self.bin_dir = scheme.scripts - self.lib_dirs = _dedup(scheme.purelib, scheme.platlib) - - -def get_runnable_pip() -> str: - """Get a file to pass to a Python executable, to run the currently-running pip. - - This is used to run a pip subprocess, for installing requirements into the build - environment. - """ - source = pathlib.Path(pip_location).resolve().parent - - if not source.is_dir(): - # This would happen if someone is using pip from inside a zip file. In that - # case, we can use that directly. - return str(source) - - return os.fsdecode(source / "__pip-runner__.py") - - -def _get_system_sitepackages() -> Set[str]: - """Get system site packages - - Usually from site.getsitepackages, - but fallback on `get_purelib()/get_platlib()` if unavailable - (e.g. in a virtualenv created by virtualenv<20) - - Returns normalized set of strings. - """ - if hasattr(site, "getsitepackages"): - system_sites = site.getsitepackages() - else: - # virtualenv < 20 overwrites site.py without getsitepackages - # fallback on get_purelib/get_platlib. - # this is known to miss things, but shouldn't in the cases - # where getsitepackages() has been removed (inside a virtualenv) - system_sites = [get_purelib(), get_platlib()] - return {os.path.normcase(path) for path in system_sites} - - -class BuildEnvironment: - """Creates and manages an isolated environment to install build deps""" - - def __init__(self) -> None: - temp_dir = TempDirectory(kind=tempdir_kinds.BUILD_ENV, globally_managed=True) - - self._prefixes = OrderedDict( - (name, _Prefix(os.path.join(temp_dir.path, name))) - for name in ("normal", "overlay") - ) - - self._bin_dirs: List[str] = [] - self._lib_dirs: List[str] = [] - for prefix in reversed(list(self._prefixes.values())): - self._bin_dirs.append(prefix.bin_dir) - self._lib_dirs.extend(prefix.lib_dirs) - - # Customize site to: - # - ensure .pth files are honored - # - prevent access to system site packages - system_sites = _get_system_sitepackages() - - self._site_dir = os.path.join(temp_dir.path, "site") - if not os.path.exists(self._site_dir): - os.mkdir(self._site_dir) - with open( - os.path.join(self._site_dir, "sitecustomize.py"), "w", encoding="utf-8" - ) as fp: - fp.write( - textwrap.dedent( - """ - import os, site, sys - - # First, drop system-sites related paths. - original_sys_path = sys.path[:] - known_paths = set() - for path in {system_sites!r}: - site.addsitedir(path, known_paths=known_paths) - system_paths = set( - os.path.normcase(path) - for path in sys.path[len(original_sys_path):] - ) - original_sys_path = [ - path for path in original_sys_path - if os.path.normcase(path) not in system_paths - ] - sys.path = original_sys_path - - # Second, add lib directories. - # ensuring .pth file are processed. - for path in {lib_dirs!r}: - assert not path in sys.path - site.addsitedir(path) - """ - ).format(system_sites=system_sites, lib_dirs=self._lib_dirs) - ) - - def __enter__(self) -> None: - self._save_env = { - name: os.environ.get(name, None) - for name in ("PATH", "PYTHONNOUSERSITE", "PYTHONPATH") - } - - path = self._bin_dirs[:] - old_path = self._save_env["PATH"] - if old_path: - path.extend(old_path.split(os.pathsep)) - - pythonpath = [self._site_dir] - - os.environ.update( - { - "PATH": os.pathsep.join(path), - "PYTHONNOUSERSITE": "1", - "PYTHONPATH": os.pathsep.join(pythonpath), - } - ) - - def __exit__( - self, - exc_type: Optional[Type[BaseException]], - exc_val: Optional[BaseException], - exc_tb: Optional[TracebackType], - ) -> None: - for varname, old_value in self._save_env.items(): - if old_value is None: - os.environ.pop(varname, None) - else: - os.environ[varname] = old_value - - def check_requirements( - self, reqs: Iterable[str] - ) -> Tuple[Set[Tuple[str, str]], Set[str]]: - """Return 2 sets: - - conflicting requirements: set of (installed, wanted) reqs tuples - - missing requirements: set of reqs - """ - missing = set() - conflicting = set() - if reqs: - env = ( - get_environment(self._lib_dirs) - if hasattr(self, "_lib_dirs") - else get_default_environment() - ) - for req_str in reqs: - req = get_requirement(req_str) - # We're explicitly evaluating with an empty extra value, since build - # environments are not provided any mechanism to select specific extras. - if req.marker is not None and not req.marker.evaluate({"extra": ""}): - continue - dist = env.get_distribution(req.name) - if not dist: - missing.add(req_str) - continue - if isinstance(dist.version, Version): - installed_req_str = f"{req.name}=={dist.version}" - else: - installed_req_str = f"{req.name}==={dist.version}" - if not req.specifier.contains(dist.version, prereleases=True): - conflicting.add((installed_req_str, req_str)) - # FIXME: Consider direct URL? - return conflicting, missing - - def install_requirements( - self, - finder: "PackageFinder", - requirements: Iterable[str], - prefix_as_string: str, - *, - kind: str, - ) -> None: - prefix = self._prefixes[prefix_as_string] - assert not prefix.setup - prefix.setup = True - if not requirements: - return - self._install_requirements( - get_runnable_pip(), - finder, - requirements, - prefix, - kind=kind, - ) - - @staticmethod - def _install_requirements( - pip_runnable: str, - finder: "PackageFinder", - requirements: Iterable[str], - prefix: _Prefix, - *, - kind: str, - ) -> None: - args: List[str] = [ - sys.executable, - pip_runnable, - "install", - "--ignore-installed", - "--no-user", - "--prefix", - prefix.path, - "--no-warn-script-location", - "--disable-pip-version-check", - ] - if logger.getEffectiveLevel() <= logging.DEBUG: - args.append("-vv") - elif logger.getEffectiveLevel() <= VERBOSE: - args.append("-v") - for format_control in ("no_binary", "only_binary"): - formats = getattr(finder.format_control, format_control) - args.extend( - ( - "--" + format_control.replace("_", "-"), - ",".join(sorted(formats or {":none:"})), - ) - ) - - index_urls = finder.index_urls - if index_urls: - args.extend(["-i", index_urls[0]]) - for extra_index in index_urls[1:]: - args.extend(["--extra-index-url", extra_index]) - else: - args.append("--no-index") - for link in finder.find_links: - args.extend(["--find-links", link]) - - for host in finder.trusted_hosts: - args.extend(["--trusted-host", host]) - if finder.allow_all_prereleases: - args.append("--pre") - if finder.prefer_binary: - args.append("--prefer-binary") - args.append("--") - args.extend(requirements) - extra_environ = {"_PIP_STANDALONE_CERT": where()} - with open_spinner(f"Installing {kind}") as spinner: - call_subprocess( - args, - command_desc=f"pip subprocess to install {kind}", - spinner=spinner, - extra_environ=extra_environ, - ) - - -class NoOpBuildEnvironment(BuildEnvironment): - """A no-op drop-in replacement for BuildEnvironment""" - - def __init__(self) -> None: - pass - - def __enter__(self) -> None: - pass - - def __exit__( - self, - exc_type: Optional[Type[BaseException]], - exc_val: Optional[BaseException], - exc_tb: Optional[TracebackType], - ) -> None: - pass - - def cleanup(self) -> None: - pass - - def install_requirements( - self, - finder: "PackageFinder", - requirements: Iterable[str], - prefix_as_string: str, - *, - kind: str, - ) -> None: - raise NotImplementedError() diff --git a/.env/Lib/site-packages/pip/_internal/cache.py b/.env/Lib/site-packages/pip/_internal/cache.py deleted file mode 100644 index 6b45126..0000000 --- a/.env/Lib/site-packages/pip/_internal/cache.py +++ /dev/null @@ -1,290 +0,0 @@ -"""Cache Management -""" - -import hashlib -import json -import logging -import os -from pathlib import Path -from typing import Any, Dict, List, Optional - -from pip._vendor.packaging.tags import Tag, interpreter_name, interpreter_version -from pip._vendor.packaging.utils import canonicalize_name - -from pip._internal.exceptions import InvalidWheelFilename -from pip._internal.models.direct_url import DirectUrl -from pip._internal.models.link import Link -from pip._internal.models.wheel import Wheel -from pip._internal.utils.temp_dir import TempDirectory, tempdir_kinds -from pip._internal.utils.urls import path_to_url - -logger = logging.getLogger(__name__) - -ORIGIN_JSON_NAME = "origin.json" - - -def _hash_dict(d: Dict[str, str]) -> str: - """Return a stable sha224 of a dictionary.""" - s = json.dumps(d, sort_keys=True, separators=(",", ":"), ensure_ascii=True) - return hashlib.sha224(s.encode("ascii")).hexdigest() - - -class Cache: - """An abstract class - provides cache directories for data from links - - :param cache_dir: The root of the cache. - """ - - def __init__(self, cache_dir: str) -> None: - super().__init__() - assert not cache_dir or os.path.isabs(cache_dir) - self.cache_dir = cache_dir or None - - def _get_cache_path_parts(self, link: Link) -> List[str]: - """Get parts of part that must be os.path.joined with cache_dir""" - - # We want to generate an url to use as our cache key, we don't want to - # just reuse the URL because it might have other items in the fragment - # and we don't care about those. - key_parts = {"url": link.url_without_fragment} - if link.hash_name is not None and link.hash is not None: - key_parts[link.hash_name] = link.hash - if link.subdirectory_fragment: - key_parts["subdirectory"] = link.subdirectory_fragment - - # Include interpreter name, major and minor version in cache key - # to cope with ill-behaved sdists that build a different wheel - # depending on the python version their setup.py is being run on, - # and don't encode the difference in compatibility tags. - # https://github.com/pypa/pip/issues/7296 - key_parts["interpreter_name"] = interpreter_name() - key_parts["interpreter_version"] = interpreter_version() - - # Encode our key url with sha224, we'll use this because it has similar - # security properties to sha256, but with a shorter total output (and - # thus less secure). However the differences don't make a lot of - # difference for our use case here. - hashed = _hash_dict(key_parts) - - # We want to nest the directories some to prevent having a ton of top - # level directories where we might run out of sub directories on some - # FS. - parts = [hashed[:2], hashed[2:4], hashed[4:6], hashed[6:]] - - return parts - - def _get_candidates(self, link: Link, canonical_package_name: str) -> List[Any]: - can_not_cache = not self.cache_dir or not canonical_package_name or not link - if can_not_cache: - return [] - - path = self.get_path_for_link(link) - if os.path.isdir(path): - return [(candidate, path) for candidate in os.listdir(path)] - return [] - - def get_path_for_link(self, link: Link) -> str: - """Return a directory to store cached items in for link.""" - raise NotImplementedError() - - def get( - self, - link: Link, - package_name: Optional[str], - supported_tags: List[Tag], - ) -> Link: - """Returns a link to a cached item if it exists, otherwise returns the - passed link. - """ - raise NotImplementedError() - - -class SimpleWheelCache(Cache): - """A cache of wheels for future installs.""" - - def __init__(self, cache_dir: str) -> None: - super().__init__(cache_dir) - - def get_path_for_link(self, link: Link) -> str: - """Return a directory to store cached wheels for link - - Because there are M wheels for any one sdist, we provide a directory - to cache them in, and then consult that directory when looking up - cache hits. - - We only insert things into the cache if they have plausible version - numbers, so that we don't contaminate the cache with things that were - not unique. E.g. ./package might have dozens of installs done for it - and build a version of 0.0...and if we built and cached a wheel, we'd - end up using the same wheel even if the source has been edited. - - :param link: The link of the sdist for which this will cache wheels. - """ - parts = self._get_cache_path_parts(link) - assert self.cache_dir - # Store wheels within the root cache_dir - return os.path.join(self.cache_dir, "wheels", *parts) - - def get( - self, - link: Link, - package_name: Optional[str], - supported_tags: List[Tag], - ) -> Link: - candidates = [] - - if not package_name: - return link - - canonical_package_name = canonicalize_name(package_name) - for wheel_name, wheel_dir in self._get_candidates(link, canonical_package_name): - try: - wheel = Wheel(wheel_name) - except InvalidWheelFilename: - continue - if canonicalize_name(wheel.name) != canonical_package_name: - logger.debug( - "Ignoring cached wheel %s for %s as it " - "does not match the expected distribution name %s.", - wheel_name, - link, - package_name, - ) - continue - if not wheel.supported(supported_tags): - # Built for a different python/arch/etc - continue - candidates.append( - ( - wheel.support_index_min(supported_tags), - wheel_name, - wheel_dir, - ) - ) - - if not candidates: - return link - - _, wheel_name, wheel_dir = min(candidates) - return Link(path_to_url(os.path.join(wheel_dir, wheel_name))) - - -class EphemWheelCache(SimpleWheelCache): - """A SimpleWheelCache that creates it's own temporary cache directory""" - - def __init__(self) -> None: - self._temp_dir = TempDirectory( - kind=tempdir_kinds.EPHEM_WHEEL_CACHE, - globally_managed=True, - ) - - super().__init__(self._temp_dir.path) - - -class CacheEntry: - def __init__( - self, - link: Link, - persistent: bool, - ): - self.link = link - self.persistent = persistent - self.origin: Optional[DirectUrl] = None - origin_direct_url_path = Path(self.link.file_path).parent / ORIGIN_JSON_NAME - if origin_direct_url_path.exists(): - try: - self.origin = DirectUrl.from_json( - origin_direct_url_path.read_text(encoding="utf-8") - ) - except Exception as e: - logger.warning( - "Ignoring invalid cache entry origin file %s for %s (%s)", - origin_direct_url_path, - link.filename, - e, - ) - - -class WheelCache(Cache): - """Wraps EphemWheelCache and SimpleWheelCache into a single Cache - - This Cache allows for gracefully degradation, using the ephem wheel cache - when a certain link is not found in the simple wheel cache first. - """ - - def __init__(self, cache_dir: str) -> None: - super().__init__(cache_dir) - self._wheel_cache = SimpleWheelCache(cache_dir) - self._ephem_cache = EphemWheelCache() - - def get_path_for_link(self, link: Link) -> str: - return self._wheel_cache.get_path_for_link(link) - - def get_ephem_path_for_link(self, link: Link) -> str: - return self._ephem_cache.get_path_for_link(link) - - def get( - self, - link: Link, - package_name: Optional[str], - supported_tags: List[Tag], - ) -> Link: - cache_entry = self.get_cache_entry(link, package_name, supported_tags) - if cache_entry is None: - return link - return cache_entry.link - - def get_cache_entry( - self, - link: Link, - package_name: Optional[str], - supported_tags: List[Tag], - ) -> Optional[CacheEntry]: - """Returns a CacheEntry with a link to a cached item if it exists or - None. The cache entry indicates if the item was found in the persistent - or ephemeral cache. - """ - retval = self._wheel_cache.get( - link=link, - package_name=package_name, - supported_tags=supported_tags, - ) - if retval is not link: - return CacheEntry(retval, persistent=True) - - retval = self._ephem_cache.get( - link=link, - package_name=package_name, - supported_tags=supported_tags, - ) - if retval is not link: - return CacheEntry(retval, persistent=False) - - return None - - @staticmethod - def record_download_origin(cache_dir: str, download_info: DirectUrl) -> None: - origin_path = Path(cache_dir) / ORIGIN_JSON_NAME - if origin_path.exists(): - try: - origin = DirectUrl.from_json(origin_path.read_text(encoding="utf-8")) - except Exception as e: - logger.warning( - "Could not read origin file %s in cache entry (%s). " - "Will attempt to overwrite it.", - origin_path, - e, - ) - else: - # TODO: use DirectUrl.equivalent when - # https://github.com/pypa/pip/pull/10564 is merged. - if origin.url != download_info.url: - logger.warning( - "Origin URL %s in cache entry %s does not match download URL " - "%s. This is likely a pip bug or a cache corruption issue. " - "Will overwrite it with the new value.", - origin.url, - cache_dir, - download_info.url, - ) - origin_path.write_text(download_info.to_json(), encoding="utf-8") diff --git a/.env/Lib/site-packages/pip/_internal/cli/__init__.py b/.env/Lib/site-packages/pip/_internal/cli/__init__.py deleted file mode 100644 index e589bb9..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/__init__.py +++ /dev/null @@ -1,4 +0,0 @@ -"""Subpackage containing all of pip's command line interface related code -""" - -# This file intentionally does not import submodules diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 5fdbd1a8b21cc6835cf054e7862eb179d93ccf7c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 278 zcmXw!F-`+95Jl|}K@=%>aEDTm?IT1(L6f8a(I6wsu_s{^uWi{*q@_k&firLkF2D_F zxd39jRLr8}H#2{J@#cSyMngtwKVI>L=67d~;)kUj(DKGCdteP8_nvR=w#%r^Mz2Ab z;EP6&el4_fBCJHjc%JEA*VYcS6ZU1;Z zmG>D^mbbwMG1uk^JZ?b7M$XZsK-mrrWH|x9lQUe(j0G+Z|71z{O8o=9c9n5hDY96U hn#5*5xU7TSI=D)wMC%IXuK&*o=lr+Fc|Y~Z{sC(pRHy&| diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/autocompletion.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/autocompletion.cpython-312.pyc deleted file mode 100644 index e6a6cf60cd7c2c5afd41d65aefae2ee369f25d47..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8606 zcmb_hX>c1ycJ2WN_kHm`K#?K^@zQBYBrRE%D2dTQQHNzSR6qzbqyT{c-2-0oV2j@M zW`l~mc@q{0#C1vKK;dTW)c_#Y)IRhBA0@_K*) zC@$ApC+RBmyyP*(rpF9SjuLcb?3a?$08)jL4UA_ny$hGG;U zs+W?Uie82ERQ4+2sSMFWs$LaEc<7LNNYkqs()MbHbiKMEeXm|>SA`5i#$KaDt3#$C zbFX>G(rc04H6iPet=9&$jnTbD_u3gft79FdFg9agos5yyvKr}4%j&c316sxeoG!-9 zIv5N5w7?_v7*I1-_);{X@Yt`ud?GR!@VldzeNp#tWQ@g}+cy%8_#;EZAvPL_gx!HK z7xjfgEaSd3;qLDr4h*{sU;F!8^xqOwobphj%757x2zzKzc`!U7YP+H=_FW3GB7Gvj zMMZ7*Fp=jA0jJ%MSzna(4gtS+*oQe5dlaGx$ zZzL4;vfP?H}zPkfl87!{)^LopPUS10;@jTo9plr@M^ zI>aNMVpLbbcH~P;5jDw!$YaEmQH$IbwaQdp20e&kphZm0qr>PT$cbrq4Ns3Ej=HAh zH8C}X!W4~obwcrGkXJLBD_ao4uBbzn6m-f|;mN3E`4r+6!Q#RfnPRjuhY^iamsrEw9DND4C2@^0Ux^oN~Ff@XUYL^SWSVpknK;kl8vyk3@pD>v@NqpUAsRT?+um{DBopi#^S;~H@TZv<`Y^Z0a(bE9=w+5?|3mePZqAROmSw~&yd_vK zw-=s4&})4GU0wlgt-Lk3b^SMlqD^vJutlZ{&ntTnS;IBF3VLrVw96E=R9$-&b_%Qnc~6!RS@xW)BH*y|_KnU1 zCU0TJLY6+rErGS-0DcvbFzxg!rFCmYs1N)Uv&Wpgop+2lSRZ8-pR}^fSE#$oU)BX1$InSZQiufX2xpuwrMe=%! z74yZQtvP2UHE^4GGh>lvq?I%<6DwS7fG;I{N!CKKQlM1CR@f@HF$%_71`$^}p;UY+ zW0t&9x@IMp%JYPk?4ePD3e&tYxK~p0nF=&2%V`8QkCpLdK{zwk7w{9Yc{zM955Bnm zD?%W9gI)z4q6g$Y!IxyJ@Z`&7&RE6sPEq_aUja7i0A2rYyM_LtyG61weQiID!gcG` zNyRh%cddy-YfhfRO32TaE_jTw2M-l2B2zp~cE!4VwLz5O*RE?9Z_n;4*speZEcr?O zfYWwDQH!Dktw^N~VTh58I;MCHC6w3xaSV+sUqfS*UqO+R!7v83MA`C`Jx3{H&e&So zmFF<$sj@r;oF2#3w$5-APq~!@lU{r__s>M3iqybyd;a{FHKgHi?P|Cu(v? zP*egu6jj`CC=eBuFdtFh6=vA+PK+ZMYl#AM2#_cy=txv@6P!r1pj+a;v4ntPGK@SZXy=Vi46~vsH_#xCj0{8PHRFnE-|#RSW-vKXMbiL| z40*|Hvv@rStHDz#s{LaT%y59~$y3z$M=%DojY*so4TTi}F35`JL`$JHYZY4R>H@V# zLQHnbd4q~d@SLa!`9+$TR@4=~lx)y}=*D4kMuW%l%TZLq*pdQX@Qz50i@IUVj%H0n zjf&J2(US8<-Y_c3E2289wWx!qH{`p-hD0hL((ne}9SHfNQ8vsnk|d5`LHGBSr~X52 zedpmK!}XnsFcJ4jpZ_u&4qRdTh9~+?2K+bz-$ny|mg{SQY4@E7TN~0WL|c{!pOx*_&9)@Puew)5PNQpcDb)Tpx|lBwA4;ZTYQDA8ft3RnRnL zwC2ZZWHe8=O|?z$pW2^nS~4_zgg?G|`|8I(y8WX&p{3U2z_V zm>N^2=6T=zrA5>B$s@}~%e#tM{fs`@Ay~JhjGoEP6@^me%GgVit~uL{Qo&vu*F4rD zvpZ#~0e*YYyRnpg^W^c2**bX?G^*b^*?r$y6sI!=OQQX*p(<0gDc*h0Xj{=BTk(n> zX^RETrf&>o$$`6uIv8*I*wnE^&$Z*<)HSB-UJ&YDSm081?ThaAM@U(t|J<^Il;?heusUY7Guou>T?346D@j{73D!+> zJ5$z%MPtKV^p`+*yQyUs0(dtZG>+8-i?%|4@ni8*(#m{IHQ zg`PXE&uyQT3N43H&cjg8+Y&vq7iKOb&tC7FtGVu7Y&jIyEjkb1+tRvZKXO}hV<7qZ zT+PDvJ6(5ob|wxcUVrC>c}-j+*pK|KzIn-h^yc6VCh42AE!aM-_@v^_;Md!aWVz-B z3D?mFWjiuPd&X$X7>m-zO2Jr}HZ}{!=2Z<{u8&*5(8ls*@QG%jI^{eV@4VN%BUQBX z_W8MUNjm9!XJY<*{ODrQ&ih4`@gtd%s^sf8bn&ilEUtv_?TO?oa|ds{Hize_?W`A^^-J1@<J-73N{Wte79Qw5TlWw8*kl;KtbNrzSmDE4hqv|~i zqjyF=52vcm!v?WbEmv(?s(oQ$>~l-1_H?SMCt-M?HO2kYm!~e@Y)=&*yV;)fE!2G4 z^hwj5v9EU={TphoT_`@5GIuR$|7@Ao{Os79$KpLdJ^tO}Hx$TN^axR8+d6-)Sy2M@ z-;Z{|I)3tK8?1sSkMzjZNjc`d4j+o(}rol#VkL zz33=9qc$#XrJ)uqhMp>=P)R|J_kmzcKrKE4l|KjYfkcQzZ1TRE0PrwN5);%Fi2hu$ zl%$~KBtw!B0I#^I0Dc+GTjXM}0oU`i zlan1XXoc}!+gMT-a#H9lqS3X#rO|}aqxGEt!yjd%kUM1E1aA<$NCN9x2)U)de;^QI z`}><9b;ty8mLOdH{cb>;ry^n2qt}zb+6L&yH{{M@jy5+3n9A+rB|h6GZdl|f)YoBJG0&Nu1{06c&Zvw#7Vint-V`>$lj%qAb({$TQw zQT{Xw5CdS6kHRGVb?PbAip$6wS*`xH3!H3-o$y<&gX%b<$i_eyVXMR_*dS;W{WUd9 z!EICzPNM!!IYxOD9%U9Z(Zmy?61;HgHUa>yXxWClKnh0&f#Jy>f2!*k1kf`+jQ3w{ z{A0kfr9*=0-2Rq4^Is7;MF0Op-vJOoimmp0ZKoG{5(gJcw=Frh32mpJaAceP*leXd zT1*l)67Iz$)q-200z@9g*i9(S5E})fV^Qe~Pk2=84heje@K6zsWMjc@In^;Si&XJI zaDNXKXyQb0X1@O8iKIF?GWYt838BQ3F4-lN>{>XxRI)E)aoo2!;>wJ{JiT{n@4Msa z;%1?^`L3ZkV{|UtoQbp7OJ=KPs^*lxw$;TCW+1MxN#9IBu(;EfdcjgZ$E7Td^9Kb> z%R)`cvNKa$Hv7gqZ`|@jVBKoZw7vMt#?KnlZO4VS<6j(FYC8?Vc&q)tkfOm30HLD# z{q`H}OV0X-24vq1P{ZnkNMy9Gg60UfljD<2x+nP zWh$TgeOnvX4#1xCarEq>9M)Da5r!*u1E`9|YNwr4XwYJZR3EI}nBZ(Pa zCC?(Jeop@l=J1>z8;z(aj3}SqkZOm&nns|SYjVfP)N|yxzE2Ls zliSbfztPMz5J7Wx&uQ_m`NK@*Bd+ZlfU|$yr4HgrfP?02xFN9x{;QJWOp}}3wB2yi zc4KyMB;>}Q_VqJ6^-LHdM>#ofS0|x*kIDz4-6Rk_$LAaTiN;eO z=1!1}(%DmM(A>2hJe6A&dtg-{Da28yJTxXZ^lYHhh9h{$?}Pg`2{RD>`p72|>O?K^ zJW1DZB#^q@lTy^f7vbe<{RWwv4MN>5;N?iF|2sL>+0^$Ojd35`t;cSUrCSdOtp`4* zQmrqiTTcqDC%?G-b?b#|l;CK5!jZ%ACp-63pKA6~xEEfq9V(BZpgeX%gQ)Rc;)oL= zP2-1*0VJx2q8x8L_-)NrK zq{_Albn#?Ye0jY=rtur57xOw4&zMFjuo;y9M%?DD>m*Tn=NEMVP*}I0? zj}#wkZfia^-!?CBOIzAA_VS;1lSIgcsSDE=r!FRUer>8=wmWAVW*TOjW}1={^P5uk z=9IDdp#v0n!V$3jlTLHz9_nB0o1yk7FljpFDHb(eFB9>5y;)!*Ey_Ch1j#K(>_j7x z5SPulsR13rIfKbg!emQea`(U_(2A9i3Tnxxf?6OI)UxKE@@y)zXdO@PuX8E6naww@ z;~wToVkMzGm<-bMSQfS+MppM_7mwtoVR@Fg@vaa+hU6eIrv?d%=T`Lk@ yRg-;swlCD)u9WC+RB*X{^Wv%scvrSnP%U$g6$Fn3PI}xK|3dj$>@jJP)cYUCF}1${ diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/base_command.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/base_command.cpython-312.pyc deleted file mode 100644 index 220b0262ce67b288c67ce638e04969204408cb0f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10196 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 S_HSF56WNmb&k-rmHU2lIzXUA+ diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/cmdoptions.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/cmdoptions.cpython-312.pyc deleted file mode 100644 index 6c672e8f8bfbead45c356effef39d57d95651519..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 30395 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 O*QF=FM`Of>*Z%@#kcv70 diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/command_context.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/command_context.cpython-312.pyc deleted file mode 100644 index 6f21702669b2fc0bade0ab2d5064533010764c51..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1774 zcmZ`(O>7%g5T3VxUZ-{)#|bSFMNZ>Z*+?Yi5D6-T;-<9XCl*Q#QjMk6cK4-hYp>1Q z&5xrTqAG$^f+M7eGg2XN;e>hsdgfxcsS+)f8pH)|RxOmOaA4l9?Iuxp($1TiH*em| zH#7TdJl+GiT0ebkD1-=h0?Xu%uOKqwS-hSTc=Jm7k!EIxAr=QX#qqi=@zV z2{hGDO;ySmvDFFHUVPK>6-K=|=9r}Ihi?dQ@U?Qtd`{NEO(0=-M8Asfy*q61v5zT^+|5JjxuSG;WscaWh1#6VO<-#uE70(hG;cIMzu-MGH5L z9ryQa`r0xJ&87Fx+bq z=hyi>|H+ZuNgdltjoeF()YXxCVq_7*iI2%O8($HZ*@4Fw|)6jMWwiBtXMCM*1Q&%(fL}rJJLyg4HR${!7 z7_Xn1yr0O{)hru8YkW3_TTVN|r`R#d6sNdH*UJ@auEhD6u75OVl){r>av1*)Fy7(p z7$-b!#s?+{6=yvhYLk{qx{fWB!lR&tCW97YN4O~`{2$|Oi~2q7QkU#2!5=7y z*;l+;)po9i5eEtI5W?7bK;J;^|cU#*V#!Fc@^E$NSY!cJg)d`c5QwlTP?6P1Tg9C$!9jluge{TFz2wI%7%Gnl&M%b1tI6@Cq{F7tbiX0>^%+x^lD1SuGH9@` zZz^vYI7~f%hs#;StN0FT~&ht4nz9qi>S^=K%Cb4R3lYFgP z>nPl1n58XNl>-Qw6l;sMj>5ZK$B|hQAc{vJx-jqbygKmxJjam&H`!c6sdWlua(iu1 z>$vph`2s%~t4-HB7P2)flX17V-rlQrT`I6Qf067a7L#pU+(T$9M>c#_WF6b>DZ)Db z2k>3fWXAIWndbiAbbR&cB*fta|200v*LbeEBd0YpeMv`tvgnj)nU+CS>5`gB1EWjt z&RP?|URPAg*i18#uYZz!`QQPH?>6^xU2Id!WmL&Db;+E{ql`!>L>jPaZXDTXeWSU7 zx&++q^LpAEGiVm@apNjARY^419&^^T!_|YyT5im+eY!C|u2S2psF$Y30m9Vn&^ab9 zXDBtOEl#It4$feQ@>)JAUjcD3DBT7h!Wh>aw1 zRsfc2W3#euWYQ=zCPjv{z0(saRTCZxN}Di3wkKoFn0URCB>iw{@U_wRO}M<#_YB35 z-brUB)Euat(fsV_J6eVsuy#eusOD%=&0QHC(k_jfnx*#T)0xZZahQg4_QS`4dIB-+ z&*)k|(`K%IO6F(nE;&P0&=>Oht;+D_0K#OJ?u4^+z|Z_As20dmF<25~MKQM2^GNJ| z>TkFnz7}5IF}&ixP>wcN1i`!g3x8832< z55KUEd%NpDn7iPZ_k@$Lko&$`SgL;#42>=R~wX0D5 z9jF(TfN>;f7{Mrp7ReYG#;f6!0Om$eeG0FNs$UJDG&4gbDBi1rE550q1EcuH1Vse1 z6bhh93A0&0V1^M#3DkN#K>C&7Rc|V?NM^WH!$p#a*v(bdT-BhUIpbk-_4i`0Y<589qg|k*Qi-=sKE}Y1e}=_(yJ|e!a#O z$gjO{r3{b|u;`>fWRzZ3W=Vn@1`FW@;R4z5%fN1$CCkvOZkyn3!Bq7zJ0Q!<+{&_v z=+Xl1f`;yaYObTs=O$fRM*zay0oC`wfBwidvFLAI@yE*t4=)P8j;z4~u|i`u?)J`dW=_5l#I z2`}g&-J}UPt{toqF>MbZ+5%#m_%+Cl5_?5EDg&(z)aQ2BebLQ8uLBKJhH8OSe59py zG59pv@|)>9!MkTxqKBR~wJr7BdHufdxM^T<@JY0(+!!x4b{8AF@9bV_+{Zwdr9<~m zKaRe-C_D*8%i)$%xU(4UTvArTyH}%akD}d6Bg_77icTjDLsg$WmSE*3u=OOgTg$>v zV7}hdlgWAy%DD{TwjeAul0UKo|2Fws5VLx_11oTIU1wGB1|>xuOHb)mA2_a{+!d)R zUrDeZdtnPZqN1XA6}R26S^5(^-3L(YO&$5GzFafDNtL!MvvWQ5Uat<|IUs4Cfog%Q zhN9PxT|2fMlFA)YsUunJNZz$d{illkr#?IKxc~g)j-kcVAHP%alkKfHkKH)7+}yhy z?u8oc#`XDY^UMAY_D;44P9XTArY)kw3f^r-fDPQ@hYmi++Eb8ph-WaCAz5x-V7}T$ z$STIXx8SASmKz6=Nu;o)Oop~hz_8$TpU`9vq}5wu5dJDMcqkPPtN;tyh7%E zJShmUzx03Xk50svQa~M2)n+*`M+N?Gz^}}O;1o?;Kw${8bv#mFsycpXp~j>I->2St zRW_a%fkx|#ltFjZoiB)T&Hn@l^bxichFNDGFl+Y+m}mvI6q5&?yQAQNFZQr6t^)&9 zYPSwlq%9<1NcYyCuRp8!e$NBV+Dj-RjA}OxI4C|_N|+SHdJ(OR)`}It>vj9zOwi+;ap zw<~I%g0~Oxr#uFJ>8&m}W_GLi$O4YvQrz~4YPr*R~0Fwpd2Yz|uWp$Q9N3L2$n((hH zJ?98WoqZJ@#NKh-OssybiJF1gG}o-rqcAhKtDZ94tX(exoS6jQVu5@YCV^wC+uKXq z_ZGMBT?{;F?)Yc^kK!N1KSur#xvQ;6N0 z3UxgSb(Pz9mD-cV_9V;?vH9!AKRSMEyqq|=oOt8b_-8%;*8i{{7B4u9$38l?+Pd?0 zEE6gch%1MO6H#;_bLp%KE@V)aoumr%!?OH~skB~Q5oK91GP2B^ zM`Rz!;$&5bH5H1sfugdfsMjEhvnXRZ`YIa&r?W@3BUNuSqw2aWCpg*z>n3`KCmg@D z8@d!L6eIIjP3Y@@>!%GBddgIvdO+IGA9P_yQOV*SKK%BjItB4TRS%pb@|i z(8n-n#kfHR{gSy1x}$ruu3#{3^2q_Be+|2VKf(^iyhkrXb|YHHSB%N^x1b*aHeR%e zihY4R6Mpuk5LoQ_&5{2Q+SzAkV7^vdTfZj3)Rv%)R|UxD2oQonXqNKM3JIUR?Hqc= zXQ?rjuSQjKzYU_epS%Q2f+@dCX~{83l4UTojFS z1hO3z%z_ON$XBg<3H(N;Q+WJy@qh{W-Q;DEz-eI74*(BmYPqWhhbGH`^#jdDcU&(! zr)q5WH%`Ub~Z7UTB?wJhZO9xYgWCFNkjP$6)^+HZXq&)dG3mKpJBhPOd}`Ec*{EM-Qxq zno6PGN1@*PV(H+CM+Z-o!%Y<-6ga-x5-+tRiYEzP%?O)#d<>S`= zib!HFtq~yxw(i>?X=;1g*mGCBA6sdB_0Jv{-i+Od{pEk6t&p|~9Df>ax;}huc)9)J zN_YeZL)V6u+ma8qFAn`@_)SP+1&;q<_QUx;dFkW;dDt8|neaT6{Mb);PaYBinh@OT?iY~D2hx9@|O(Ele8`?f=HjeE> zEG}lYoXlnyavCs`lX*;nf0sn%>Lwm3YO<~N1c9oW(hXuM>K!LLiaLj$!U|&tI{?Ax z5X;v(*TNVLbCt|wGfToQfL_MA{aE46{NR7XaVg%U6o#lkO}wLuhvT?!cpoRMH4-lR z1qpmXg3n0%f02Vla_}p%^DA=b89Di-r}3H4^_6gRH4tBnl>+f%Aii|;abVAaui|Uw zqIdoGJBs_?TqDqb-up9-i!I0do)hd;CU_6mac8tbpn32XYd*8qaP&Of#T{P~m%9&t JN1(?}`@bubtd{@) diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/main.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/main.cpython-312.pyc deleted file mode 100644 index 02c43318af30d751b525792a4666a0874760749c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2300 zcma)7U1$_n6u$Gjvp?BQ*0_nQ(M{Zlt0ZZtV3o9JO(@bBB~n_a3(L53-Of5YGu)Z{ zY}jZXq$D83x31J;A5-E(-`XdsVxiEN1Ot)LP@(q0w`fu*`qFb}Ga0onz3k5U?sx7z z_sqHHocX<}DS}{Z-2GZ(q6qy-C;kw&z{Xtw-yjvqNM%%3XJp2ctjv0nlQ~cFGVe)2 z7C>^kSP99YiX=-6-N)l!4s5e#Oa@@lqJyB_vn>{S)EtRC4WDtum zTRz}B6-eB=$TVj1PBo^6v4lf7wF@ZvS0lLX60b(D337Xt%{5JB&JnGmkZMM;EL|%q zj%FGeY&fK9nVRABMxIk+BRSUPhc(-ArLz`YqUbrsjVWV}Su`t_jvWm1CQBhUE&x+e z3>AoYkzmDnlabD8)({~k0c}*p7Qu}pU0xw2+Z73R#)whedINeiqVeY7zp=p5c-PR( z4u|sICZMyy9Oe`I+D+-+#P$RdwhK$Ypn0jd9)unNnevi#F*yZh-s^@31qn~z&=ui8uu;wZrdt&<*QKnf7)=` z{VnCRj{@oY3zoMtqP!>2`$%Pil^38Qr@1TK^`9DYf|)>*+yykjz&i)=&H8%-3313w z^Uiinm2-j4M{20X`*CKFcNTwzzy2GX1u{4*7a4g@U3NvSWS9g`C3+88mrDhLZF7ti zv7O^wQ7O_=awRAMTrx@36)UDXrsJv5(CFyd(c>8xO?989`!%DK!L)2L-F9Z;D{N$J z>^PLw&7uN3B0W8tZR%i%)}6GC+g#4B+Fo^yyV8V04A^A5@k^TSV6r_=4hE(8d1Da? zQ}f9%jNN;=D1?R zQ8x^ZZ$WYRRN>_Cuq${Cfv%vYT|vbc$4Vqb?eY~xGjbxK-5BCjQ45Ar!42d!3=U5g zT?V_Mt)^pBpEBNfb>YcaiVK#&~{M zyO_7Z`yNXvURFv#Tbk8ZfF_6u^t-;IuJuuOdaWwyhEu#p9{?LVIhsL_LeZJplSFcU za&EHD)Y;kEEWa5;Lenb0dzs(;DABgkb!6$ok-s@s5H}-8XkO(HEb|9e_=C^*=4&JC zydbnb={U4-fAQm< zU)q=YPp`y3o%!TxG&w&oH&B-zMmv}I&VM#LP^|SO;=rmDosY~#8b<0zzkmPx?)ttJ zDfbqCD|tJ8Grb~ZDJ#uMH(KVSbJ5r9Au#ZIGl^oJ^_PeRkDf{K`Pf`+Y3IPt%Bpl? zSvv80U7(6rc0WYqUQ7JrEV`G;4(>(wv*E$r-2I-D*})X|ASJ-$!QLH1E!>0t)KHB3 zH6{SgF(bKMZm3XD&0?WIXp@8lf_#8%CA8oP?I$EfdC{p_l&1Zf@RZ5GMfGl9R4-gq z?;BqxC^0Wx-H!h*qxV_xzFNe~W;gZDn#QNhZe(No+e%11?0pRFMiK0ITRrAzy4|Rv za0xrZjqH{RUJAoY?x(JEwnJPFa?H!7AJ7Jnh|uQ#N*ZW%gOA8@n9^oqe+iFuj$xP= zEW_{{B4Xl?5&XiBQRFd7{DC_ELdVw7(KXb+hI-b};iuvBY};x$y&O*0KX@3<&WJAt dI+?E7$vR&@e>1$C=vqfGUnp);L5_j_e*q8FBV7Oh diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/main_parser.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/main_parser.cpython-312.pyc deleted file mode 100644 index 7628865ad627e76460a1b8837852c944f1965dce..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4906 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 diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/parser.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/parser.cpython-312.pyc deleted file mode 100644 index 8968829bddd264c205370681fe94b05c8ecfcebe..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15013 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 Z(9(OK!^8dVQx3jozGdg{ILfdA{(sNjjRF7w diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/progress_bars.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/progress_bars.cpython-312.pyc deleted file mode 100644 index a18b96bc86d021f734cfd6c5aa14e73e5a185b02..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3834 zcmahMTTC0-^^V8m8QX&mHV*>&&-Jc1_flN>iW<*)c)X8<3kQ~Z{<#486uFo{c z4Vj1>$wcL-!|&A_Gfi@ngL8UwrbTXH5N0ulW36D!7=wKkEbXCdxmESKdn%9HQ~|fE ze(c3Dl~I{AOL@|Q8o23|_iW=7Fpt~m-Y#CB8U)@BkQRT%R5fn#wx)e*-A$j|InBgF z&p;ksiu-J#SJ8E4RL8cz9~+n`R+iYp+ZJGySiv0H<(#Eu4Mn%P;e1ZV@qq0=r;y&P zp3fL|s4qKZ=vhT|*KcY%9@1uTbuGL@vSS3BrX6yz>Q3EIPBY-c)n#B9U$d%s!?2dY z=XE7#VpngJk_K@`(G1NPtMWyMiDH;(OonpU<%~0Sa8R-E1udh&VPF{^!&Xu=RBTuZ zJ&7&QjmjxU6_T{3bJ+HpV6*Kb*vbQiKc4-LjqU`?4meoSd>V{Y4(!N=*6?Lfvuj*F=8uJTgmMd@*O)hO=4CLui&S@E>{ZFhJoZXe7Kc3O=jd zB^7wr;-!$Qk3k_cKc|C%02(ZLh8wc&uXm*Jr_PIg>iI&0Q%)^;dsdA(h!*zp)XBB6i}}r5en)lpbt)HA|>=ui3q?J6je=9 z^GG9I`tmsyqRD1;Y{UgZpOZ9GK%bD%xUl(@lL|~*aL;Pc~N92v~2S?QvZ8pnn@MErrdI1U9aYT+l{GOo-ScIVClz zi~%{P<&Hu?Ll8q&I-1h8qwa^35YJ{JH*H6fL`%VSRWF~?NdkZDy(04Sjz?bV$L(>kCruPSSayJ+AD`H1Qwp_RJwU-DN+yzH;#7qhrelPcKVve0rr6>RJg6%=O#;hNn#(^XJzi zEw={N#m3v07cMXMFAY6xdLYkV{!Bdi6^EKTU$7_;xgA@G{m1Nt1K&7zviBtO`%{N{ z53`T=`vHG^nCm^kKR(6-UdwniWBK6cWW0X?QptEKV*y?ns%IO{stC%9_ojE&qjIxs zfw{(*5JiA{x8*#a8;i4?2NgU8w&2~0J}0`*L3H~H>{d=J)MRELcY$VZqLvq3OSUqk z$x)p&CzIB1_yRXk%Wb=5A)EL?L*5*Zd&u{}9OsxOwOx|P5t!_d!&M8SnRt>Y zQ?(_Hsl;hLn^JW1OkyW%4~Uyb0nDKdA$U8m5Lgpp%R;Qwns|8d(TUQrp5@k_RpH!@ z&Ci6kO%OqZ+EC4erL%F4W>SmLt0wTi$mR%E)M+}I*oyO&oaCgVA5xy$S!@xS8l2QC zjQ}|enMl$Z6oAKl2uVN}dRKxb{!jGJ7)_HDS2>V&|$Tt%suT2fqrUNZY?n8kRSDx_e${ z?lVsMO##6f^unlv(Q>_61OP1xxR(yG&7q>ZIowR9^kD#rkUJ`fnBLcIW> z+Jul$U;M!nzGWjETMJ9eVQF>m@kir-7+wtzmW07|e?#%g6Mx&b-Pay9u7=N)gfm}; zqw_)14SM2Tghm&64S?-~-WA$mLLZILmn75;2z?PkU7gT3uv?*ECX!d-cC#cw>te#a z)omSs(;CnYfY4U4(k?iSgVQNE{Q+$p>^4W4TYVEyeG_J-kvE+fwcB?xU+kQ%pd?P}xth0P=RU{dX{^Zt4zj^$@%X}#GEY~+t5SYSO`s#k{=teM+0a;a<~;}Q4t;#l!9V9Z Ko^!D0oc+HJW1PMK diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/req_command.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/req_command.cpython-312.pyc deleted file mode 100644 index 253b6d75c8c175c8da8d738a8ac7fde633fe298a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 12241 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 Vfv<=A-!gNq^?{vV5CquL{ueUe9OM81 diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/spinners.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/spinners.cpython-312.pyc deleted file mode 100644 index 5ca1b1bc1d94c93a70317e83e660edcc01025db0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7833 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 Dl)ljH diff --git a/.env/Lib/site-packages/pip/_internal/cli/__pycache__/status_codes.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/cli/__pycache__/status_codes.cpython-312.pyc deleted file mode 100644 index 150d99e43a8d52d0eec94586c3765b337f6dc6fe..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 375 zcmX@j%ge<81fTb{WtITx#~=<2FhUuhWq^$73@Hpz3@MCJj44b}OexG!%qc8UEGeu} ztd(q<>@Pt|UxGBeWC9V)Ac6%%hyw{fO}1O?!J*F1uED{#SY3mH{DW@shWdH?`G@<( zL%1Sgo%gX5h-J$+o_T|9%J>IMA#<9!`N zoIPBFA!=7Ldn;Jz#wr$So9);KoLJsDF9!TYT^I@ diff --git a/.env/Lib/site-packages/pip/_internal/cli/autocompletion.py b/.env/Lib/site-packages/pip/_internal/cli/autocompletion.py deleted file mode 100644 index f3f70ac..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/autocompletion.py +++ /dev/null @@ -1,176 +0,0 @@ -"""Logic that powers autocompletion installed by ``pip completion``. -""" - -import optparse -import os -import sys -from itertools import chain -from typing import Any, Iterable, List, Optional - -from pip._internal.cli.main_parser import create_main_parser -from pip._internal.commands import commands_dict, create_command -from pip._internal.metadata import get_default_environment - - -def autocomplete() -> None: - """Entry Point for completion of main and subcommand options.""" - # Don't complete if user hasn't sourced bash_completion file. - if "PIP_AUTO_COMPLETE" not in os.environ: - return - # Don't complete if autocompletion environment variables - # are not present - if not os.environ.get("COMP_WORDS") or not os.environ.get("COMP_CWORD"): - return - cwords = os.environ["COMP_WORDS"].split()[1:] - cword = int(os.environ["COMP_CWORD"]) - try: - current = cwords[cword - 1] - except IndexError: - current = "" - - parser = create_main_parser() - subcommands = list(commands_dict) - options = [] - - # subcommand - subcommand_name: Optional[str] = None - for word in cwords: - if word in subcommands: - subcommand_name = word - break - # subcommand options - if subcommand_name is not None: - # special case: 'help' subcommand has no options - if subcommand_name == "help": - sys.exit(1) - # special case: list locally installed dists for show and uninstall - should_list_installed = not current.startswith("-") and subcommand_name in [ - "show", - "uninstall", - ] - if should_list_installed: - env = get_default_environment() - lc = current.lower() - installed = [ - dist.canonical_name - for dist in env.iter_installed_distributions(local_only=True) - if dist.canonical_name.startswith(lc) - and dist.canonical_name not in cwords[1:] - ] - # if there are no dists installed, fall back to option completion - if installed: - for dist in installed: - print(dist) - sys.exit(1) - - should_list_installables = ( - not current.startswith("-") and subcommand_name == "install" - ) - if should_list_installables: - for path in auto_complete_paths(current, "path"): - print(path) - sys.exit(1) - - subcommand = create_command(subcommand_name) - - for opt in subcommand.parser.option_list_all: - if opt.help != optparse.SUPPRESS_HELP: - options += [ - (opt_str, opt.nargs) for opt_str in opt._long_opts + opt._short_opts - ] - - # filter out previously specified options from available options - prev_opts = [x.split("=")[0] for x in cwords[1 : cword - 1]] - options = [(x, v) for (x, v) in options if x not in prev_opts] - # filter options by current input - options = [(k, v) for k, v in options if k.startswith(current)] - # get completion type given cwords and available subcommand options - completion_type = get_path_completion_type( - cwords, - cword, - subcommand.parser.option_list_all, - ) - # get completion files and directories if ``completion_type`` is - # ````, ``

`` or ```` - if completion_type: - paths = auto_complete_paths(current, completion_type) - options = [(path, 0) for path in paths] - for option in options: - opt_label = option[0] - # append '=' to options which require args - if option[1] and option[0][:2] == "--": - opt_label += "=" - print(opt_label) - else: - # show main parser options only when necessary - - opts = [i.option_list for i in parser.option_groups] - opts.append(parser.option_list) - flattened_opts = chain.from_iterable(opts) - if current.startswith("-"): - for opt in flattened_opts: - if opt.help != optparse.SUPPRESS_HELP: - subcommands += opt._long_opts + opt._short_opts - else: - # get completion type given cwords and all available options - completion_type = get_path_completion_type(cwords, cword, flattened_opts) - if completion_type: - subcommands = list(auto_complete_paths(current, completion_type)) - - print(" ".join([x for x in subcommands if x.startswith(current)])) - sys.exit(1) - - -def get_path_completion_type( - cwords: List[str], cword: int, opts: Iterable[Any] -) -> Optional[str]: - """Get the type of path completion (``file``, ``dir``, ``path`` or None) - - :param cwords: same as the environmental variable ``COMP_WORDS`` - :param cword: same as the environmental variable ``COMP_CWORD`` - :param opts: The available options to check - :return: path completion type (``file``, ``dir``, ``path`` or None) - """ - if cword < 2 or not cwords[cword - 2].startswith("-"): - return None - for opt in opts: - if opt.help == optparse.SUPPRESS_HELP: - continue - for o in str(opt).split("/"): - if cwords[cword - 2].split("=")[0] == o: - if not opt.metavar or any( - x in ("path", "file", "dir") for x in opt.metavar.split("/") - ): - return opt.metavar - return None - - -def auto_complete_paths(current: str, completion_type: str) -> Iterable[str]: - """If ``completion_type`` is ``file`` or ``path``, list all regular files - and directories starting with ``current``; otherwise only list directories - starting with ``current``. - - :param current: The word to be completed - :param completion_type: path completion type(``file``, ``path`` or ``dir``) - :return: A generator of regular files and/or directories - """ - directory, filename = os.path.split(current) - current_path = os.path.abspath(directory) - # Don't complete paths if they can't be accessed - if not os.access(current_path, os.R_OK): - return - filename = os.path.normcase(filename) - # list all files that start with ``filename`` - file_list = ( - x for x in os.listdir(current_path) if os.path.normcase(x).startswith(filename) - ) - for f in file_list: - opt = os.path.join(current_path, f) - comp_file = os.path.normcase(os.path.join(directory, f)) - # complete regular files when there is not ```` after option - # complete directories when there is ````, ```` or - # ````after option - if completion_type != "dir" and os.path.isfile(opt): - yield comp_file - elif os.path.isdir(opt): - yield os.path.join(comp_file, "") diff --git a/.env/Lib/site-packages/pip/_internal/cli/base_command.py b/.env/Lib/site-packages/pip/_internal/cli/base_command.py deleted file mode 100644 index bc1ab65..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/base_command.py +++ /dev/null @@ -1,231 +0,0 @@ -"""Base Command class, and related routines""" - -import logging -import logging.config -import optparse -import os -import sys -import traceback -from optparse import Values -from typing import List, Optional, Tuple - -from pip._vendor.rich import reconfigure -from pip._vendor.rich import traceback as rich_traceback - -from pip._internal.cli import cmdoptions -from pip._internal.cli.command_context import CommandContextMixIn -from pip._internal.cli.parser import ConfigOptionParser, UpdatingDefaultsHelpFormatter -from pip._internal.cli.status_codes import ( - ERROR, - PREVIOUS_BUILD_DIR_ERROR, - UNKNOWN_ERROR, - VIRTUALENV_NOT_FOUND, -) -from pip._internal.exceptions import ( - BadCommand, - CommandError, - DiagnosticPipError, - InstallationError, - NetworkConnectionError, - PreviousBuildDirError, -) -from pip._internal.utils.filesystem import check_path_owner -from pip._internal.utils.logging import BrokenStdoutLoggingError, setup_logging -from pip._internal.utils.misc import get_prog, normalize_path -from pip._internal.utils.temp_dir import TempDirectoryTypeRegistry as TempDirRegistry -from pip._internal.utils.temp_dir import global_tempdir_manager, tempdir_registry -from pip._internal.utils.virtualenv import running_under_virtualenv - -__all__ = ["Command"] - -logger = logging.getLogger(__name__) - - -class Command(CommandContextMixIn): - usage: str = "" - ignore_require_venv: bool = False - - def __init__(self, name: str, summary: str, isolated: bool = False) -> None: - super().__init__() - - self.name = name - self.summary = summary - self.parser = ConfigOptionParser( - usage=self.usage, - prog=f"{get_prog()} {name}", - formatter=UpdatingDefaultsHelpFormatter(), - add_help_option=False, - name=name, - description=self.__doc__, - isolated=isolated, - ) - - self.tempdir_registry: Optional[TempDirRegistry] = None - - # Commands should add options to this option group - optgroup_name = f"{self.name.capitalize()} Options" - self.cmd_opts = optparse.OptionGroup(self.parser, optgroup_name) - - # Add the general options - gen_opts = cmdoptions.make_option_group( - cmdoptions.general_group, - self.parser, - ) - self.parser.add_option_group(gen_opts) - - self.add_options() - - def add_options(self) -> None: - pass - - def handle_pip_version_check(self, options: Values) -> None: - """ - This is a no-op so that commands by default do not do the pip version - check. - """ - # Make sure we do the pip version check if the index_group options - # are present. - assert not hasattr(options, "no_index") - - def run(self, options: Values, args: List[str]) -> int: - raise NotImplementedError - - def _run_wrapper(self, level_number: int, options: Values, args: List[str]) -> int: - def _inner_run() -> int: - try: - return self.run(options, args) - finally: - self.handle_pip_version_check(options) - - if options.debug_mode: - rich_traceback.install(show_locals=True) - return _inner_run() - - try: - status = _inner_run() - assert isinstance(status, int) - return status - except DiagnosticPipError as exc: - logger.error("%s", exc, extra={"rich": True}) - logger.debug("Exception information:", exc_info=True) - - return ERROR - except PreviousBuildDirError as exc: - logger.critical(str(exc)) - logger.debug("Exception information:", exc_info=True) - - return PREVIOUS_BUILD_DIR_ERROR - except ( - InstallationError, - BadCommand, - NetworkConnectionError, - ) as exc: - logger.critical(str(exc)) - logger.debug("Exception information:", exc_info=True) - - return ERROR - except CommandError as exc: - logger.critical("%s", exc) - logger.debug("Exception information:", exc_info=True) - - return ERROR - except BrokenStdoutLoggingError: - # Bypass our logger and write any remaining messages to - # stderr because stdout no longer works. - print("ERROR: Pipe to stdout was broken", file=sys.stderr) - if level_number <= logging.DEBUG: - traceback.print_exc(file=sys.stderr) - - return ERROR - except KeyboardInterrupt: - logger.critical("Operation cancelled by user") - logger.debug("Exception information:", exc_info=True) - - return ERROR - except BaseException: - logger.critical("Exception:", exc_info=True) - - return UNKNOWN_ERROR - - def parse_args(self, args: List[str]) -> Tuple[Values, List[str]]: - # factored out for testability - return self.parser.parse_args(args) - - def main(self, args: List[str]) -> int: - try: - with self.main_context(): - return self._main(args) - finally: - logging.shutdown() - - def _main(self, args: List[str]) -> int: - # We must initialize this before the tempdir manager, otherwise the - # configuration would not be accessible by the time we clean up the - # tempdir manager. - self.tempdir_registry = self.enter_context(tempdir_registry()) - # Intentionally set as early as possible so globally-managed temporary - # directories are available to the rest of the code. - self.enter_context(global_tempdir_manager()) - - options, args = self.parse_args(args) - - # Set verbosity so that it can be used elsewhere. - self.verbosity = options.verbose - options.quiet - - reconfigure(no_color=options.no_color) - level_number = setup_logging( - verbosity=self.verbosity, - no_color=options.no_color, - user_log_file=options.log, - ) - - always_enabled_features = set(options.features_enabled) & set( - cmdoptions.ALWAYS_ENABLED_FEATURES - ) - if always_enabled_features: - logger.warning( - "The following features are always enabled: %s. ", - ", ".join(sorted(always_enabled_features)), - ) - - # Make sure that the --python argument isn't specified after the - # subcommand. We can tell, because if --python was specified, - # we should only reach this point if we're running in the created - # subprocess, which has the _PIP_RUNNING_IN_SUBPROCESS environment - # variable set. - if options.python and "_PIP_RUNNING_IN_SUBPROCESS" not in os.environ: - logger.critical( - "The --python option must be placed before the pip subcommand name" - ) - sys.exit(ERROR) - - # TODO: Try to get these passing down from the command? - # without resorting to os.environ to hold these. - # This also affects isolated builds and it should. - - if options.no_input: - os.environ["PIP_NO_INPUT"] = "1" - - if options.exists_action: - os.environ["PIP_EXISTS_ACTION"] = " ".join(options.exists_action) - - if options.require_venv and not self.ignore_require_venv: - # If a venv is required check if it can really be found - if not running_under_virtualenv(): - logger.critical("Could not find an activated virtualenv (required).") - sys.exit(VIRTUALENV_NOT_FOUND) - - if options.cache_dir: - options.cache_dir = normalize_path(options.cache_dir) - if not check_path_owner(options.cache_dir): - logger.warning( - "The directory '%s' or its parent directory is not owned " - "or is not writable by the current user. The cache " - "has been disabled. Check the permissions and owner of " - "that directory. If executing pip with sudo, you should " - "use sudo's -H flag.", - options.cache_dir, - ) - options.cache_dir = None - - return self._run_wrapper(level_number, options, args) diff --git a/.env/Lib/site-packages/pip/_internal/cli/cmdoptions.py b/.env/Lib/site-packages/pip/_internal/cli/cmdoptions.py deleted file mode 100644 index 0b7cff7..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/cmdoptions.py +++ /dev/null @@ -1,1075 +0,0 @@ -""" -shared options and groups - -The principle here is to define options once, but *not* instantiate them -globally. One reason being that options with action='append' can carry state -between parses. pip parses general options twice internally, and shouldn't -pass on state. To be consistent, all options will follow this design. -""" - -# The following comment should be removed at some point in the future. -# mypy: strict-optional=False - -import importlib.util -import logging -import os -import textwrap -from functools import partial -from optparse import SUPPRESS_HELP, Option, OptionGroup, OptionParser, Values -from textwrap import dedent -from typing import Any, Callable, Dict, Optional, Tuple - -from pip._vendor.packaging.utils import canonicalize_name - -from pip._internal.cli.parser import ConfigOptionParser -from pip._internal.exceptions import CommandError -from pip._internal.locations import USER_CACHE_DIR, get_src_prefix -from pip._internal.models.format_control import FormatControl -from pip._internal.models.index import PyPI -from pip._internal.models.target_python import TargetPython -from pip._internal.utils.hashes import STRONG_HASHES -from pip._internal.utils.misc import strtobool - -logger = logging.getLogger(__name__) - - -def raise_option_error(parser: OptionParser, option: Option, msg: str) -> None: - """ - Raise an option parsing error using parser.error(). - - Args: - parser: an OptionParser instance. - option: an Option instance. - msg: the error text. - """ - msg = f"{option} error: {msg}" - msg = textwrap.fill(" ".join(msg.split())) - parser.error(msg) - - -def make_option_group(group: Dict[str, Any], parser: ConfigOptionParser) -> OptionGroup: - """ - Return an OptionGroup object - group -- assumed to be dict with 'name' and 'options' keys - parser -- an optparse Parser - """ - option_group = OptionGroup(parser, group["name"]) - for option in group["options"]: - option_group.add_option(option()) - return option_group - - -def check_dist_restriction(options: Values, check_target: bool = False) -> None: - """Function for determining if custom platform options are allowed. - - :param options: The OptionParser options. - :param check_target: Whether or not to check if --target is being used. - """ - dist_restriction_set = any( - [ - options.python_version, - options.platforms, - options.abis, - options.implementation, - ] - ) - - binary_only = FormatControl(set(), {":all:"}) - sdist_dependencies_allowed = ( - options.format_control != binary_only and not options.ignore_dependencies - ) - - # Installations or downloads using dist restrictions must not combine - # source distributions and dist-specific wheels, as they are not - # guaranteed to be locally compatible. - if dist_restriction_set and sdist_dependencies_allowed: - raise CommandError( - "When restricting platform and interpreter constraints using " - "--python-version, --platform, --abi, or --implementation, " - "either --no-deps must be set, or --only-binary=:all: must be " - "set and --no-binary must not be set (or must be set to " - ":none:)." - ) - - if check_target: - if not options.dry_run and dist_restriction_set and not options.target_dir: - raise CommandError( - "Can not use any platform or abi specific options unless " - "installing via '--target' or using '--dry-run'" - ) - - -def _path_option_check(option: Option, opt: str, value: str) -> str: - return os.path.expanduser(value) - - -def _package_name_option_check(option: Option, opt: str, value: str) -> str: - return canonicalize_name(value) - - -class PipOption(Option): - TYPES = Option.TYPES + ("path", "package_name") - TYPE_CHECKER = Option.TYPE_CHECKER.copy() - TYPE_CHECKER["package_name"] = _package_name_option_check - TYPE_CHECKER["path"] = _path_option_check - - -########### -# options # -########### - -help_: Callable[..., Option] = partial( - Option, - "-h", - "--help", - dest="help", - action="help", - help="Show help.", -) - -debug_mode: Callable[..., Option] = partial( - Option, - "--debug", - dest="debug_mode", - action="store_true", - default=False, - help=( - "Let unhandled exceptions propagate outside the main subroutine, " - "instead of logging them to stderr." - ), -) - -isolated_mode: Callable[..., Option] = partial( - Option, - "--isolated", - dest="isolated_mode", - action="store_true", - default=False, - help=( - "Run pip in an isolated mode, ignoring environment variables and user " - "configuration." - ), -) - -require_virtualenv: Callable[..., Option] = partial( - Option, - "--require-virtualenv", - "--require-venv", - dest="require_venv", - action="store_true", - default=False, - help=( - "Allow pip to only run in a virtual environment; " - "exit with an error otherwise." - ), -) - -override_externally_managed: Callable[..., Option] = partial( - Option, - "--break-system-packages", - dest="override_externally_managed", - action="store_true", - help="Allow pip to modify an EXTERNALLY-MANAGED Python installation", -) - -python: Callable[..., Option] = partial( - Option, - "--python", - dest="python", - help="Run pip with the specified Python interpreter.", -) - -verbose: Callable[..., Option] = partial( - Option, - "-v", - "--verbose", - dest="verbose", - action="count", - default=0, - help="Give more output. Option is additive, and can be used up to 3 times.", -) - -no_color: Callable[..., Option] = partial( - Option, - "--no-color", - dest="no_color", - action="store_true", - default=False, - help="Suppress colored output.", -) - -version: Callable[..., Option] = partial( - Option, - "-V", - "--version", - dest="version", - action="store_true", - help="Show version and exit.", -) - -quiet: Callable[..., Option] = partial( - Option, - "-q", - "--quiet", - dest="quiet", - action="count", - default=0, - help=( - "Give less output. Option is additive, and can be used up to 3" - " times (corresponding to WARNING, ERROR, and CRITICAL logging" - " levels)." - ), -) - -progress_bar: Callable[..., Option] = partial( - Option, - "--progress-bar", - dest="progress_bar", - type="choice", - choices=["on", "off", "raw"], - default="on", - help="Specify whether the progress bar should be used [on, off, raw] (default: on)", -) - -log: Callable[..., Option] = partial( - PipOption, - "--log", - "--log-file", - "--local-log", - dest="log", - metavar="path", - type="path", - help="Path to a verbose appending log.", -) - -no_input: Callable[..., Option] = partial( - Option, - # Don't ask for input - "--no-input", - dest="no_input", - action="store_true", - default=False, - help="Disable prompting for input.", -) - -keyring_provider: Callable[..., Option] = partial( - Option, - "--keyring-provider", - dest="keyring_provider", - choices=["auto", "disabled", "import", "subprocess"], - default="auto", - help=( - "Enable the credential lookup via the keyring library if user input is allowed." - " Specify which mechanism to use [disabled, import, subprocess]." - " (default: disabled)" - ), -) - -proxy: Callable[..., Option] = partial( - Option, - "--proxy", - dest="proxy", - type="str", - default="", - help="Specify a proxy in the form scheme://[user:passwd@]proxy.server:port.", -) - -retries: Callable[..., Option] = partial( - Option, - "--retries", - dest="retries", - type="int", - default=5, - help="Maximum number of retries each connection should attempt " - "(default %default times).", -) - -timeout: Callable[..., Option] = partial( - Option, - "--timeout", - "--default-timeout", - metavar="sec", - dest="timeout", - type="float", - default=15, - help="Set the socket timeout (default %default seconds).", -) - - -def exists_action() -> Option: - return Option( - # Option when path already exist - "--exists-action", - dest="exists_action", - type="choice", - choices=["s", "i", "w", "b", "a"], - default=[], - action="append", - metavar="action", - help="Default action when a path already exists: " - "(s)witch, (i)gnore, (w)ipe, (b)ackup, (a)bort.", - ) - - -cert: Callable[..., Option] = partial( - PipOption, - "--cert", - dest="cert", - type="path", - metavar="path", - help=( - "Path to PEM-encoded CA certificate bundle. " - "If provided, overrides the default. " - "See 'SSL Certificate Verification' in pip documentation " - "for more information." - ), -) - -client_cert: Callable[..., Option] = partial( - PipOption, - "--client-cert", - dest="client_cert", - type="path", - default=None, - metavar="path", - help="Path to SSL client certificate, a single file containing the " - "private key and the certificate in PEM format.", -) - -index_url: Callable[..., Option] = partial( - Option, - "-i", - "--index-url", - "--pypi-url", - dest="index_url", - metavar="URL", - default=PyPI.simple_url, - help="Base URL of the Python Package Index (default %default). " - "This should point to a repository compliant with PEP 503 " - "(the simple repository API) or a local directory laid out " - "in the same format.", -) - - -def extra_index_url() -> Option: - return Option( - "--extra-index-url", - dest="extra_index_urls", - metavar="URL", - action="append", - default=[], - help="Extra URLs of package indexes to use in addition to " - "--index-url. Should follow the same rules as " - "--index-url.", - ) - - -no_index: Callable[..., Option] = partial( - Option, - "--no-index", - dest="no_index", - action="store_true", - default=False, - help="Ignore package index (only looking at --find-links URLs instead).", -) - - -def find_links() -> Option: - return Option( - "-f", - "--find-links", - dest="find_links", - action="append", - default=[], - metavar="url", - help="If a URL or path to an html file, then parse for links to " - "archives such as sdist (.tar.gz) or wheel (.whl) files. " - "If a local path or file:// URL that's a directory, " - "then look for archives in the directory listing. " - "Links to VCS project URLs are not supported.", - ) - - -def trusted_host() -> Option: - return Option( - "--trusted-host", - dest="trusted_hosts", - action="append", - metavar="HOSTNAME", - default=[], - help="Mark this host or host:port pair as trusted, even though it " - "does not have valid or any HTTPS.", - ) - - -def constraints() -> Option: - return Option( - "-c", - "--constraint", - dest="constraints", - action="append", - default=[], - metavar="file", - help="Constrain versions using the given constraints file. " - "This option can be used multiple times.", - ) - - -def requirements() -> Option: - return Option( - "-r", - "--requirement", - dest="requirements", - action="append", - default=[], - metavar="file", - help="Install from the given requirements file. " - "This option can be used multiple times.", - ) - - -def editable() -> Option: - return Option( - "-e", - "--editable", - dest="editables", - action="append", - default=[], - metavar="path/url", - help=( - "Install a project in editable mode (i.e. setuptools " - '"develop mode") from a local project path or a VCS url.' - ), - ) - - -def _handle_src(option: Option, opt_str: str, value: str, parser: OptionParser) -> None: - value = os.path.abspath(value) - setattr(parser.values, option.dest, value) - - -src: Callable[..., Option] = partial( - PipOption, - "--src", - "--source", - "--source-dir", - "--source-directory", - dest="src_dir", - type="path", - metavar="dir", - default=get_src_prefix(), - action="callback", - callback=_handle_src, - help="Directory to check out editable projects into. " - 'The default in a virtualenv is "/src". ' - 'The default for global installs is "/src".', -) - - -def _get_format_control(values: Values, option: Option) -> Any: - """Get a format_control object.""" - return getattr(values, option.dest) - - -def _handle_no_binary( - option: Option, opt_str: str, value: str, parser: OptionParser -) -> None: - existing = _get_format_control(parser.values, option) - FormatControl.handle_mutual_excludes( - value, - existing.no_binary, - existing.only_binary, - ) - - -def _handle_only_binary( - option: Option, opt_str: str, value: str, parser: OptionParser -) -> None: - existing = _get_format_control(parser.values, option) - FormatControl.handle_mutual_excludes( - value, - existing.only_binary, - existing.no_binary, - ) - - -def no_binary() -> Option: - format_control = FormatControl(set(), set()) - return Option( - "--no-binary", - dest="format_control", - action="callback", - callback=_handle_no_binary, - type="str", - default=format_control, - help="Do not use binary packages. Can be supplied multiple times, and " - 'each time adds to the existing value. Accepts either ":all:" to ' - 'disable all binary packages, ":none:" to empty the set (notice ' - "the colons), or one or more package names with commas between " - "them (no colons). Note that some packages are tricky to compile " - "and may fail to install when this option is used on them.", - ) - - -def only_binary() -> Option: - format_control = FormatControl(set(), set()) - return Option( - "--only-binary", - dest="format_control", - action="callback", - callback=_handle_only_binary, - type="str", - default=format_control, - help="Do not use source packages. Can be supplied multiple times, and " - 'each time adds to the existing value. Accepts either ":all:" to ' - 'disable all source packages, ":none:" to empty the set, or one ' - "or more package names with commas between them. Packages " - "without binary distributions will fail to install when this " - "option is used on them.", - ) - - -platforms: Callable[..., Option] = partial( - Option, - "--platform", - dest="platforms", - metavar="platform", - action="append", - default=None, - help=( - "Only use wheels compatible with . Defaults to the " - "platform of the running system. Use this option multiple times to " - "specify multiple platforms supported by the target interpreter." - ), -) - - -# This was made a separate function for unit-testing purposes. -def _convert_python_version(value: str) -> Tuple[Tuple[int, ...], Optional[str]]: - """ - Convert a version string like "3", "37", or "3.7.3" into a tuple of ints. - - :return: A 2-tuple (version_info, error_msg), where `error_msg` is - non-None if and only if there was a parsing error. - """ - if not value: - # The empty string is the same as not providing a value. - return (None, None) - - parts = value.split(".") - if len(parts) > 3: - return ((), "at most three version parts are allowed") - - if len(parts) == 1: - # Then we are in the case of "3" or "37". - value = parts[0] - if len(value) > 1: - parts = [value[0], value[1:]] - - try: - version_info = tuple(int(part) for part in parts) - except ValueError: - return ((), "each version part must be an integer") - - return (version_info, None) - - -def _handle_python_version( - option: Option, opt_str: str, value: str, parser: OptionParser -) -> None: - """ - Handle a provided --python-version value. - """ - version_info, error_msg = _convert_python_version(value) - if error_msg is not None: - msg = f"invalid --python-version value: {value!r}: {error_msg}" - raise_option_error(parser, option=option, msg=msg) - - parser.values.python_version = version_info - - -python_version: Callable[..., Option] = partial( - Option, - "--python-version", - dest="python_version", - metavar="python_version", - action="callback", - callback=_handle_python_version, - type="str", - default=None, - help=dedent( - """\ - The Python interpreter version to use for wheel and "Requires-Python" - compatibility checks. Defaults to a version derived from the running - interpreter. The version can be specified using up to three dot-separated - integers (e.g. "3" for 3.0.0, "3.7" for 3.7.0, or "3.7.3"). A major-minor - version can also be given as a string without dots (e.g. "37" for 3.7.0). - """ - ), -) - - -implementation: Callable[..., Option] = partial( - Option, - "--implementation", - dest="implementation", - metavar="implementation", - default=None, - help=( - "Only use wheels compatible with Python " - "implementation , e.g. 'pp', 'jy', 'cp', " - " or 'ip'. If not specified, then the current " - "interpreter implementation is used. Use 'py' to force " - "implementation-agnostic wheels." - ), -) - - -abis: Callable[..., Option] = partial( - Option, - "--abi", - dest="abis", - metavar="abi", - action="append", - default=None, - help=( - "Only use wheels compatible with Python abi , e.g. 'pypy_41'. " - "If not specified, then the current interpreter abi tag is used. " - "Use this option multiple times to specify multiple abis supported " - "by the target interpreter. Generally you will need to specify " - "--implementation, --platform, and --python-version when using this " - "option." - ), -) - - -def add_target_python_options(cmd_opts: OptionGroup) -> None: - cmd_opts.add_option(platforms()) - cmd_opts.add_option(python_version()) - cmd_opts.add_option(implementation()) - cmd_opts.add_option(abis()) - - -def make_target_python(options: Values) -> TargetPython: - target_python = TargetPython( - platforms=options.platforms, - py_version_info=options.python_version, - abis=options.abis, - implementation=options.implementation, - ) - - return target_python - - -def prefer_binary() -> Option: - return Option( - "--prefer-binary", - dest="prefer_binary", - action="store_true", - default=False, - help=( - "Prefer binary packages over source packages, even if the " - "source packages are newer." - ), - ) - - -cache_dir: Callable[..., Option] = partial( - PipOption, - "--cache-dir", - dest="cache_dir", - default=USER_CACHE_DIR, - metavar="dir", - type="path", - help="Store the cache data in .", -) - - -def _handle_no_cache_dir( - option: Option, opt: str, value: str, parser: OptionParser -) -> None: - """ - Process a value provided for the --no-cache-dir option. - - This is an optparse.Option callback for the --no-cache-dir option. - """ - # The value argument will be None if --no-cache-dir is passed via the - # command-line, since the option doesn't accept arguments. However, - # the value can be non-None if the option is triggered e.g. by an - # environment variable, like PIP_NO_CACHE_DIR=true. - if value is not None: - # Then parse the string value to get argument error-checking. - try: - strtobool(value) - except ValueError as exc: - raise_option_error(parser, option=option, msg=str(exc)) - - # Originally, setting PIP_NO_CACHE_DIR to a value that strtobool() - # converted to 0 (like "false" or "no") caused cache_dir to be disabled - # rather than enabled (logic would say the latter). Thus, we disable - # the cache directory not just on values that parse to True, but (for - # backwards compatibility reasons) also on values that parse to False. - # In other words, always set it to False if the option is provided in - # some (valid) form. - parser.values.cache_dir = False - - -no_cache: Callable[..., Option] = partial( - Option, - "--no-cache-dir", - dest="cache_dir", - action="callback", - callback=_handle_no_cache_dir, - help="Disable the cache.", -) - -no_deps: Callable[..., Option] = partial( - Option, - "--no-deps", - "--no-dependencies", - dest="ignore_dependencies", - action="store_true", - default=False, - help="Don't install package dependencies.", -) - -ignore_requires_python: Callable[..., Option] = partial( - Option, - "--ignore-requires-python", - dest="ignore_requires_python", - action="store_true", - help="Ignore the Requires-Python information.", -) - -no_build_isolation: Callable[..., Option] = partial( - Option, - "--no-build-isolation", - dest="build_isolation", - action="store_false", - default=True, - help="Disable isolation when building a modern source distribution. " - "Build dependencies specified by PEP 518 must be already installed " - "if this option is used.", -) - -check_build_deps: Callable[..., Option] = partial( - Option, - "--check-build-dependencies", - dest="check_build_deps", - action="store_true", - default=False, - help="Check the build dependencies when PEP517 is used.", -) - - -def _handle_no_use_pep517( - option: Option, opt: str, value: str, parser: OptionParser -) -> None: - """ - Process a value provided for the --no-use-pep517 option. - - This is an optparse.Option callback for the no_use_pep517 option. - """ - # Since --no-use-pep517 doesn't accept arguments, the value argument - # will be None if --no-use-pep517 is passed via the command-line. - # However, the value can be non-None if the option is triggered e.g. - # by an environment variable, for example "PIP_NO_USE_PEP517=true". - if value is not None: - msg = """A value was passed for --no-use-pep517, - probably using either the PIP_NO_USE_PEP517 environment variable - or the "no-use-pep517" config file option. Use an appropriate value - of the PIP_USE_PEP517 environment variable or the "use-pep517" - config file option instead. - """ - raise_option_error(parser, option=option, msg=msg) - - # If user doesn't wish to use pep517, we check if setuptools and wheel are installed - # and raise error if it is not. - packages = ("setuptools", "wheel") - if not all(importlib.util.find_spec(package) for package in packages): - msg = ( - f"It is not possible to use --no-use-pep517 " - f"without {' and '.join(packages)} installed." - ) - raise_option_error(parser, option=option, msg=msg) - - # Otherwise, --no-use-pep517 was passed via the command-line. - parser.values.use_pep517 = False - - -use_pep517: Any = partial( - Option, - "--use-pep517", - dest="use_pep517", - action="store_true", - default=None, - help="Use PEP 517 for building source distributions " - "(use --no-use-pep517 to force legacy behaviour).", -) - -no_use_pep517: Any = partial( - Option, - "--no-use-pep517", - dest="use_pep517", - action="callback", - callback=_handle_no_use_pep517, - default=None, - help=SUPPRESS_HELP, -) - - -def _handle_config_settings( - option: Option, opt_str: str, value: str, parser: OptionParser -) -> None: - key, sep, val = value.partition("=") - if sep != "=": - parser.error(f"Arguments to {opt_str} must be of the form KEY=VAL") - dest = getattr(parser.values, option.dest) - if dest is None: - dest = {} - setattr(parser.values, option.dest, dest) - if key in dest: - if isinstance(dest[key], list): - dest[key].append(val) - else: - dest[key] = [dest[key], val] - else: - dest[key] = val - - -config_settings: Callable[..., Option] = partial( - Option, - "-C", - "--config-settings", - dest="config_settings", - type=str, - action="callback", - callback=_handle_config_settings, - metavar="settings", - help="Configuration settings to be passed to the PEP 517 build backend. " - "Settings take the form KEY=VALUE. Use multiple --config-settings options " - "to pass multiple keys to the backend.", -) - -build_options: Callable[..., Option] = partial( - Option, - "--build-option", - dest="build_options", - metavar="options", - action="append", - help="Extra arguments to be supplied to 'setup.py bdist_wheel'.", -) - -global_options: Callable[..., Option] = partial( - Option, - "--global-option", - dest="global_options", - action="append", - metavar="options", - help="Extra global options to be supplied to the setup.py " - "call before the install or bdist_wheel command.", -) - -no_clean: Callable[..., Option] = partial( - Option, - "--no-clean", - action="store_true", - default=False, - help="Don't clean up build directories.", -) - -pre: Callable[..., Option] = partial( - Option, - "--pre", - action="store_true", - default=False, - help="Include pre-release and development versions. By default, " - "pip only finds stable versions.", -) - -disable_pip_version_check: Callable[..., Option] = partial( - Option, - "--disable-pip-version-check", - dest="disable_pip_version_check", - action="store_true", - default=False, - help="Don't periodically check PyPI to determine whether a new version " - "of pip is available for download. Implied with --no-index.", -) - -root_user_action: Callable[..., Option] = partial( - Option, - "--root-user-action", - dest="root_user_action", - default="warn", - choices=["warn", "ignore"], - help="Action if pip is run as a root user [warn, ignore] (default: warn)", -) - - -def _handle_merge_hash( - option: Option, opt_str: str, value: str, parser: OptionParser -) -> None: - """Given a value spelled "algo:digest", append the digest to a list - pointed to in a dict by the algo name.""" - if not parser.values.hashes: - parser.values.hashes = {} - try: - algo, digest = value.split(":", 1) - except ValueError: - parser.error( - f"Arguments to {opt_str} must be a hash name " - "followed by a value, like --hash=sha256:" - "abcde..." - ) - if algo not in STRONG_HASHES: - parser.error( - "Allowed hash algorithms for {} are {}.".format( - opt_str, ", ".join(STRONG_HASHES) - ) - ) - parser.values.hashes.setdefault(algo, []).append(digest) - - -hash: Callable[..., Option] = partial( - Option, - "--hash", - # Hash values eventually end up in InstallRequirement.hashes due to - # __dict__ copying in process_line(). - dest="hashes", - action="callback", - callback=_handle_merge_hash, - type="string", - help="Verify that the package's archive matches this " - "hash before installing. Example: --hash=sha256:abcdef...", -) - - -require_hashes: Callable[..., Option] = partial( - Option, - "--require-hashes", - dest="require_hashes", - action="store_true", - default=False, - help="Require a hash to check each requirement against, for " - "repeatable installs. This option is implied when any package in a " - "requirements file has a --hash option.", -) - - -list_path: Callable[..., Option] = partial( - PipOption, - "--path", - dest="path", - type="path", - action="append", - help="Restrict to the specified installation path for listing " - "packages (can be used multiple times).", -) - - -def check_list_path_option(options: Values) -> None: - if options.path and (options.user or options.local): - raise CommandError("Cannot combine '--path' with '--user' or '--local'") - - -list_exclude: Callable[..., Option] = partial( - PipOption, - "--exclude", - dest="excludes", - action="append", - metavar="package", - type="package_name", - help="Exclude specified package from the output", -) - - -no_python_version_warning: Callable[..., Option] = partial( - Option, - "--no-python-version-warning", - dest="no_python_version_warning", - action="store_true", - default=False, - help="Silence deprecation warnings for upcoming unsupported Pythons.", -) - - -# Features that are now always on. A warning is printed if they are used. -ALWAYS_ENABLED_FEATURES = [ - "truststore", # always on since 24.2 - "no-binary-enable-wheel-cache", # always on since 23.1 -] - -use_new_feature: Callable[..., Option] = partial( - Option, - "--use-feature", - dest="features_enabled", - metavar="feature", - action="append", - default=[], - choices=[ - "fast-deps", - ] - + ALWAYS_ENABLED_FEATURES, - help="Enable new functionality, that may be backward incompatible.", -) - -use_deprecated_feature: Callable[..., Option] = partial( - Option, - "--use-deprecated", - dest="deprecated_features_enabled", - metavar="feature", - action="append", - default=[], - choices=[ - "legacy-resolver", - "legacy-certs", - ], - help=("Enable deprecated functionality, that will be removed in the future."), -) - - -########## -# groups # -########## - -general_group: Dict[str, Any] = { - "name": "General Options", - "options": [ - help_, - debug_mode, - isolated_mode, - require_virtualenv, - python, - verbose, - version, - quiet, - log, - no_input, - keyring_provider, - proxy, - retries, - timeout, - exists_action, - trusted_host, - cert, - client_cert, - cache_dir, - no_cache, - disable_pip_version_check, - no_color, - no_python_version_warning, - use_new_feature, - use_deprecated_feature, - ], -} - -index_group: Dict[str, Any] = { - "name": "Package Index Options", - "options": [ - index_url, - extra_index_url, - no_index, - find_links, - ], -} diff --git a/.env/Lib/site-packages/pip/_internal/cli/command_context.py b/.env/Lib/site-packages/pip/_internal/cli/command_context.py deleted file mode 100644 index 139995a..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/command_context.py +++ /dev/null @@ -1,27 +0,0 @@ -from contextlib import ExitStack, contextmanager -from typing import ContextManager, Generator, TypeVar - -_T = TypeVar("_T", covariant=True) - - -class CommandContextMixIn: - def __init__(self) -> None: - super().__init__() - self._in_main_context = False - self._main_context = ExitStack() - - @contextmanager - def main_context(self) -> Generator[None, None, None]: - assert not self._in_main_context - - self._in_main_context = True - try: - with self._main_context: - yield - finally: - self._in_main_context = False - - def enter_context(self, context_provider: ContextManager[_T]) -> _T: - assert self._in_main_context - - return self._main_context.enter_context(context_provider) diff --git a/.env/Lib/site-packages/pip/_internal/cli/index_command.py b/.env/Lib/site-packages/pip/_internal/cli/index_command.py deleted file mode 100644 index 226f8da..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/index_command.py +++ /dev/null @@ -1,170 +0,0 @@ -""" -Contains command classes which may interact with an index / the network. - -Unlike its sister module, req_command, this module still uses lazy imports -so commands which don't always hit the network (e.g. list w/o --outdated or ---uptodate) don't need waste time importing PipSession and friends. -""" - -import logging -import os -import sys -from optparse import Values -from typing import TYPE_CHECKING, List, Optional - -from pip._vendor import certifi - -from pip._internal.cli.base_command import Command -from pip._internal.cli.command_context import CommandContextMixIn - -if TYPE_CHECKING: - from ssl import SSLContext - - from pip._internal.network.session import PipSession - -logger = logging.getLogger(__name__) - - -def _create_truststore_ssl_context() -> Optional["SSLContext"]: - if sys.version_info < (3, 10): - logger.debug("Disabling truststore because Python version isn't 3.10+") - return None - - try: - import ssl - except ImportError: - logger.warning("Disabling truststore since ssl support is missing") - return None - - try: - from pip._vendor import truststore - except ImportError: - logger.warning("Disabling truststore because platform isn't supported") - return None - - ctx = truststore.SSLContext(ssl.PROTOCOL_TLS_CLIENT) - ctx.load_verify_locations(certifi.where()) - return ctx - - -class SessionCommandMixin(CommandContextMixIn): - """ - A class mixin for command classes needing _build_session(). - """ - - def __init__(self) -> None: - super().__init__() - self._session: Optional["PipSession"] = None - - @classmethod - def _get_index_urls(cls, options: Values) -> Optional[List[str]]: - """Return a list of index urls from user-provided options.""" - index_urls = [] - if not getattr(options, "no_index", False): - url = getattr(options, "index_url", None) - if url: - index_urls.append(url) - urls = getattr(options, "extra_index_urls", None) - if urls: - index_urls.extend(urls) - # Return None rather than an empty list - return index_urls or None - - def get_default_session(self, options: Values) -> "PipSession": - """Get a default-managed session.""" - if self._session is None: - self._session = self.enter_context(self._build_session(options)) - # there's no type annotation on requests.Session, so it's - # automatically ContextManager[Any] and self._session becomes Any, - # then https://github.com/python/mypy/issues/7696 kicks in - assert self._session is not None - return self._session - - def _build_session( - self, - options: Values, - retries: Optional[int] = None, - timeout: Optional[int] = None, - ) -> "PipSession": - from pip._internal.network.session import PipSession - - cache_dir = options.cache_dir - assert not cache_dir or os.path.isabs(cache_dir) - - if "legacy-certs" not in options.deprecated_features_enabled: - ssl_context = _create_truststore_ssl_context() - else: - ssl_context = None - - session = PipSession( - cache=os.path.join(cache_dir, "http-v2") if cache_dir else None, - retries=retries if retries is not None else options.retries, - trusted_hosts=options.trusted_hosts, - index_urls=self._get_index_urls(options), - ssl_context=ssl_context, - ) - - # Handle custom ca-bundles from the user - if options.cert: - session.verify = options.cert - - # Handle SSL client certificate - if options.client_cert: - session.cert = options.client_cert - - # Handle timeouts - if options.timeout or timeout: - session.timeout = timeout if timeout is not None else options.timeout - - # Handle configured proxies - if options.proxy: - session.proxies = { - "http": options.proxy, - "https": options.proxy, - } - session.trust_env = False - - # Determine if we can prompt the user for authentication or not - session.auth.prompting = not options.no_input - session.auth.keyring_provider = options.keyring_provider - - return session - - -def _pip_self_version_check(session: "PipSession", options: Values) -> None: - from pip._internal.self_outdated_check import pip_self_version_check as check - - check(session, options) - - -class IndexGroupCommand(Command, SessionCommandMixin): - """ - Abstract base class for commands with the index_group options. - - This also corresponds to the commands that permit the pip version check. - """ - - def handle_pip_version_check(self, options: Values) -> None: - """ - Do the pip version check if not disabled. - - This overrides the default behavior of not doing the check. - """ - # Make sure the index_group options are present. - assert hasattr(options, "no_index") - - if options.disable_pip_version_check or options.no_index: - return - - try: - # Otherwise, check if we're using the latest version of pip available. - session = self._build_session( - options, - retries=0, - timeout=min(5, options.timeout), - ) - with session: - _pip_self_version_check(session, options) - except Exception: - logger.warning("There was an error checking the latest version of pip.") - logger.debug("See below for error", exc_info=True) diff --git a/.env/Lib/site-packages/pip/_internal/cli/main.py b/.env/Lib/site-packages/pip/_internal/cli/main.py deleted file mode 100644 index 563ac79..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/main.py +++ /dev/null @@ -1,80 +0,0 @@ -"""Primary application entrypoint. -""" - -import locale -import logging -import os -import sys -import warnings -from typing import List, Optional - -from pip._internal.cli.autocompletion import autocomplete -from pip._internal.cli.main_parser import parse_command -from pip._internal.commands import create_command -from pip._internal.exceptions import PipError -from pip._internal.utils import deprecation - -logger = logging.getLogger(__name__) - - -# Do not import and use main() directly! Using it directly is actively -# discouraged by pip's maintainers. The name, location and behavior of -# this function is subject to change, so calling it directly is not -# portable across different pip versions. - -# In addition, running pip in-process is unsupported and unsafe. This is -# elaborated in detail at -# https://pip.pypa.io/en/stable/user_guide/#using-pip-from-your-program. -# That document also provides suggestions that should work for nearly -# all users that are considering importing and using main() directly. - -# However, we know that certain users will still want to invoke pip -# in-process. If you understand and accept the implications of using pip -# in an unsupported manner, the best approach is to use runpy to avoid -# depending on the exact location of this entry point. - -# The following example shows how to use runpy to invoke pip in that -# case: -# -# sys.argv = ["pip", your, args, here] -# runpy.run_module("pip", run_name="__main__") -# -# Note that this will exit the process after running, unlike a direct -# call to main. As it is not safe to do any processing after calling -# main, this should not be an issue in practice. - - -def main(args: Optional[List[str]] = None) -> int: - if args is None: - args = sys.argv[1:] - - # Suppress the pkg_resources deprecation warning - # Note - we use a module of .*pkg_resources to cover - # the normal case (pip._vendor.pkg_resources) and the - # devendored case (a bare pkg_resources) - warnings.filterwarnings( - action="ignore", category=DeprecationWarning, module=".*pkg_resources" - ) - - # Configure our deprecation warnings to be sent through loggers - deprecation.install_warning_logger() - - autocomplete() - - try: - cmd_name, cmd_args = parse_command(args) - except PipError as exc: - sys.stderr.write(f"ERROR: {exc}") - sys.stderr.write(os.linesep) - sys.exit(1) - - # Needed for locale.getpreferredencoding(False) to work - # in pip._internal.utils.encoding.auto_decode - try: - locale.setlocale(locale.LC_ALL, "") - except locale.Error as e: - # setlocale can apparently crash if locale are uninitialized - logger.debug("Ignoring error %s when setting locale", e) - command = create_command(cmd_name, isolated=("--isolated" in cmd_args)) - - return command.main(cmd_args) diff --git a/.env/Lib/site-packages/pip/_internal/cli/main_parser.py b/.env/Lib/site-packages/pip/_internal/cli/main_parser.py deleted file mode 100644 index 5ade356..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/main_parser.py +++ /dev/null @@ -1,134 +0,0 @@ -"""A single place for constructing and exposing the main parser -""" - -import os -import subprocess -import sys -from typing import List, Optional, Tuple - -from pip._internal.build_env import get_runnable_pip -from pip._internal.cli import cmdoptions -from pip._internal.cli.parser import ConfigOptionParser, UpdatingDefaultsHelpFormatter -from pip._internal.commands import commands_dict, get_similar_commands -from pip._internal.exceptions import CommandError -from pip._internal.utils.misc import get_pip_version, get_prog - -__all__ = ["create_main_parser", "parse_command"] - - -def create_main_parser() -> ConfigOptionParser: - """Creates and returns the main parser for pip's CLI""" - - parser = ConfigOptionParser( - usage="\n%prog [options]", - add_help_option=False, - formatter=UpdatingDefaultsHelpFormatter(), - name="global", - prog=get_prog(), - ) - parser.disable_interspersed_args() - - parser.version = get_pip_version() - - # add the general options - gen_opts = cmdoptions.make_option_group(cmdoptions.general_group, parser) - parser.add_option_group(gen_opts) - - # so the help formatter knows - parser.main = True # type: ignore - - # create command listing for description - description = [""] + [ - f"{name:27} {command_info.summary}" - for name, command_info in commands_dict.items() - ] - parser.description = "\n".join(description) - - return parser - - -def identify_python_interpreter(python: str) -> Optional[str]: - # If the named file exists, use it. - # If it's a directory, assume it's a virtual environment and - # look for the environment's Python executable. - if os.path.exists(python): - if os.path.isdir(python): - # bin/python for Unix, Scripts/python.exe for Windows - # Try both in case of odd cases like cygwin. - for exe in ("bin/python", "Scripts/python.exe"): - py = os.path.join(python, exe) - if os.path.exists(py): - return py - else: - return python - - # Could not find the interpreter specified - return None - - -def parse_command(args: List[str]) -> Tuple[str, List[str]]: - parser = create_main_parser() - - # Note: parser calls disable_interspersed_args(), so the result of this - # call is to split the initial args into the general options before the - # subcommand and everything else. - # For example: - # args: ['--timeout=5', 'install', '--user', 'INITools'] - # general_options: ['--timeout==5'] - # args_else: ['install', '--user', 'INITools'] - general_options, args_else = parser.parse_args(args) - - # --python - if general_options.python and "_PIP_RUNNING_IN_SUBPROCESS" not in os.environ: - # Re-invoke pip using the specified Python interpreter - interpreter = identify_python_interpreter(general_options.python) - if interpreter is None: - raise CommandError( - f"Could not locate Python interpreter {general_options.python}" - ) - - pip_cmd = [ - interpreter, - get_runnable_pip(), - ] - pip_cmd.extend(args) - - # Set a flag so the child doesn't re-invoke itself, causing - # an infinite loop. - os.environ["_PIP_RUNNING_IN_SUBPROCESS"] = "1" - returncode = 0 - try: - proc = subprocess.run(pip_cmd) - returncode = proc.returncode - except (subprocess.SubprocessError, OSError) as exc: - raise CommandError(f"Failed to run pip under {interpreter}: {exc}") - sys.exit(returncode) - - # --version - if general_options.version: - sys.stdout.write(parser.version) - sys.stdout.write(os.linesep) - sys.exit() - - # pip || pip help -> print_help() - if not args_else or (args_else[0] == "help" and len(args_else) == 1): - parser.print_help() - sys.exit() - - # the subcommand name - cmd_name = args_else[0] - - if cmd_name not in commands_dict: - guess = get_similar_commands(cmd_name) - - msg = [f'unknown command "{cmd_name}"'] - if guess: - msg.append(f'maybe you meant "{guess}"') - - raise CommandError(" - ".join(msg)) - - # all the args without the subcommand - cmd_args = args[:] - cmd_args.remove(cmd_name) - - return cmd_name, cmd_args diff --git a/.env/Lib/site-packages/pip/_internal/cli/parser.py b/.env/Lib/site-packages/pip/_internal/cli/parser.py deleted file mode 100644 index b7d7c1f..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/parser.py +++ /dev/null @@ -1,294 +0,0 @@ -"""Base option parser setup""" - -import logging -import optparse -import shutil -import sys -import textwrap -from contextlib import suppress -from typing import Any, Dict, Generator, List, Optional, Tuple - -from pip._internal.cli.status_codes import UNKNOWN_ERROR -from pip._internal.configuration import Configuration, ConfigurationError -from pip._internal.utils.misc import redact_auth_from_url, strtobool - -logger = logging.getLogger(__name__) - - -class PrettyHelpFormatter(optparse.IndentedHelpFormatter): - """A prettier/less verbose help formatter for optparse.""" - - def __init__(self, *args: Any, **kwargs: Any) -> None: - # help position must be aligned with __init__.parseopts.description - kwargs["max_help_position"] = 30 - kwargs["indent_increment"] = 1 - kwargs["width"] = shutil.get_terminal_size()[0] - 2 - super().__init__(*args, **kwargs) - - def format_option_strings(self, option: optparse.Option) -> str: - return self._format_option_strings(option) - - def _format_option_strings( - self, option: optparse.Option, mvarfmt: str = " <{}>", optsep: str = ", " - ) -> str: - """ - Return a comma-separated list of option strings and metavars. - - :param option: tuple of (short opt, long opt), e.g: ('-f', '--format') - :param mvarfmt: metavar format string - :param optsep: separator - """ - opts = [] - - if option._short_opts: - opts.append(option._short_opts[0]) - if option._long_opts: - opts.append(option._long_opts[0]) - if len(opts) > 1: - opts.insert(1, optsep) - - if option.takes_value(): - assert option.dest is not None - metavar = option.metavar or option.dest.lower() - opts.append(mvarfmt.format(metavar.lower())) - - return "".join(opts) - - def format_heading(self, heading: str) -> str: - if heading == "Options": - return "" - return heading + ":\n" - - def format_usage(self, usage: str) -> str: - """ - Ensure there is only one newline between usage and the first heading - if there is no description. - """ - msg = "\nUsage: {}\n".format(self.indent_lines(textwrap.dedent(usage), " ")) - return msg - - def format_description(self, description: Optional[str]) -> str: - # leave full control over description to us - if description: - if hasattr(self.parser, "main"): - label = "Commands" - else: - label = "Description" - # some doc strings have initial newlines, some don't - description = description.lstrip("\n") - # some doc strings have final newlines and spaces, some don't - description = description.rstrip() - # dedent, then reindent - description = self.indent_lines(textwrap.dedent(description), " ") - description = f"{label}:\n{description}\n" - return description - else: - return "" - - def format_epilog(self, epilog: Optional[str]) -> str: - # leave full control over epilog to us - if epilog: - return epilog - else: - return "" - - def indent_lines(self, text: str, indent: str) -> str: - new_lines = [indent + line for line in text.split("\n")] - return "\n".join(new_lines) - - -class UpdatingDefaultsHelpFormatter(PrettyHelpFormatter): - """Custom help formatter for use in ConfigOptionParser. - - This is updates the defaults before expanding them, allowing - them to show up correctly in the help listing. - - Also redact auth from url type options - """ - - def expand_default(self, option: optparse.Option) -> str: - default_values = None - if self.parser is not None: - assert isinstance(self.parser, ConfigOptionParser) - self.parser._update_defaults(self.parser.defaults) - assert option.dest is not None - default_values = self.parser.defaults.get(option.dest) - help_text = super().expand_default(option) - - if default_values and option.metavar == "URL": - if isinstance(default_values, str): - default_values = [default_values] - - # If its not a list, we should abort and just return the help text - if not isinstance(default_values, list): - default_values = [] - - for val in default_values: - help_text = help_text.replace(val, redact_auth_from_url(val)) - - return help_text - - -class CustomOptionParser(optparse.OptionParser): - def insert_option_group( - self, idx: int, *args: Any, **kwargs: Any - ) -> optparse.OptionGroup: - """Insert an OptionGroup at a given position.""" - group = self.add_option_group(*args, **kwargs) - - self.option_groups.pop() - self.option_groups.insert(idx, group) - - return group - - @property - def option_list_all(self) -> List[optparse.Option]: - """Get a list of all options, including those in option groups.""" - res = self.option_list[:] - for i in self.option_groups: - res.extend(i.option_list) - - return res - - -class ConfigOptionParser(CustomOptionParser): - """Custom option parser which updates its defaults by checking the - configuration files and environmental variables""" - - def __init__( - self, - *args: Any, - name: str, - isolated: bool = False, - **kwargs: Any, - ) -> None: - self.name = name - self.config = Configuration(isolated) - - assert self.name - super().__init__(*args, **kwargs) - - def check_default(self, option: optparse.Option, key: str, val: Any) -> Any: - try: - return option.check_value(key, val) - except optparse.OptionValueError as exc: - print(f"An error occurred during configuration: {exc}") - sys.exit(3) - - def _get_ordered_configuration_items( - self, - ) -> Generator[Tuple[str, Any], None, None]: - # Configuration gives keys in an unordered manner. Order them. - override_order = ["global", self.name, ":env:"] - - # Pool the options into different groups - section_items: Dict[str, List[Tuple[str, Any]]] = { - name: [] for name in override_order - } - for section_key, val in self.config.items(): - # ignore empty values - if not val: - logger.debug( - "Ignoring configuration key '%s' as it's value is empty.", - section_key, - ) - continue - - section, key = section_key.split(".", 1) - if section in override_order: - section_items[section].append((key, val)) - - # Yield each group in their override order - for section in override_order: - for key, val in section_items[section]: - yield key, val - - def _update_defaults(self, defaults: Dict[str, Any]) -> Dict[str, Any]: - """Updates the given defaults with values from the config files and - the environ. Does a little special handling for certain types of - options (lists).""" - - # Accumulate complex default state. - self.values = optparse.Values(self.defaults) - late_eval = set() - # Then set the options with those values - for key, val in self._get_ordered_configuration_items(): - # '--' because configuration supports only long names - option = self.get_option("--" + key) - - # Ignore options not present in this parser. E.g. non-globals put - # in [global] by users that want them to apply to all applicable - # commands. - if option is None: - continue - - assert option.dest is not None - - if option.action in ("store_true", "store_false"): - try: - val = strtobool(val) - except ValueError: - self.error( - f"{val} is not a valid value for {key} option, " - "please specify a boolean value like yes/no, " - "true/false or 1/0 instead." - ) - elif option.action == "count": - with suppress(ValueError): - val = strtobool(val) - with suppress(ValueError): - val = int(val) - if not isinstance(val, int) or val < 0: - self.error( - f"{val} is not a valid value for {key} option, " - "please instead specify either a non-negative integer " - "or a boolean value like yes/no or false/true " - "which is equivalent to 1/0." - ) - elif option.action == "append": - val = val.split() - val = [self.check_default(option, key, v) for v in val] - elif option.action == "callback": - assert option.callback is not None - late_eval.add(option.dest) - opt_str = option.get_opt_string() - val = option.convert_value(opt_str, val) - # From take_action - args = option.callback_args or () - kwargs = option.callback_kwargs or {} - option.callback(option, opt_str, val, self, *args, **kwargs) - else: - val = self.check_default(option, key, val) - - defaults[option.dest] = val - - for key in late_eval: - defaults[key] = getattr(self.values, key) - self.values = None - return defaults - - def get_default_values(self) -> optparse.Values: - """Overriding to make updating the defaults after instantiation of - the option parser possible, _update_defaults() does the dirty work.""" - if not self.process_default_values: - # Old, pre-Optik 1.5 behaviour. - return optparse.Values(self.defaults) - - # Load the configuration, or error out in case of an error - try: - self.config.load() - except ConfigurationError as err: - self.exit(UNKNOWN_ERROR, str(err)) - - defaults = self._update_defaults(self.defaults.copy()) # ours - for option in self._get_all_options(): - assert option.dest is not None - default = defaults.get(option.dest) - if isinstance(default, str): - opt_str = option.get_opt_string() - defaults[option.dest] = option.check_value(opt_str, default) - return optparse.Values(defaults) - - def error(self, msg: str) -> None: - self.print_usage(sys.stderr) - self.exit(UNKNOWN_ERROR, f"{msg}\n") diff --git a/.env/Lib/site-packages/pip/_internal/cli/progress_bars.py b/.env/Lib/site-packages/pip/_internal/cli/progress_bars.py deleted file mode 100644 index 883359c..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/progress_bars.py +++ /dev/null @@ -1,94 +0,0 @@ -import functools -import sys -from typing import Callable, Generator, Iterable, Iterator, Optional, Tuple - -from pip._vendor.rich.progress import ( - BarColumn, - DownloadColumn, - FileSizeColumn, - Progress, - ProgressColumn, - SpinnerColumn, - TextColumn, - TimeElapsedColumn, - TimeRemainingColumn, - TransferSpeedColumn, -) - -from pip._internal.cli.spinners import RateLimiter -from pip._internal.utils.logging import get_indentation - -DownloadProgressRenderer = Callable[[Iterable[bytes]], Iterator[bytes]] - - -def _rich_progress_bar( - iterable: Iterable[bytes], - *, - bar_type: str, - size: int, -) -> Generator[bytes, None, None]: - assert bar_type == "on", "This should only be used in the default mode." - - if not size: - total = float("inf") - columns: Tuple[ProgressColumn, ...] = ( - TextColumn("[progress.description]{task.description}"), - SpinnerColumn("line", speed=1.5), - FileSizeColumn(), - TransferSpeedColumn(), - TimeElapsedColumn(), - ) - else: - total = size - columns = ( - TextColumn("[progress.description]{task.description}"), - BarColumn(), - DownloadColumn(), - TransferSpeedColumn(), - TextColumn("eta"), - TimeRemainingColumn(), - ) - - progress = Progress(*columns, refresh_per_second=5) - task_id = progress.add_task(" " * (get_indentation() + 2), total=total) - with progress: - for chunk in iterable: - yield chunk - progress.update(task_id, advance=len(chunk)) - - -def _raw_progress_bar( - iterable: Iterable[bytes], - *, - size: Optional[int], -) -> Generator[bytes, None, None]: - def write_progress(current: int, total: int) -> None: - sys.stdout.write("Progress %d of %d\n" % (current, total)) - sys.stdout.flush() - - current = 0 - total = size or 0 - rate_limiter = RateLimiter(0.25) - - write_progress(current, total) - for chunk in iterable: - current += len(chunk) - if rate_limiter.ready() or current == total: - write_progress(current, total) - rate_limiter.reset() - yield chunk - - -def get_download_progress_renderer( - *, bar_type: str, size: Optional[int] = None -) -> DownloadProgressRenderer: - """Get an object that can be used to render the download progress. - - Returns a callable, that takes an iterable to "wrap". - """ - if bar_type == "on": - return functools.partial(_rich_progress_bar, bar_type=bar_type, size=size) - elif bar_type == "raw": - return functools.partial(_raw_progress_bar, size=size) - else: - return iter # no-op, when passed an iterator diff --git a/.env/Lib/site-packages/pip/_internal/cli/req_command.py b/.env/Lib/site-packages/pip/_internal/cli/req_command.py deleted file mode 100644 index 92900f9..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/req_command.py +++ /dev/null @@ -1,329 +0,0 @@ -"""Contains the RequirementCommand base class. - -This class is in a separate module so the commands that do not always -need PackageFinder capability don't unnecessarily import the -PackageFinder machinery and all its vendored dependencies, etc. -""" - -import logging -from functools import partial -from optparse import Values -from typing import Any, List, Optional, Tuple - -from pip._internal.cache import WheelCache -from pip._internal.cli import cmdoptions -from pip._internal.cli.index_command import IndexGroupCommand -from pip._internal.cli.index_command import SessionCommandMixin as SessionCommandMixin -from pip._internal.exceptions import CommandError, PreviousBuildDirError -from pip._internal.index.collector import LinkCollector -from pip._internal.index.package_finder import PackageFinder -from pip._internal.models.selection_prefs import SelectionPreferences -from pip._internal.models.target_python import TargetPython -from pip._internal.network.session import PipSession -from pip._internal.operations.build.build_tracker import BuildTracker -from pip._internal.operations.prepare import RequirementPreparer -from pip._internal.req.constructors import ( - install_req_from_editable, - install_req_from_line, - install_req_from_parsed_requirement, - install_req_from_req_string, -) -from pip._internal.req.req_file import parse_requirements -from pip._internal.req.req_install import InstallRequirement -from pip._internal.resolution.base import BaseResolver -from pip._internal.utils.temp_dir import ( - TempDirectory, - TempDirectoryTypeRegistry, - tempdir_kinds, -) - -logger = logging.getLogger(__name__) - - -KEEPABLE_TEMPDIR_TYPES = [ - tempdir_kinds.BUILD_ENV, - tempdir_kinds.EPHEM_WHEEL_CACHE, - tempdir_kinds.REQ_BUILD, -] - - -def with_cleanup(func: Any) -> Any: - """Decorator for common logic related to managing temporary - directories. - """ - - def configure_tempdir_registry(registry: TempDirectoryTypeRegistry) -> None: - for t in KEEPABLE_TEMPDIR_TYPES: - registry.set_delete(t, False) - - def wrapper( - self: RequirementCommand, options: Values, args: List[Any] - ) -> Optional[int]: - assert self.tempdir_registry is not None - if options.no_clean: - configure_tempdir_registry(self.tempdir_registry) - - try: - return func(self, options, args) - except PreviousBuildDirError: - # This kind of conflict can occur when the user passes an explicit - # build directory with a pre-existing folder. In that case we do - # not want to accidentally remove it. - configure_tempdir_registry(self.tempdir_registry) - raise - - return wrapper - - -class RequirementCommand(IndexGroupCommand): - def __init__(self, *args: Any, **kw: Any) -> None: - super().__init__(*args, **kw) - - self.cmd_opts.add_option(cmdoptions.no_clean()) - - @staticmethod - def determine_resolver_variant(options: Values) -> str: - """Determines which resolver should be used, based on the given options.""" - if "legacy-resolver" in options.deprecated_features_enabled: - return "legacy" - - return "resolvelib" - - @classmethod - def make_requirement_preparer( - cls, - temp_build_dir: TempDirectory, - options: Values, - build_tracker: BuildTracker, - session: PipSession, - finder: PackageFinder, - use_user_site: bool, - download_dir: Optional[str] = None, - verbosity: int = 0, - ) -> RequirementPreparer: - """ - Create a RequirementPreparer instance for the given parameters. - """ - temp_build_dir_path = temp_build_dir.path - assert temp_build_dir_path is not None - legacy_resolver = False - - resolver_variant = cls.determine_resolver_variant(options) - if resolver_variant == "resolvelib": - lazy_wheel = "fast-deps" in options.features_enabled - if lazy_wheel: - logger.warning( - "pip is using lazily downloaded wheels using HTTP " - "range requests to obtain dependency information. " - "This experimental feature is enabled through " - "--use-feature=fast-deps and it is not ready for " - "production." - ) - else: - legacy_resolver = True - lazy_wheel = False - if "fast-deps" in options.features_enabled: - logger.warning( - "fast-deps has no effect when used with the legacy resolver." - ) - - return RequirementPreparer( - build_dir=temp_build_dir_path, - src_dir=options.src_dir, - download_dir=download_dir, - build_isolation=options.build_isolation, - check_build_deps=options.check_build_deps, - build_tracker=build_tracker, - session=session, - progress_bar=options.progress_bar, - finder=finder, - require_hashes=options.require_hashes, - use_user_site=use_user_site, - lazy_wheel=lazy_wheel, - verbosity=verbosity, - legacy_resolver=legacy_resolver, - ) - - @classmethod - def make_resolver( - cls, - preparer: RequirementPreparer, - finder: PackageFinder, - options: Values, - wheel_cache: Optional[WheelCache] = None, - use_user_site: bool = False, - ignore_installed: bool = True, - ignore_requires_python: bool = False, - force_reinstall: bool = False, - upgrade_strategy: str = "to-satisfy-only", - use_pep517: Optional[bool] = None, - py_version_info: Optional[Tuple[int, ...]] = None, - ) -> BaseResolver: - """ - Create a Resolver instance for the given parameters. - """ - make_install_req = partial( - install_req_from_req_string, - isolated=options.isolated_mode, - use_pep517=use_pep517, - ) - resolver_variant = cls.determine_resolver_variant(options) - # The long import name and duplicated invocation is needed to convince - # Mypy into correctly typechecking. Otherwise it would complain the - # "Resolver" class being redefined. - if resolver_variant == "resolvelib": - import pip._internal.resolution.resolvelib.resolver - - return pip._internal.resolution.resolvelib.resolver.Resolver( - preparer=preparer, - finder=finder, - wheel_cache=wheel_cache, - make_install_req=make_install_req, - use_user_site=use_user_site, - ignore_dependencies=options.ignore_dependencies, - ignore_installed=ignore_installed, - ignore_requires_python=ignore_requires_python, - force_reinstall=force_reinstall, - upgrade_strategy=upgrade_strategy, - py_version_info=py_version_info, - ) - import pip._internal.resolution.legacy.resolver - - return pip._internal.resolution.legacy.resolver.Resolver( - preparer=preparer, - finder=finder, - wheel_cache=wheel_cache, - make_install_req=make_install_req, - use_user_site=use_user_site, - ignore_dependencies=options.ignore_dependencies, - ignore_installed=ignore_installed, - ignore_requires_python=ignore_requires_python, - force_reinstall=force_reinstall, - upgrade_strategy=upgrade_strategy, - py_version_info=py_version_info, - ) - - def get_requirements( - self, - args: List[str], - options: Values, - finder: PackageFinder, - session: PipSession, - ) -> List[InstallRequirement]: - """ - Parse command-line arguments into the corresponding requirements. - """ - requirements: List[InstallRequirement] = [] - for filename in options.constraints: - for parsed_req in parse_requirements( - filename, - constraint=True, - finder=finder, - options=options, - session=session, - ): - req_to_add = install_req_from_parsed_requirement( - parsed_req, - isolated=options.isolated_mode, - user_supplied=False, - ) - requirements.append(req_to_add) - - for req in args: - req_to_add = install_req_from_line( - req, - comes_from=None, - isolated=options.isolated_mode, - use_pep517=options.use_pep517, - user_supplied=True, - config_settings=getattr(options, "config_settings", None), - ) - requirements.append(req_to_add) - - for req in options.editables: - req_to_add = install_req_from_editable( - req, - user_supplied=True, - isolated=options.isolated_mode, - use_pep517=options.use_pep517, - config_settings=getattr(options, "config_settings", None), - ) - requirements.append(req_to_add) - - # NOTE: options.require_hashes may be set if --require-hashes is True - for filename in options.requirements: - for parsed_req in parse_requirements( - filename, finder=finder, options=options, session=session - ): - req_to_add = install_req_from_parsed_requirement( - parsed_req, - isolated=options.isolated_mode, - use_pep517=options.use_pep517, - user_supplied=True, - config_settings=( - parsed_req.options.get("config_settings") - if parsed_req.options - else None - ), - ) - requirements.append(req_to_add) - - # If any requirement has hash options, enable hash checking. - if any(req.has_hash_options for req in requirements): - options.require_hashes = True - - if not (args or options.editables or options.requirements): - opts = {"name": self.name} - if options.find_links: - raise CommandError( - "You must give at least one requirement to {name} " - '(maybe you meant "pip {name} {links}"?)'.format( - **dict(opts, links=" ".join(options.find_links)) - ) - ) - else: - raise CommandError( - "You must give at least one requirement to {name} " - '(see "pip help {name}")'.format(**opts) - ) - - return requirements - - @staticmethod - def trace_basic_info(finder: PackageFinder) -> None: - """ - Trace basic information about the provided objects. - """ - # Display where finder is looking for packages - search_scope = finder.search_scope - locations = search_scope.get_formatted_locations() - if locations: - logger.info(locations) - - def _build_package_finder( - self, - options: Values, - session: PipSession, - target_python: Optional[TargetPython] = None, - ignore_requires_python: Optional[bool] = None, - ) -> PackageFinder: - """ - Create a package finder appropriate to this requirement command. - - :param ignore_requires_python: Whether to ignore incompatible - "Requires-Python" values in links. Defaults to False. - """ - link_collector = LinkCollector.create(session, options=options) - selection_prefs = SelectionPreferences( - allow_yanked=True, - format_control=options.format_control, - allow_all_prereleases=options.pre, - prefer_binary=options.prefer_binary, - ignore_requires_python=ignore_requires_python, - ) - - return PackageFinder.create( - link_collector=link_collector, - selection_prefs=selection_prefs, - target_python=target_python, - ) diff --git a/.env/Lib/site-packages/pip/_internal/cli/spinners.py b/.env/Lib/site-packages/pip/_internal/cli/spinners.py deleted file mode 100644 index cf2b976..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/spinners.py +++ /dev/null @@ -1,159 +0,0 @@ -import contextlib -import itertools -import logging -import sys -import time -from typing import IO, Generator, Optional - -from pip._internal.utils.compat import WINDOWS -from pip._internal.utils.logging import get_indentation - -logger = logging.getLogger(__name__) - - -class SpinnerInterface: - def spin(self) -> None: - raise NotImplementedError() - - def finish(self, final_status: str) -> None: - raise NotImplementedError() - - -class InteractiveSpinner(SpinnerInterface): - def __init__( - self, - message: str, - file: Optional[IO[str]] = None, - spin_chars: str = "-\\|/", - # Empirically, 8 updates/second looks nice - min_update_interval_seconds: float = 0.125, - ): - self._message = message - if file is None: - file = sys.stdout - self._file = file - self._rate_limiter = RateLimiter(min_update_interval_seconds) - self._finished = False - - self._spin_cycle = itertools.cycle(spin_chars) - - self._file.write(" " * get_indentation() + self._message + " ... ") - self._width = 0 - - def _write(self, status: str) -> None: - assert not self._finished - # Erase what we wrote before by backspacing to the beginning, writing - # spaces to overwrite the old text, and then backspacing again - backup = "\b" * self._width - self._file.write(backup + " " * self._width + backup) - # Now we have a blank slate to add our status - self._file.write(status) - self._width = len(status) - self._file.flush() - self._rate_limiter.reset() - - def spin(self) -> None: - if self._finished: - return - if not self._rate_limiter.ready(): - return - self._write(next(self._spin_cycle)) - - def finish(self, final_status: str) -> None: - if self._finished: - return - self._write(final_status) - self._file.write("\n") - self._file.flush() - self._finished = True - - -# Used for dumb terminals, non-interactive installs (no tty), etc. -# We still print updates occasionally (once every 60 seconds by default) to -# act as a keep-alive for systems like Travis-CI that take lack-of-output as -# an indication that a task has frozen. -class NonInteractiveSpinner(SpinnerInterface): - def __init__(self, message: str, min_update_interval_seconds: float = 60.0) -> None: - self._message = message - self._finished = False - self._rate_limiter = RateLimiter(min_update_interval_seconds) - self._update("started") - - def _update(self, status: str) -> None: - assert not self._finished - self._rate_limiter.reset() - logger.info("%s: %s", self._message, status) - - def spin(self) -> None: - if self._finished: - return - if not self._rate_limiter.ready(): - return - self._update("still running...") - - def finish(self, final_status: str) -> None: - if self._finished: - return - self._update(f"finished with status '{final_status}'") - self._finished = True - - -class RateLimiter: - def __init__(self, min_update_interval_seconds: float) -> None: - self._min_update_interval_seconds = min_update_interval_seconds - self._last_update: float = 0 - - def ready(self) -> bool: - now = time.time() - delta = now - self._last_update - return delta >= self._min_update_interval_seconds - - def reset(self) -> None: - self._last_update = time.time() - - -@contextlib.contextmanager -def open_spinner(message: str) -> Generator[SpinnerInterface, None, None]: - # Interactive spinner goes directly to sys.stdout rather than being routed - # through the logging system, but it acts like it has level INFO, - # i.e. it's only displayed if we're at level INFO or better. - # Non-interactive spinner goes through the logging system, so it is always - # in sync with logging configuration. - if sys.stdout.isatty() and logger.getEffectiveLevel() <= logging.INFO: - spinner: SpinnerInterface = InteractiveSpinner(message) - else: - spinner = NonInteractiveSpinner(message) - try: - with hidden_cursor(sys.stdout): - yield spinner - except KeyboardInterrupt: - spinner.finish("canceled") - raise - except Exception: - spinner.finish("error") - raise - else: - spinner.finish("done") - - -HIDE_CURSOR = "\x1b[?25l" -SHOW_CURSOR = "\x1b[?25h" - - -@contextlib.contextmanager -def hidden_cursor(file: IO[str]) -> Generator[None, None, None]: - # The Windows terminal does not support the hide/show cursor ANSI codes, - # even via colorama. So don't even try. - if WINDOWS: - yield - # We don't want to clutter the output with control characters if we're - # writing to a file, or if the user is running with --quiet. - # See https://github.com/pypa/pip/issues/3418 - elif not file.isatty() or logger.getEffectiveLevel() > logging.INFO: - yield - else: - file.write(HIDE_CURSOR) - try: - yield - finally: - file.write(SHOW_CURSOR) diff --git a/.env/Lib/site-packages/pip/_internal/cli/status_codes.py b/.env/Lib/site-packages/pip/_internal/cli/status_codes.py deleted file mode 100644 index 5e29502..0000000 --- a/.env/Lib/site-packages/pip/_internal/cli/status_codes.py +++ /dev/null @@ -1,6 +0,0 @@ -SUCCESS = 0 -ERROR = 1 -UNKNOWN_ERROR = 2 -VIRTUALENV_NOT_FOUND = 3 -PREVIOUS_BUILD_DIR_ERROR = 4 -NO_MATCHES_FOUND = 23 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__init__.py b/.env/Lib/site-packages/pip/_internal/commands/__init__.py deleted file mode 100644 index 858a410..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/__init__.py +++ /dev/null @@ -1,132 +0,0 @@ -""" -Package containing all pip commands -""" - -import importlib -from collections import namedtuple -from typing import Any, Dict, Optional - -from pip._internal.cli.base_command import Command - -CommandInfo = namedtuple("CommandInfo", "module_path, class_name, summary") - -# This dictionary does a bunch of heavy lifting for help output: -# - Enables avoiding additional (costly) imports for presenting `--help`. -# - The ordering matters for help display. -# -# Even though the module path starts with the same "pip._internal.commands" -# prefix, the full path makes testing easier (specifically when modifying -# `commands_dict` in test setup / teardown). -commands_dict: Dict[str, CommandInfo] = { - "install": CommandInfo( - "pip._internal.commands.install", - "InstallCommand", - "Install packages.", - ), - "download": CommandInfo( - "pip._internal.commands.download", - "DownloadCommand", - "Download packages.", - ), - "uninstall": CommandInfo( - "pip._internal.commands.uninstall", - "UninstallCommand", - "Uninstall packages.", - ), - "freeze": CommandInfo( - "pip._internal.commands.freeze", - "FreezeCommand", - "Output installed packages in requirements format.", - ), - "inspect": CommandInfo( - "pip._internal.commands.inspect", - "InspectCommand", - "Inspect the python environment.", - ), - "list": CommandInfo( - "pip._internal.commands.list", - "ListCommand", - "List installed packages.", - ), - "show": CommandInfo( - "pip._internal.commands.show", - "ShowCommand", - "Show information about installed packages.", - ), - "check": CommandInfo( - "pip._internal.commands.check", - "CheckCommand", - "Verify installed packages have compatible dependencies.", - ), - "config": CommandInfo( - "pip._internal.commands.configuration", - "ConfigurationCommand", - "Manage local and global configuration.", - ), - "search": CommandInfo( - "pip._internal.commands.search", - "SearchCommand", - "Search PyPI for packages.", - ), - "cache": CommandInfo( - "pip._internal.commands.cache", - "CacheCommand", - "Inspect and manage pip's wheel cache.", - ), - "index": CommandInfo( - "pip._internal.commands.index", - "IndexCommand", - "Inspect information available from package indexes.", - ), - "wheel": CommandInfo( - "pip._internal.commands.wheel", - "WheelCommand", - "Build wheels from your requirements.", - ), - "hash": CommandInfo( - "pip._internal.commands.hash", - "HashCommand", - "Compute hashes of package archives.", - ), - "completion": CommandInfo( - "pip._internal.commands.completion", - "CompletionCommand", - "A helper command used for command completion.", - ), - "debug": CommandInfo( - "pip._internal.commands.debug", - "DebugCommand", - "Show information useful for debugging.", - ), - "help": CommandInfo( - "pip._internal.commands.help", - "HelpCommand", - "Show help for commands.", - ), -} - - -def create_command(name: str, **kwargs: Any) -> Command: - """ - Create an instance of the Command class with the given name. - """ - module_path, class_name, summary = commands_dict[name] - module = importlib.import_module(module_path) - command_class = getattr(module, class_name) - command = command_class(name=name, summary=summary, **kwargs) - - return command - - -def get_similar_commands(name: str) -> Optional[str]: - """Command name auto-correct.""" - from difflib import get_close_matches - - name = name.lower() - - close_commands = get_close_matches(name, commands_dict.keys()) - - if close_commands: - return close_commands[0] - else: - return None diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 12201716a95154e0ade0fa577004ae52bc18f0d2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4002 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 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/check.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/check.cpython-312.pyc deleted file mode 100644 index 24bf3670b823cd2ddf2ca0de440668956ae99b0e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2592 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 P={s;g{BI6E_GJGB-`y17 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/configuration.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/configuration.cpython-312.pyc deleted file mode 100644 index 6d49c91a759a84502dc59486817f2ef10c837ddd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13163 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 ZUsQV?ayk|j%yDN;&gQcPk>X^cq5K)53X3J-ek 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 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/download.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/download.cpython-312.pyc deleted file mode 100644 index 8df25910577844701fefcbf0d0656436ae0babd1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 7503 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 V1?a!X+z`dV&)@rxz*=tO{{d;L3Qqt4 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/hash.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/hash.cpython-312.pyc deleted file mode 100644 index f89bf8233a6bab6c1c973e7963fe5d930a38bbf8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2968 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 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/index.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/index.cpython-312.pyc deleted file mode 100644 index 4e8f9964700df3bf04b9a2fe2ab466ba17f4165a..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6671 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 diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/install.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/install.cpython-312.pyc deleted file mode 100644 index 8f93260017593d42e2a61a4d5ea9b2fb34d9b87e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 29114 zcmb__d2n3EdFPuGgE@d1oP+xf4&om025*Wacu5pQi(|HE-CuwG_1E9`b>lyKJWdY3)j$93$ODPv{+>ScUzrwIoiKCU zRZidpQtgjI^|6ML2KMcWHpZGpnqti(&FtG9ZHe(CJPUiGfmrKED+`xJ+hXk_?JQg# z?TBp{*}%fyXlJZzq$}1v(#^ho(VkfENG}UlMEhd>BmJ@9NRWM3Mh9XWM>eu>RdiEq z^T=ivu8wYrZ5`Rl!Zp!tvF#(_Oh@) zx-YhWWPj|bk*C;qUGzZg;K;$)p^-x-&diBh#i931>gbFdHgiWgq5e%yXb@}PvuLqL zN;&;MAt*G8M};QQB_69ove0aVS`g10Ujfmn<`{Det#8^!j<3rZ5RZ=8g|;`XBTpN! z2ZVN%Kg!^C7?3s?p-v;zg}MlWTj&PtXM`T{gwTt)J{Es$%pvpx^0U*X?u|dEP7HUO z70;;`PM&=J@TpUwV~0Et<- zFe??$i(j3JNMcNkCl4iJv2a{a+!rFrbD`0w7>-X(B9HCx^Upu`yi#)Ng+qtXQpC8m ztcN8jAt}D&aXA@|M#I!p7DY{=YGgEoHM5+)#P}-_DG{fts6BCc}}atkeST==o4Ytv58PC9`hok>in9#CS*&ze?ED zD=X-&(xd{3isRwY>5wcYrzU}yP#$9?FCHHrdhW$j@1j{rQBH;~NRedtY*Y*h5eaQA zMO#lK#>de{yW&@&i5HWigz<__hk(oQxF{%2F)mL@tdPR+Fd3N)y&_67u%&cON{OU6 ziV;ENn3RZxVhJHKhEXPDhVbt~BrYT_$V%mfuoMqP#zN9mJQS8gQX-MWP@zd;C^40s zoFepjPK&Y0A&lo}G9gW)<9_6hN8;n5skk6YAq+rrDjdZ~C{_7zD4qz7M-yj3W*BXT zeTA3LC8jV6>R5~=qylU+G8t9pq55=aAFHznT;)WLm?s#g2~5)rhFKxlK>9Y(F4)Bq zp+t0urEEf3L?=M|E6dxYd)-*=# zG-+rxTU~L{q)BrVEPCx@LR;?&W0N-hYr?LF@^5`Ay{FY>dNRoRl+FX@J}1ab9xDKg z;O7L{y%s8ceNK>dNwJ`U-vqK?63oP`1luk9d)jFJI^!gi2o3>k5)!BDmis-83QSx9 zWq1pkss)Qv*ptwS;1SBugYsM6_pB=LU#s;#l&(lt=q-FgD3wAL!$37K@FX~_5o%d^ z|C5z3z;%`0ll=QPqK-PDUT6S*8-bZ8=~()r+Vl&!EFI#zYZ_|{+uA!=OOB$&k1s)4ln=yoFF&nA?l&e2{Nz4 z%fCM-$gO#Zdgyb4+^)mRzkd_RLJ#Ibuh1v-KOw&lLWJ%W25xP9B7VCGr8lGYEl*hc zR+Qg{^4p)Re8HUCkjJMU`c3d!0J%$tmw$gwkbCkF_0Z=8xlf0ee}7JpgL#O0=yQU+ zQHPg*KMc8GTsG(P>7k@1GjQC*Ry{8NhLo~B|D}iCgNBZNGp6F{6Fb%P!ly8pHtFS< zd{PANBgz$(#n-?ptVjzNlhuVC!cJiqcmcrZ6A>hySz?GloQcc#7-U?-@=Wq z9PG9#cC>`F7FGf35VTVq3r|IpiuIfrolKeglPO1kzp7zUoSLQy`&`9>N9u)ArL=sJyjE1ob7B4~{#?BjI5#C8}z);KUoBe4XiWGht`+6R|O5!6He04%%bcz&f>Ht$M)VSaarw%_K zI(XpJVOH_j6p$*YI<#PSw}o|8q6w|`5cC*NsGW4F^f(4hgGfDR+NM%^hG2C2=frQE{nn1~vR8Y*{tHYDd&WkhUm^ zd^B>lj|UNh{B^}lGBMd7MH8avpGpv6WMmX01o>goq9kcrsKhLTSc!zRVNt7yLHhv( zA{2{pULKVqlhCN-E1>B>S{P6y)1RQxXi6LeS%7>HvM8acc=Cxc9vYfH27>69RgcK8 zd`0i_SH2SDY4n&9hoG<|Mm1mg3a@HJ_{bQmfFQ?d%JG;JATXVv3P>5T&%Oh=m}N{JRoed7gf+JFoVZHo=aTFHvtu}-ci$3r6%!3 z)>`nqFKKLN7vG_UUJmlD3~DRipNB#nA==VfB!Jq-!;%1AgpQ37g|Z=_4vtPqAdzIA zU!xk0!SFJOPKLS=qlugaN-_qe06hlwyX0;(Xl=KtWouhQ>C^~@V(`)gW1ebl(C+;j zb!i-sF+Qba^jsnW>R0)LufHFhWmN2!M2!%YauxwKyB3k!{k$FofkWM?>sK*JLBx!z zp<)Cg!%;~L3p6ZQXu*~PIRz-VN$$mjIM$fh1BiiZ1K`2iN1j8wSj*E;iMa< zK}YY2&C;?06AFJ8^`2Lmuskg%#h51c_ z2pP&{{^Wtv$5NH~Dh&IKq5=<}Mb5m{Ce=|0oF!tA66ZlMg?vOJ36NF}EbP#GD_$~s zf%51hL>N=7^bEfUUT{>5gDZghhxw;ZJvYph-~dLw!U8gm9$--*O}Y?aVvo*OX`*N< za+=|WnqU5>$Wxju$5{WvMEn@(iihbgj%f{=gXkdepf2hSPj)^&0>O?9J!}n-lLF8e zR0|k14*flg@5AZ`tv(2yX`8jhUa`p^~ko`JOJp%$600(9wU338CMaE)zK$|+jNuVK55pt^j zxGMY}QZoh%(vG7=^50<-_!^gUxZm{@c{ekf@$6Bt|! z>|Ao}dMIg2Ca`5Oux-h)eO=N!&2Kj^1~x4@Hs4!c)7DI2>tbO0l4Ed9nVh>eSJ{&D z)#s|}R;*^DezK)f0mf2nEd{EDo zR%RW28AspE(?5Fo-Ivj&HEqbbt8zXbb*TkikF2GZW#gB9%`5OqKvQgJihEsC0IN}L zN6vQ=1|`a8N4~Ok`Ow~Gy}fo{i`Ka^ zOyJ;RU}(v4c+D77(!NY!-(ujYCC7n>lD208+ZO{nmK-}DO4^wT>|6}&UUKYtC}~e7 zuxBx_Z^^O$p`@dkz|qCP@g>L8Ybb!4u{#sky%^ZL(L%4$hYH>`KU|VGJ$Q2fx#um zj)%JR8B_Q(Fq=ojLjXU6UF65(esxQ>dQ+x))3UoJ>u$}sTS3HcJ#*=qRhy;8IqO_; z65?E!Tv$*9l!p#e*MNG2-H$~}iF)Q8pHq?gOT1zWO3|I%Ak zP&S~ap)Go_;EmWKm_TfiaYSqptRuFy0H7AktKZY6kdZ8KGjK(mu@{Fg{h44vZUd7^ zLj^xkL}fnFYk)NZ#o_7{2iOp})!$=a_v`uc@6U!UZG8;HfjlHV^w|*Q!dX+I_u(6+ zB<+pZPjK8S)#GPuXJe+pR`JSe5pK%s%cXGd8%Aj?<6|OvH0R1UFU;gce z3j$`ATX)QyahHJ(M=5O=)M^VYJ4RJZz+y(w>6(bu=G8aZC^A-v*A#(w-si zoHjSxx=8y`mCmgl_^^<&119A7XJ zj6m8;5YuIkfNGj(H?kUm;^T$b=Z!2zAWi)6wDD;K5c@^r!wB3ed(WkQ%=pmWmy84> zkap@=1g-iezGNgAfwTv(i*PJAcdJ|s+d|40Y-+JvcmBk?l z-t*v3QbRJVmq}MnjOyY1Tc_AtJ{_WoiyuphdX7gxuMjG4Rp~Uljhm@Tj_V~R&gr52 z`?Kp#R|(ZZ&8=FUvYAUW+`)A$F>X*~RhEt<&+Byv{x`Yem>PP>iD{-fT`jdHV@6px zj7%i-ul!qR0LXG9OBS(F@Ot6e|>5^M5x*N9`QxywoKAh5!ppD3pF!*^4og(bW^$nIg3dyB!?7% zkfs{-;stNSey8A53*KsbPva^xf#g*^{SgqT#$qig-j3puRl80dGp)(%Mt!XY9^2A^ ziSOz0`FHa7_3*^M(nI;T(4m*VwLuTD)xSN$&2*&OCf?5H)kArzf4Kp4x*@$`;t%w+ z{5#!&_#foI^$;!pmz%Km1+Xq2LHW*f=fofCsrmOzSGr^3-Tb#6O6t{f=`Ov+c|JDN zo$i(#>8|f_f;Zj$J?{4{Gd*#0x@)E<-9GV0I*j}~-J`=WahH#wRn4!rf?|J^&!LCX z&4~Z8{*`~{C7)ZJd9GEQ+ojv}+Gcvwy^_0tKX#>i^QA#8MJQkDP1|Ji(;Vh3H}Plr z7U-ekrP4OU8?v>Y7jCDtufvaV{(q;23!g$STiu+=f3L^th=&tr0Z(q@HgU4)0%XRQ z0L#S9SuSt5XnBddU~1!%7xl93oMgwEQ!`Wh5;teQ{O1=qHP0x@jhbibE^x^GZwhih zJJW|=7Ljv=?(=6rHT}sy)^n%((*5Y8{#o$W`#2LU*h~FMK?W_D4yOBW_2luNyrbtC zHNS*aa;EQVeW$ed1(V+P?!IA|M_{iZE2^%{?o@5?olM7$4^?&CnJR|~Ni%l!lP>l& z&O*3VeXp7&<#>U)?sv$rZ|y?;5((!LN!W@6WBof2~HG zx@rPzYPWLJ=0CDrFkt|!WN}eSv3HWMS|z>mS8o?tQ& zi}Gh-P=i5UjgtAUS7*9Y=DxKkCrW1YNsPrnx%biZsUf-oh~@~<@0x&P=^H35kx=@i zsQqI?Ek;^HZQ#I$Zcs3En+Z^Dr^D|70xo=bukm0ylSc-(>$yD8`sZm&# zWO6_=g z#-T_K)Y*{$moXTKVTHumLE^o3GH}9nMzb5-P?Q3cyHdarE*Jrs6;7K|5^)s90u^%t zx?5Tdl#=jSnFZ|XvY^-^GO=LNIw&5nFIe$tA%HcJk)BdXrx`W35*QNM0wB34ceTdB zHR~Ge*Nj&)c1we&IAC35M=-Gn07#)}RU*x$1f>VqI^)-L9#N^8B<~lTveay=8VgqH zm_4zee8}MbQv|}Jr8BobN#*s7&6lZ%M0HlLcxiPJVYQSa+EA6~16@UcY76R+L&=2h zts#*huaqD{z%VG*1YCo_N?^Rl$}kn1RO~V=_@W@SBcJp;^eCqi?5tz~OvSA`&)`(z zv=YGd4;uE!AelYs93(lC2qCr{DUN5v>BDqHRl+jCNyiAP{S|h?R_dmhE_(P_H2LC% zlS&OCTcZJHQ4S>}ETXWsGJg{?VjGT4(jh8Yg*eUeDDTps&Lh%7O9v>EpTNR@&2Xp) zjpLXvC_<{FWGi7naR4)LTEK}~XE>RpbGm>F>o<`Vd})9_qGX67AZ1y4g9?_xp+!9t z%nk&ziKtjmQX*{?{52=3xu7=Y)UnR*LaEQUTpvk&$|@z#onolNSzGwa&>2%X1QvjE zpfNNYPFHZfVyGpRzET0}HoU)#^TjkPAV+di8y?*QQkcRd2T7!5QQ9;N6Bma?8I-QN zOcfpxX46?I*Q6yT^bna>t&3oegKV}jsYi-aB}sb7R8uL+sE1*}?r;`ZT`nvLO0Q75 zL+`j!UO;MO^rtg>%+JI$hO@WqEUeb*5L@|_5)dHGOX;6eZNEzoQXnWEs)y*GEi;NK zq?D=B*%%fe90Z2vM1FE$2J45Hl?Ls|ZLzUp1f`T}Er8>jr;wW9S3yU5OXMjS<>8M| zVdT?iFZ>P)$p?*B9kW<8{3WMV`e8(6B?hLHRRFS$3| zYv@~W_;ap`xAuQ)f39Uiwq-EWGPvMwyzi*UI=VBC?wg(OTR(6dzV9e|tL#$QTU6dxocs_Jhf^u*H)}>W?#hvC)d(<+rQMjYxcQZ+vaTBV5SXy87wZRSpILQs{>C{=u7A^<3@T-LGr9HlA(Xo@w5G$MXL82X(^>&!1ZM zH{K|{UOK3+51Iv|LZXZ~x-2RD;^Ht>P_N~}BPyGsq zfXfpYGiMr!_ehAsG^^ z7Xt?t{Rc2(cGaxfY&30bz6!Y9)VF5qc4X>y+-X^?J3jj~kc1h0{2uX=flSrFf_LDq z_vG!P*{vrtTTf!>JY63HkG;7@KHIo0)41*S#A4%txl(khy*JyoKhw6Krd@eWwmfjR zJU~>u<8H@}TxZ{cw;l5<=dI6qn{(BT+3F3M>J7Q-ZZ=hVGPOOqj-7cjtPk86xIXZ} zS6Wv#d;CES*R+k5>B&_0P#L27X(I{%4VN^P{viZ&i5}?Y@9uWws~sa z8oo6A{fpU_&6$?Xx3@30?3x{3a_`R7_-Bvi0&NTK<{aOa<#%QHU4M%~bLS2YXAgcc zbMT7`V7_~@ZTm89`|f)80|9}(f8WqPR|1ZDBYi!Mob`R#`i+_TjZ2#jzQ1R2)5(SB zzqnZc;#|pxmG!yio@-yex#{MMH;*qgY`-#euc32p2z;J#x#S1lc1+F24$Mh!-9mXg zrb*V@lkxWCs+w{g-Pw+zOvljsFD`aG3nK9KKHw}EF(=m?xba)pe`|hxu_-t^{9$?1 zvKQV|1G%bJ&?YkQ9C$b5yB}7zF1TCg_A@5fmT|YS6^1P*&)t6UPeMNqJ+PQPWf*cQ z(Ux(w&A)Qj)xYd+S#k%K%WG!Ke)$C}&74f#@IO9i<*FJ#;Y^+lIlklDo)rs1|M!zd z(B-clM9r94pTO;`4vg-`m#=^M=2tUK8)t{_*YxMS)imC;j&)?b9jhhQ+OpZBm=B)H zw@zF-ajo{|zS$E??p^mQ{l9zSA6KdY<5QV7G=H_>aNU^!?yvWEopD+Irt<)OWX!%Z zelAnF7va0!sxv*-yY0t(_*iz?&QzBydmR+^yUui4mfI{8-e96|CyO5_{j$xnyyJlR z%bew-T_$||9cRN+Rj*}8D5AaDD5);=AMFP$_Z!2atfZ^RRUlglxf_5kCerda|dWK2H zIzd~Cb-#QLD0qK;_=c+j+-OSD_Vd`HKe8qx?-SMJcx+FM98hjshqyV@%jclwC{23w zE*DdXrwykHR^kLoIN`A1U~2FkkHHA-h}ZqdfUl4Ix7>NxpIP@99DUw&r0JF}Yd!a{ zkxnvQKJWzf!F55eZ^reQ^2I2`W#ILC+9KB`X>-Va6Ql=Q_Y-VsmtGt6lh+?pUl9i; zl-FjJ^tih0YwG%&$J9momUX{^x@wXEJ#m6I-Rpk&oG{W?7f?+5x{UZN%v%6;!^1fr z)8b;(Q<@*m!rsrBEn<)$>CA@yy_{+ktDFZtS_f=cbj!(7|J>j_{3!X4B9AAAYH4POJ71 zV4hBK0C>zJO;rL~ZK zILIu)LM2*B=`z)=t4&B#LCMks#LC2=3peObsr2+Wh)x9_OHXA)JU;|!!)SrcVrk_Z zykwgp2m0%0-H_fZ8m}Fk4`n-cW;%8*`F73Pb2W`u<5%Ldu6&@f?&|IET={Zk_0_>EgID)m**8D1RJr-XhSp_& z^Nq6WW!NM%?Yr05i*0FLJ%q}-z;bQf)eBcHT>aXWugxcxYPTC1Elr)kTvHz;Yh*1e z$U1XnWplL-POh%+W9($xB@v@4{S=Q;jGOTz35uA>>RCLl!sVx!>~)IF#YQO@mvR4O z!I^zhEw20J9gxXTJ}H-V3kv6gD_JJz+e*|K;{9KRUW`mRoI4V(M+ZE z@WoM)U4y93)>4=n)Nh!w!pOm}w8q@>7l=)@t)CHjZK=`lA(|6pPKF)NUQ+5`uI#zF zYq4_sf_wX~mdpL0;HZXw*c?5Wwammuegx#S9Rm9q@2;!5>uXF?^g{ag*SNucx++_$Re%5bZi{ecS{on;?VDx?g@% zkjE#)j`Cqxs@S!ZsUN5lK>f-ehn&XJrW$N;)ECDBRUe5soxYR`bLo^s$JcSZ5JnOx z!)f~Ir@FLrv1i3|;a4JbF9DWGIxMIiM8)w}`oiJ*XaW~eju(X;RGeCCum+{xQCn;n zN{TksI3b=>qga+sZfj{z$qdI`yx*8!;i#*qA5EbozZBwU9}6pfyODWf@&0p4png#0F`hW zgZtFv?6xt%Mf!7;Q<5=p%Pm8f*cPh$7&hZ)INJ6R}+};%hU|qHf3ryL!s5& zF&|xO-ZJOBSKBrhpFc1^KL66qi97Z0ANbRzJLM~!>6j^qXLS(zET12%yRYV|`-=P8 ziywG9K6Y^}TcL1jYG1A3nz|ka9^84SZz*^bda60cwcdGo#cHZPV)}0t{wo`Dbsf3J zHfYjvO&vMDHwPVT^lt0H>(zhhz}dITx>c^EvSOu#>a1B)XB{_i%tV!TJ;2$%^0&$_ zm0zpchiyq0m=x{S1*kzDhzN z+%Ix7LpAQoZ3y|(ihBX>3>liGml%(CK6GJL`{kKzC;{Ws-eeU z(1S?eT}~BRMk%6Ke}I*9@@+iceUqH`|;m2ko4_V#R#GYOB4r zeTAbp)c>RJ)~Vs!@+T~A)na5@IpWxAd)kz%Z(6aiz{b5=;MfT@aIe&}YNMo;sL5&b zL*c&w(@Fj26$`%ZmHOreGNrAnHhir3H`;n~{J@HZ-iMB`H;O;7;alwt>&I2<%y;PV zZF*dz$9L&Lm>(sa=|4j^b{a8-nlb3Vu*~BKg(Yo;&P!jQ=zxkLsp&wlceWMNT#cJg zaov_-X$2X<8Xnau>^|Af=zk-d^h?TXe`H?CgvlsVYt)IYPU!zYCF$`7%~u*hY#GY9 zGJnqD%z1p;KVMz0y!wHsq|E->5g56kyo5UW+L0T>*M}Eey$jY}hQT%BE3KhS?iog4 zEqdPM^2);e$v8q#jSEmD_*jJ289UfrTrrzZS?k8b9=% z(A;~fYfbN(^P~o+HVal&OiTC$u%YRsK?6Vx8XGXBrg%G~9%TG3)!dEDuR$eT1|v_o z{l2eisk;C6*7v)Xs-Ic%oq(;U+zwlh+YWCw8#cPgtvc|KY@BzsWL`CE=$)7b8_*x1 zCi3We-Jj;YKVYi!( z@z7&@qg9VSxJomL9&UHBNjTr-@3PT^TIS`1n?Ka+U*yz?4j0ZyHI z8ss(CL$Eu2NLGF&LRV0cv<`G(4D#2^2UBV}4N9?Yolb3esP_Ui0U)k^?Zz;=8clcn z*2XQh^`Sy%#nCVbOQpuO4TT>J`FsizalzEaM;2XMirv3LEE(XsOI5+#CWvRJ#wF(K*opb5x}{mEu;M*DeBj`T!y$H?+o|KH z4>P_D4_tD-wmp*3$|P72b!%ew+XR*#-zScK7Xe7otz2o%f+O(mv&-;UabI#TRI|$s z%By}>QuU#$27WfAw%D?2u~e3UxqHf>>2vc-uE26d8>MVtlL9G|!C;5~%PYwWTDRch z7py#nU}Te3hjCfrnDgZtTzVmo-`x5?k2fa}5Z&O1cu9 zQ!lC^F;3;uc@DYn5xENVD8+N+C6qL$FH!m%x))#ZQi8FZp>|vve$jlw7p@gIzB>PFvKeoR#mDc@Of489|$+zo%Qg^aik1C!+ zA1T3keuyUMBSx!WxfI+Jq=H=cOB;BZpkZA1)3GPmQb>VZ>PbmQm~$!&BRuoDZTjxrZgy4#EtEjl~K0!cUe zsB2)V?+`m;P@w%VbVh*2IC8$`lzA8bsVOM^0xgy*h^@9Bf{!ZGMyY=?gDS!eh{P=Q zI}TIThv=`ksLCGlU1sjla>_~^K4lI%BUXa+SI3SIpWdZj-=_6GGJY3+!v$ix9vT`iTz@nf4n`6Knkh&gQ1jwmwfH#hP)eSYo$9{JqpDqr}bJ_WwzxbqP=8$#oT{*rMR8HS%bJbYSlkbazjesrHQUzW@xPC1`A*a}!rljg1F1IqNrX|CI)j9v{*&fj>({%LIHO%U?ST zer|Jq^Tg{X7JU1ctWSODs>?Y%#PR1&FFES9@OMsKtN!-L2M+$8mUe2%QO8Qow!U$E zt}0`#e&aZpwZrqxS6_eiJL1*omFV4))?CAWa;ldi2v+z`Z*OgkwNx z&0*>i92UdnukmrFHeo6erGbbE%0c|mA1SDA43Qy7mPA=~>h>1qz{!Sls)Kzci`MVw z3aSUWXrrU})}=%v>V0jE55))be=$lurqJ|o76_p5b7!nVBGUy^{wU24XM{OBIVB@6 z%&rnKZq=>dzc!~~jFDB@hg@fr+O>HM;gsnmaMFQLWb{-&ou#_ccdbiA?f4Tsg|e=8 zWuR0=HTA8{u6p2;e|?boY^zWGPfhu7JnX29dXA!vKHI6>Ito(D>8|>!-mhp2^Lqlu^CS{@*Ef?L$x?mj!3JTF;?y%6 z^)$)I?5?T^1N2GERMl*d!h*@T(tn{WjF;iB;1y^VKXvR;Rh#>y7m$E|oO~HiINzB} zKjGXz;aorA%74O@{)F=&{8O&yr(ED4xSbzyo&Uh~e8hEq#O?m2*Dd)33AMM~IotO7ORFCGYPXu~^ZuJ{Z#TgW)?~la z&R*|tU8OJhPPW1$u|3z)maA|1*tN^#%~dvB8@jS@#e(oXU){B~D@`jlgi$%ZDzA7} zO6Uuo$Np=UOEW8_^yOqmP6|=ws~!raOxsP}@M2%F(7U}$eeZ@7D8BF691m=GGw%BT E0o2V|egFUf diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/list.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/list.cpython-312.pyc deleted file mode 100644 index 82257813f97fea3c2c0c52c4eea76cf7b78c00b6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15757 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 Pv_t@0>EFbLDbfD|U#AVk diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/show.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/show.cpython-312.pyc deleted file mode 100644 index 3844e263f6df8f4a2cb625f7ffd7d252f1ef9340..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10477 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 R6m&?jl-97S2a0IN{{lDIicbIl diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/uninstall.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/uninstall.cpython-312.pyc deleted file mode 100644 index cd9ae8daa6643b14bbc315ff43b54e0dfc559c64..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4709 zcmbVP-ESMm5#QtSD#uJc5InW$T*Ih4?~gb$VTiYG1@9lYevEO%v(t(-v_gI zluUsJf+7f%!a!2=!3l~YDNvv?WT5F&lb7ZX=!;}46>+hF0!<$JMpq63ZhmA-n!s(D6HDaY0rzJgZBuWWRhxI-qSxRy` zqNj{>Db4Aqo-wketkGZU=lz&IU<{T9jiJ(zF!=#V8_B$PXjhA+re#hMD2>< zmrZNQ#N=w3v^<9uN=~$t74T21lwcKHUUf9x?nw10TGE_(eCbl9=FD3rwuw`%6e~54 z;nJli@Y1|ip6AA4tP^FC;LBtlb`qNwHi)CBilZ=FRRS(Cz?C4OICxgmiH(;C7|=1h z$ki0^beU=u2QN{jQX$mtGT_U#U5ru5BGIkNrAsaOyHFG%L$-5BW*=Id9vHKpc`C7aJs-b+}`DI^$F2@Un9$&h#o{sGXS#X)E>t~U; zywFWKKn?87lWsQXkRXs#n#qUfv_)d#UR69siWt8(52@LTHVpU)d4Q`n2pU!0(JCxs zH3Kp^A8`FDu^q6W%%H~&sbp3G9yi4DWl^E7f1cihSU!FQKa{-v_nV(y#gt!`o4Avk+>}o4_$HfrX(RX2o!q%i=_hO|`d)NP8sCt{SB2H%e}3WS z3tvjlZ-cSO4YkqD^(%DFF82Zr0~8KTS!g*lwbP9WdQV;O()*gIp=AI#(p?MIgS$Zp9j*uKeusf1e|G2cESLbE>nEKK5bT=%zIwp> zs(`M)?nhu-qOHE?1*`ZOtRm#3yuo^?F4Vrr3>_UP^#?BCz*d6XKyavEYO;(7I2 zxE_Z6JX#Mg9Q1g)Z}0oy`%~Qahq`kfrADtD26T%!EUEd^m2!@FW78-fW5ute*%s8L zL|vXfXD&j)tM-EX34D-ZZO#PXRdInI03Cz@)a4a0R96)!EN#r{ug_CCt2a6@*h zL|#5}(W=6U0M!~hNHA1FP&7f$g2JfhP+=9*auOf53BmhW)yGZ*yzJYbcSY!e-Em&N z)--dDRkSTE_qj2p>R597IB-LN)t3QDT|X;n^8;>XjyQ7rxRKi?(HJu@?rnn%dO?7# zg|hm<4eHk19HDNA>448s7OXVN8g@QYmamjKma;sgaG3%b*hsihxZ%!bQ5H;YxMh&$ z)Z8H3J3wEZq%CL9UrfvxnBjr5V#XYZt+ zUJ3nk&%SkO_^a&D%IUl5@kTP+7{*(}#f{)11 zci+)Qa_E!sPoMox^o=Dxh&%|Q!4r+_zSY64{NzS{@`30ZOn)GPj_lx#=YIQK00IdtlF_Qm^sAlnqtaN&L$C3APv2O8;#&tCiD;^!AXd%1~(v%=oS z-rUC*Kf3tg%b(DuD5M_W7GV`lB*qiX2ucoa^&Pw2ckFKLvCY__ucf1nk+Jmy$Ja-W ze=GXnT$51t;8y1Gjm+c!$P`wl8u-AU;(v%Y_CEIU8y~&V7~1o3;-kcBYGdd~Gk|i% zZ6xLr>#@E6fgQ>_sFOx@_ag;5>AsNeTMGtI54}7wCJ6gCK8{3TK?)D9= zBq-!Gf^=eVeLf#`C7It#W!a6$EM}`Z0XinjZ&ekYMFaFgvaDKVAOckzZc%g`WGTac z%8oNvv>ob-P(x7W>zjywf`BZ(8)<1{I`aG65KAIR9#?_`yHzDIJKUgCtFWBn_KZRz7iB0f zRf`r|g$~RW`Q@yYiz&|GRXxSBuC?m4eLom4URL04-9{+NY5{lbhfHu!BUd|ItW1|7bKj@?5idHc*gbo4u)DEPjU{DSX3f~FZmk%Xw! diff --git a/.env/Lib/site-packages/pip/_internal/commands/__pycache__/wheel.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/commands/__pycache__/wheel.cpython-312.pyc deleted file mode 100644 index 2f5ed9eb276f4f1e8d9671b8e2c8e27129b642f1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8865 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 diff --git a/.env/Lib/site-packages/pip/_internal/commands/cache.py b/.env/Lib/site-packages/pip/_internal/commands/cache.py deleted file mode 100644 index 3283361..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/cache.py +++ /dev/null @@ -1,225 +0,0 @@ -import os -import textwrap -from optparse import Values -from typing import Any, List - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.exceptions import CommandError, PipError -from pip._internal.utils import filesystem -from pip._internal.utils.logging import getLogger - -logger = getLogger(__name__) - - -class CacheCommand(Command): - """ - Inspect and manage pip's wheel cache. - - Subcommands: - - - dir: Show the cache directory. - - info: Show information about the cache. - - list: List filenames of packages stored in the cache. - - remove: Remove one or more package from the cache. - - purge: Remove all items from the cache. - - ```` can be a glob expression or a package name. - """ - - ignore_require_venv = True - usage = """ - %prog dir - %prog info - %prog list [] [--format=[human, abspath]] - %prog remove - %prog purge - """ - - def add_options(self) -> None: - self.cmd_opts.add_option( - "--format", - action="store", - dest="list_format", - default="human", - choices=("human", "abspath"), - help="Select the output format among: human (default) or abspath", - ) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - handlers = { - "dir": self.get_cache_dir, - "info": self.get_cache_info, - "list": self.list_cache_items, - "remove": self.remove_cache_items, - "purge": self.purge_cache, - } - - if not options.cache_dir: - logger.error("pip cache commands can not function since cache is disabled.") - return ERROR - - # Determine action - if not args or args[0] not in handlers: - logger.error( - "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), - ) - return ERROR - - action = args[0] - - # Error handling happens here, not in the action-handlers. - try: - handlers[action](options, args[1:]) - except PipError as e: - logger.error(e.args[0]) - return ERROR - - return SUCCESS - - def get_cache_dir(self, options: Values, args: List[Any]) -> None: - if args: - raise CommandError("Too many arguments") - - logger.info(options.cache_dir) - - def get_cache_info(self, options: Values, args: List[Any]) -> None: - if args: - raise CommandError("Too many arguments") - - num_http_files = len(self._find_http_files(options)) - num_packages = len(self._find_wheels(options, "*")) - - http_cache_location = self._cache_dir(options, "http-v2") - old_http_cache_location = self._cache_dir(options, "http") - wheels_cache_location = self._cache_dir(options, "wheels") - http_cache_size = filesystem.format_size( - filesystem.directory_size(http_cache_location) - + filesystem.directory_size(old_http_cache_location) - ) - wheels_cache_size = filesystem.format_directory_size(wheels_cache_location) - - message = ( - textwrap.dedent( - """ - Package index page cache location (pip v23.3+): {http_cache_location} - Package index page cache location (older pips): {old_http_cache_location} - Package index page cache size: {http_cache_size} - Number of HTTP files: {num_http_files} - Locally built wheels location: {wheels_cache_location} - Locally built wheels size: {wheels_cache_size} - Number of locally built wheels: {package_count} - """ # noqa: E501 - ) - .format( - http_cache_location=http_cache_location, - old_http_cache_location=old_http_cache_location, - http_cache_size=http_cache_size, - num_http_files=num_http_files, - wheels_cache_location=wheels_cache_location, - package_count=num_packages, - wheels_cache_size=wheels_cache_size, - ) - .strip() - ) - - logger.info(message) - - def list_cache_items(self, options: Values, args: List[Any]) -> None: - if len(args) > 1: - raise CommandError("Too many arguments") - - if args: - pattern = args[0] - else: - pattern = "*" - - files = self._find_wheels(options, pattern) - if options.list_format == "human": - self.format_for_human(files) - else: - self.format_for_abspath(files) - - def format_for_human(self, files: List[str]) -> None: - if not files: - logger.info("No locally built wheels cached.") - return - - results = [] - for filename in files: - wheel = os.path.basename(filename) - size = filesystem.format_file_size(filename) - results.append(f" - {wheel} ({size})") - logger.info("Cache contents:\n") - logger.info("\n".join(sorted(results))) - - def format_for_abspath(self, files: List[str]) -> None: - if files: - logger.info("\n".join(sorted(files))) - - def remove_cache_items(self, options: Values, args: List[Any]) -> None: - if len(args) > 1: - raise CommandError("Too many arguments") - - if not args: - raise CommandError("Please provide a pattern") - - files = self._find_wheels(options, args[0]) - - no_matching_msg = "No matching packages" - if args[0] == "*": - # Only fetch http files if no specific pattern given - files += self._find_http_files(options) - else: - # Add the pattern to the log message - no_matching_msg += f' for pattern "{args[0]}"' - - if not files: - logger.warning(no_matching_msg) - - for filename in files: - os.unlink(filename) - logger.verbose("Removed %s", filename) - logger.info("Files removed: %s", len(files)) - - def purge_cache(self, options: Values, args: List[Any]) -> None: - if args: - raise CommandError("Too many arguments") - - return self.remove_cache_items(options, ["*"]) - - def _cache_dir(self, options: Values, subdir: str) -> str: - return os.path.join(options.cache_dir, subdir) - - def _find_http_files(self, options: Values) -> List[str]: - old_http_dir = self._cache_dir(options, "http") - new_http_dir = self._cache_dir(options, "http-v2") - return filesystem.find_files(old_http_dir, "*") + filesystem.find_files( - new_http_dir, "*" - ) - - def _find_wheels(self, options: Values, pattern: str) -> List[str]: - wheel_dir = self._cache_dir(options, "wheels") - - # The wheel filename format, as specified in PEP 427, is: - # {distribution}-{version}(-{build})?-{python}-{abi}-{platform}.whl - # - # Additionally, non-alphanumeric values in the distribution are - # normalized to underscores (_), meaning hyphens can never occur - # before `-{version}`. - # - # Given that information: - # - If the pattern we're given contains a hyphen (-), the user is - # providing at least the version. Thus, we can just append `*.whl` - # to match the rest of it. - # - If the pattern we're given doesn't contain a hyphen (-), the - # user is only providing the name. Thus, we append `-*.whl` to - # match the hyphen before the version, followed by anything else. - # - # PEP 427: https://www.python.org/dev/peps/pep-0427/ - pattern = pattern + ("*.whl" if "-" in pattern else "-*.whl") - - return filesystem.find_files(wheel_dir, pattern) diff --git a/.env/Lib/site-packages/pip/_internal/commands/check.py b/.env/Lib/site-packages/pip/_internal/commands/check.py deleted file mode 100644 index f54a16d..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/check.py +++ /dev/null @@ -1,67 +0,0 @@ -import logging -from optparse import Values -from typing import List - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.metadata import get_default_environment -from pip._internal.operations.check import ( - check_package_set, - check_unsupported, - create_package_set_from_installed, -) -from pip._internal.utils.compatibility_tags import get_supported -from pip._internal.utils.misc import write_output - -logger = logging.getLogger(__name__) - - -class CheckCommand(Command): - """Verify installed packages have compatible dependencies.""" - - ignore_require_venv = True - usage = """ - %prog [options]""" - - def run(self, options: Values, args: List[str]) -> int: - package_set, parsing_probs = create_package_set_from_installed() - missing, conflicting = check_package_set(package_set) - unsupported = list( - check_unsupported( - get_default_environment().iter_installed_distributions(), - get_supported(), - ) - ) - - for project_name in missing: - version = package_set[project_name].version - for dependency in missing[project_name]: - write_output( - "%s %s requires %s, which is not installed.", - project_name, - version, - dependency[0], - ) - - for project_name in conflicting: - version = package_set[project_name].version - for dep_name, dep_version, req in conflicting[project_name]: - write_output( - "%s %s has requirement %s, but you have %s %s.", - project_name, - version, - req, - dep_name, - dep_version, - ) - for package in unsupported: - write_output( - "%s %s is not supported on this platform", - package.raw_name, - package.version, - ) - if missing or conflicting or parsing_probs or unsupported: - return ERROR - else: - write_output("No broken requirements found.") - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/completion.py b/.env/Lib/site-packages/pip/_internal/commands/completion.py deleted file mode 100644 index 9e89e27..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/completion.py +++ /dev/null @@ -1,130 +0,0 @@ -import sys -import textwrap -from optparse import Values -from typing import List - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.utils.misc import get_prog - -BASE_COMPLETION = """ -# pip {shell} completion start{script}# pip {shell} completion end -""" - -COMPLETION_SCRIPTS = { - "bash": """ - _pip_completion() - {{ - COMPREPLY=( $( COMP_WORDS="${{COMP_WORDS[*]}}" \\ - COMP_CWORD=$COMP_CWORD \\ - PIP_AUTO_COMPLETE=1 $1 2>/dev/null ) ) - }} - complete -o default -F _pip_completion {prog} - """, - "zsh": """ - #compdef -P pip[0-9.]# - __pip() {{ - compadd $( COMP_WORDS="$words[*]" \\ - COMP_CWORD=$((CURRENT-1)) \\ - PIP_AUTO_COMPLETE=1 $words[1] 2>/dev/null ) - }} - if [[ $zsh_eval_context[-1] == loadautofunc ]]; then - # autoload from fpath, call function directly - __pip "$@" - else - # eval/source/. command, register function for later - compdef __pip -P 'pip[0-9.]#' - fi - """, - "fish": """ - function __fish_complete_pip - set -lx COMP_WORDS (commandline -o) "" - set -lx COMP_CWORD ( \\ - math (contains -i -- (commandline -t) $COMP_WORDS)-1 \\ - ) - set -lx PIP_AUTO_COMPLETE 1 - string split \\ -- (eval $COMP_WORDS[1]) - end - complete -fa "(__fish_complete_pip)" -c {prog} - """, - "powershell": """ - if ((Test-Path Function:\\TabExpansion) -and -not ` - (Test-Path Function:\\_pip_completeBackup)) {{ - Rename-Item Function:\\TabExpansion _pip_completeBackup - }} - function TabExpansion($line, $lastWord) {{ - $lastBlock = [regex]::Split($line, '[|;]')[-1].TrimStart() - if ($lastBlock.StartsWith("{prog} ")) {{ - $Env:COMP_WORDS=$lastBlock - $Env:COMP_CWORD=$lastBlock.Split().Length - 1 - $Env:PIP_AUTO_COMPLETE=1 - (& {prog}).Split() - Remove-Item Env:COMP_WORDS - Remove-Item Env:COMP_CWORD - Remove-Item Env:PIP_AUTO_COMPLETE - }} - elseif (Test-Path Function:\\_pip_completeBackup) {{ - # Fall back on existing tab expansion - _pip_completeBackup $line $lastWord - }} - }} - """, -} - - -class CompletionCommand(Command): - """A helper command to be used for command completion.""" - - ignore_require_venv = True - - def add_options(self) -> None: - self.cmd_opts.add_option( - "--bash", - "-b", - action="store_const", - const="bash", - dest="shell", - help="Emit completion code for bash", - ) - self.cmd_opts.add_option( - "--zsh", - "-z", - action="store_const", - const="zsh", - dest="shell", - help="Emit completion code for zsh", - ) - self.cmd_opts.add_option( - "--fish", - "-f", - action="store_const", - const="fish", - dest="shell", - help="Emit completion code for fish", - ) - self.cmd_opts.add_option( - "--powershell", - "-p", - action="store_const", - const="powershell", - dest="shell", - help="Emit completion code for powershell", - ) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - """Prints the completion code of the given shell""" - shells = COMPLETION_SCRIPTS.keys() - shell_options = ["--" + shell for shell in sorted(shells)] - if options.shell in shells: - script = textwrap.dedent( - COMPLETION_SCRIPTS.get(options.shell, "").format(prog=get_prog()) - ) - print(BASE_COMPLETION.format(script=script, shell=options.shell)) - return SUCCESS - else: - sys.stderr.write( - "ERROR: You must pass {}\n".format(" or ".join(shell_options)) - ) - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/configuration.py b/.env/Lib/site-packages/pip/_internal/commands/configuration.py deleted file mode 100644 index 1a1dc6b..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/configuration.py +++ /dev/null @@ -1,280 +0,0 @@ -import logging -import os -import subprocess -from optparse import Values -from typing import Any, List, Optional - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.configuration import ( - Configuration, - Kind, - get_configuration_files, - kinds, -) -from pip._internal.exceptions import PipError -from pip._internal.utils.logging import indent_log -from pip._internal.utils.misc import get_prog, write_output - -logger = logging.getLogger(__name__) - - -class ConfigurationCommand(Command): - """ - Manage local and global configuration. - - Subcommands: - - - list: List the active configuration (or from the file specified) - - edit: Edit the configuration file in an editor - - get: Get the value associated with command.option - - set: Set the command.option=value - - unset: Unset the value associated with command.option - - debug: List the configuration files and values defined under them - - Configuration keys should be dot separated command and option name, - with the special prefix "global" affecting any command. For example, - "pip config set global.index-url https://example.org/" would configure - the index url for all commands, but "pip config set download.timeout 10" - would configure a 10 second timeout only for "pip download" commands. - - If none of --user, --global and --site are passed, a virtual - environment configuration file is used if one is active and the file - exists. Otherwise, all modifications happen to the user file by - default. - """ - - ignore_require_venv = True - usage = """ - %prog [] list - %prog [] [--editor ] edit - - %prog [] get command.option - %prog [] set command.option value - %prog [] unset command.option - %prog [] debug - """ - - def add_options(self) -> None: - self.cmd_opts.add_option( - "--editor", - dest="editor", - action="store", - default=None, - help=( - "Editor to use to edit the file. Uses VISUAL or EDITOR " - "environment variables if not provided." - ), - ) - - self.cmd_opts.add_option( - "--global", - dest="global_file", - action="store_true", - default=False, - help="Use the system-wide configuration file only", - ) - - self.cmd_opts.add_option( - "--user", - dest="user_file", - action="store_true", - default=False, - help="Use the user configuration file only", - ) - - self.cmd_opts.add_option( - "--site", - dest="site_file", - action="store_true", - default=False, - help="Use the current environment configuration file only", - ) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - handlers = { - "list": self.list_values, - "edit": self.open_in_editor, - "get": self.get_name, - "set": self.set_name_value, - "unset": self.unset_name, - "debug": self.list_config_values, - } - - # Determine action - if not args or args[0] not in handlers: - logger.error( - "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), - ) - return ERROR - - action = args[0] - - # Determine which configuration files are to be loaded - # Depends on whether the command is modifying. - try: - load_only = self._determine_file( - options, need_value=(action in ["get", "set", "unset", "edit"]) - ) - except PipError as e: - logger.error(e.args[0]) - return ERROR - - # Load a new configuration - self.configuration = Configuration( - isolated=options.isolated_mode, load_only=load_only - ) - self.configuration.load() - - # Error handling happens here, not in the action-handlers. - try: - handlers[action](options, args[1:]) - except PipError as e: - logger.error(e.args[0]) - return ERROR - - return SUCCESS - - def _determine_file(self, options: Values, need_value: bool) -> Optional[Kind]: - file_options = [ - key - for key, value in ( - (kinds.USER, options.user_file), - (kinds.GLOBAL, options.global_file), - (kinds.SITE, options.site_file), - ) - if value - ] - - if not file_options: - if not need_value: - return None - # Default to user, unless there's a site file. - elif any( - os.path.exists(site_config_file) - for site_config_file in get_configuration_files()[kinds.SITE] - ): - return kinds.SITE - else: - return kinds.USER - elif len(file_options) == 1: - return file_options[0] - - raise PipError( - "Need exactly one file to operate upon " - "(--user, --site, --global) to perform." - ) - - def list_values(self, options: Values, args: List[str]) -> None: - self._get_n_args(args, "list", n=0) - - for key, value in sorted(self.configuration.items()): - write_output("%s=%r", key, value) - - def get_name(self, options: Values, args: List[str]) -> None: - key = self._get_n_args(args, "get [name]", n=1) - value = self.configuration.get_value(key) - - write_output("%s", value) - - def set_name_value(self, options: Values, args: List[str]) -> None: - key, value = self._get_n_args(args, "set [name] [value]", n=2) - self.configuration.set_value(key, value) - - self._save_configuration() - - def unset_name(self, options: Values, args: List[str]) -> None: - key = self._get_n_args(args, "unset [name]", n=1) - self.configuration.unset_value(key) - - self._save_configuration() - - def list_config_values(self, options: Values, args: List[str]) -> None: - """List config key-value pairs across different config files""" - self._get_n_args(args, "debug", n=0) - - self.print_env_var_values() - # Iterate over config files and print if they exist, and the - # key-value pairs present in them if they do - for variant, files in sorted(self.configuration.iter_config_files()): - write_output("%s:", variant) - for fname in files: - with indent_log(): - file_exists = os.path.exists(fname) - write_output("%s, exists: %r", fname, file_exists) - if file_exists: - self.print_config_file_values(variant) - - def print_config_file_values(self, variant: Kind) -> None: - """Get key-value pairs from the file of a variant""" - for name, value in self.configuration.get_values_in_config(variant).items(): - with indent_log(): - write_output("%s: %s", name, value) - - def print_env_var_values(self) -> None: - """Get key-values pairs present as environment variables""" - write_output("%s:", "env_var") - with indent_log(): - for key, value in sorted(self.configuration.get_environ_vars()): - env_var = f"PIP_{key.upper()}" - write_output("%s=%r", env_var, value) - - def open_in_editor(self, options: Values, args: List[str]) -> None: - editor = self._determine_editor(options) - - fname = self.configuration.get_file_to_edit() - if fname is None: - raise PipError("Could not determine appropriate file.") - elif '"' in fname: - # This shouldn't happen, unless we see a username like that. - # If that happens, we'd appreciate a pull request fixing this. - raise PipError( - f'Can not open an editor for a file name containing "\n{fname}' - ) - - try: - subprocess.check_call(f'{editor} "{fname}"', shell=True) - except FileNotFoundError as e: - if not e.filename: - e.filename = editor - raise - except subprocess.CalledProcessError as e: - raise PipError(f"Editor Subprocess exited with exit code {e.returncode}") - - def _get_n_args(self, args: List[str], example: str, n: int) -> Any: - """Helper to make sure the command got the right number of arguments""" - if len(args) != n: - msg = ( - f"Got unexpected number of arguments, expected {n}. " - f'(example: "{get_prog()} config {example}")' - ) - raise PipError(msg) - - if n == 1: - return args[0] - else: - return args - - def _save_configuration(self) -> None: - # We successfully ran a modifying command. Need to save the - # configuration. - try: - self.configuration.save() - except Exception: - logger.exception( - "Unable to save configuration. Please report this as a bug." - ) - raise PipError("Internal Error.") - - def _determine_editor(self, options: Values) -> str: - if options.editor is not None: - return options.editor - elif "VISUAL" in os.environ: - return os.environ["VISUAL"] - elif "EDITOR" in os.environ: - return os.environ["EDITOR"] - else: - raise PipError("Could not determine editor to use.") diff --git a/.env/Lib/site-packages/pip/_internal/commands/debug.py b/.env/Lib/site-packages/pip/_internal/commands/debug.py deleted file mode 100644 index 567ca96..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/debug.py +++ /dev/null @@ -1,201 +0,0 @@ -import locale -import logging -import os -import sys -from optparse import Values -from types import ModuleType -from typing import Any, Dict, List, Optional - -import pip._vendor -from pip._vendor.certifi import where -from pip._vendor.packaging.version import parse as parse_version - -from pip._internal.cli import cmdoptions -from pip._internal.cli.base_command import Command -from pip._internal.cli.cmdoptions import make_target_python -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.configuration import Configuration -from pip._internal.metadata import get_environment -from pip._internal.utils.compat import open_text_resource -from pip._internal.utils.logging import indent_log -from pip._internal.utils.misc import get_pip_version - -logger = logging.getLogger(__name__) - - -def show_value(name: str, value: Any) -> None: - logger.info("%s: %s", name, value) - - -def show_sys_implementation() -> None: - logger.info("sys.implementation:") - implementation_name = sys.implementation.name - with indent_log(): - show_value("name", implementation_name) - - -def create_vendor_txt_map() -> Dict[str, str]: - with open_text_resource("pip._vendor", "vendor.txt") as f: - # Purge non version specifying lines. - # Also, remove any space prefix or suffixes (including comments). - lines = [ - line.strip().split(" ", 1)[0] for line in f.readlines() if "==" in line - ] - - # Transform into "module" -> version dict. - return dict(line.split("==", 1) for line in lines) - - -def get_module_from_module_name(module_name: str) -> Optional[ModuleType]: - # Module name can be uppercase in vendor.txt for some reason... - module_name = module_name.lower().replace("-", "_") - # PATCH: setuptools is actually only pkg_resources. - if module_name == "setuptools": - module_name = "pkg_resources" - - try: - __import__(f"pip._vendor.{module_name}", globals(), locals(), level=0) - return getattr(pip._vendor, module_name) - except ImportError: - # We allow 'truststore' to fail to import due - # to being unavailable on Python 3.9 and earlier. - if module_name == "truststore" and sys.version_info < (3, 10): - return None - raise - - -def get_vendor_version_from_module(module_name: str) -> Optional[str]: - module = get_module_from_module_name(module_name) - version = getattr(module, "__version__", None) - - if module and not version: - # Try to find version in debundled module info. - assert module.__file__ is not None - env = get_environment([os.path.dirname(module.__file__)]) - dist = env.get_distribution(module_name) - if dist: - version = str(dist.version) - - return version - - -def show_actual_vendor_versions(vendor_txt_versions: Dict[str, str]) -> None: - """Log the actual version and print extra info if there is - a conflict or if the actual version could not be imported. - """ - for module_name, expected_version in vendor_txt_versions.items(): - extra_message = "" - actual_version = get_vendor_version_from_module(module_name) - if not actual_version: - extra_message = ( - " (Unable to locate actual module version, using" - " vendor.txt specified version)" - ) - actual_version = expected_version - elif parse_version(actual_version) != parse_version(expected_version): - extra_message = ( - " (CONFLICT: vendor.txt suggests version should" - f" be {expected_version})" - ) - logger.info("%s==%s%s", module_name, actual_version, extra_message) - - -def show_vendor_versions() -> None: - logger.info("vendored library versions:") - - vendor_txt_versions = create_vendor_txt_map() - with indent_log(): - show_actual_vendor_versions(vendor_txt_versions) - - -def show_tags(options: Values) -> None: - tag_limit = 10 - - target_python = make_target_python(options) - tags = target_python.get_sorted_tags() - - # Display the target options that were explicitly provided. - formatted_target = target_python.format_given() - suffix = "" - if formatted_target: - suffix = f" (target: {formatted_target})" - - msg = f"Compatible tags: {len(tags)}{suffix}" - logger.info(msg) - - if options.verbose < 1 and len(tags) > tag_limit: - tags_limited = True - tags = tags[:tag_limit] - else: - tags_limited = False - - with indent_log(): - for tag in tags: - logger.info(str(tag)) - - if tags_limited: - msg = f"...\n[First {tag_limit} tags shown. Pass --verbose to show all.]" - logger.info(msg) - - -def ca_bundle_info(config: Configuration) -> str: - levels = {key.split(".", 1)[0] for key, _ in config.items()} - if not levels: - return "Not specified" - - levels_that_override_global = ["install", "wheel", "download"] - global_overriding_level = [ - level for level in levels if level in levels_that_override_global - ] - if not global_overriding_level: - return "global" - - if "global" in levels: - levels.remove("global") - return ", ".join(levels) - - -class DebugCommand(Command): - """ - Display debug information. - """ - - usage = """ - %prog """ - ignore_require_venv = True - - def add_options(self) -> None: - cmdoptions.add_target_python_options(self.cmd_opts) - self.parser.insert_option_group(0, self.cmd_opts) - self.parser.config.load() - - def run(self, options: Values, args: List[str]) -> int: - logger.warning( - "This command is only meant for debugging. " - "Do not use this with automation for parsing and getting these " - "details, since the output and options of this command may " - "change without notice." - ) - show_value("pip version", get_pip_version()) - show_value("sys.version", sys.version) - show_value("sys.executable", sys.executable) - show_value("sys.getdefaultencoding", sys.getdefaultencoding()) - show_value("sys.getfilesystemencoding", sys.getfilesystemencoding()) - show_value( - "locale.getpreferredencoding", - locale.getpreferredencoding(), - ) - show_value("sys.platform", sys.platform) - show_sys_implementation() - - show_value("'cert' config value", ca_bundle_info(self.parser.config)) - show_value("REQUESTS_CA_BUNDLE", os.environ.get("REQUESTS_CA_BUNDLE")) - show_value("CURL_CA_BUNDLE", os.environ.get("CURL_CA_BUNDLE")) - show_value("pip._vendor.certifi.where()", where()) - show_value("pip._vendor.DEBUNDLED", pip._vendor.DEBUNDLED) - - show_vendor_versions() - - show_tags(options) - - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/download.py b/.env/Lib/site-packages/pip/_internal/commands/download.py deleted file mode 100644 index 917bbb9..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/download.py +++ /dev/null @@ -1,146 +0,0 @@ -import logging -import os -from optparse import Values -from typing import List - -from pip._internal.cli import cmdoptions -from pip._internal.cli.cmdoptions import make_target_python -from pip._internal.cli.req_command import RequirementCommand, with_cleanup -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.operations.build.build_tracker import get_build_tracker -from pip._internal.req.req_install import check_legacy_setup_py_options -from pip._internal.utils.misc import ensure_dir, normalize_path, write_output -from pip._internal.utils.temp_dir import TempDirectory - -logger = logging.getLogger(__name__) - - -class DownloadCommand(RequirementCommand): - """ - Download packages from: - - - PyPI (and other indexes) using requirement specifiers. - - VCS project urls. - - Local project directories. - - Local or remote source archives. - - pip also supports downloading from "requirements files", which provide - an easy way to specify a whole environment to be downloaded. - """ - - usage = """ - %prog [options] [package-index-options] ... - %prog [options] -r [package-index-options] ... - %prog [options] ... - %prog [options] ... - %prog [options] ...""" - - def add_options(self) -> None: - self.cmd_opts.add_option(cmdoptions.constraints()) - self.cmd_opts.add_option(cmdoptions.requirements()) - self.cmd_opts.add_option(cmdoptions.no_deps()) - self.cmd_opts.add_option(cmdoptions.global_options()) - self.cmd_opts.add_option(cmdoptions.no_binary()) - self.cmd_opts.add_option(cmdoptions.only_binary()) - self.cmd_opts.add_option(cmdoptions.prefer_binary()) - self.cmd_opts.add_option(cmdoptions.src()) - self.cmd_opts.add_option(cmdoptions.pre()) - self.cmd_opts.add_option(cmdoptions.require_hashes()) - self.cmd_opts.add_option(cmdoptions.progress_bar()) - self.cmd_opts.add_option(cmdoptions.no_build_isolation()) - self.cmd_opts.add_option(cmdoptions.use_pep517()) - self.cmd_opts.add_option(cmdoptions.no_use_pep517()) - self.cmd_opts.add_option(cmdoptions.check_build_deps()) - self.cmd_opts.add_option(cmdoptions.ignore_requires_python()) - - self.cmd_opts.add_option( - "-d", - "--dest", - "--destination-dir", - "--destination-directory", - dest="download_dir", - metavar="dir", - default=os.curdir, - help="Download packages into .", - ) - - cmdoptions.add_target_python_options(self.cmd_opts) - - index_opts = cmdoptions.make_option_group( - cmdoptions.index_group, - self.parser, - ) - - self.parser.insert_option_group(0, index_opts) - self.parser.insert_option_group(0, self.cmd_opts) - - @with_cleanup - def run(self, options: Values, args: List[str]) -> int: - options.ignore_installed = True - # editable doesn't really make sense for `pip download`, but the bowels - # of the RequirementSet code require that property. - options.editables = [] - - cmdoptions.check_dist_restriction(options) - - options.download_dir = normalize_path(options.download_dir) - ensure_dir(options.download_dir) - - session = self.get_default_session(options) - - target_python = make_target_python(options) - finder = self._build_package_finder( - options=options, - session=session, - target_python=target_python, - ignore_requires_python=options.ignore_requires_python, - ) - - build_tracker = self.enter_context(get_build_tracker()) - - directory = TempDirectory( - delete=not options.no_clean, - kind="download", - globally_managed=True, - ) - - reqs = self.get_requirements(args, options, finder, session) - check_legacy_setup_py_options(options, reqs) - - preparer = self.make_requirement_preparer( - temp_build_dir=directory, - options=options, - build_tracker=build_tracker, - session=session, - finder=finder, - download_dir=options.download_dir, - use_user_site=False, - verbosity=self.verbosity, - ) - - resolver = self.make_resolver( - preparer=preparer, - finder=finder, - options=options, - ignore_requires_python=options.ignore_requires_python, - use_pep517=options.use_pep517, - py_version_info=options.python_version, - ) - - self.trace_basic_info(finder) - - requirement_set = resolver.resolve(reqs, check_supported_wheels=True) - - downloaded: List[str] = [] - for req in requirement_set.requirements.values(): - if req.satisfied_by is None: - assert req.name is not None - preparer.save_linked_requirement(req) - downloaded.append(req.name) - - preparer.prepare_linked_requirements_more(requirement_set.requirements.values()) - - if downloaded: - write_output("Successfully downloaded %s", " ".join(downloaded)) - - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/freeze.py b/.env/Lib/site-packages/pip/_internal/commands/freeze.py deleted file mode 100644 index 885fdfe..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/freeze.py +++ /dev/null @@ -1,109 +0,0 @@ -import sys -from optparse import Values -from typing import AbstractSet, List - -from pip._internal.cli import cmdoptions -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.operations.freeze import freeze -from pip._internal.utils.compat import stdlib_pkgs - - -def _should_suppress_build_backends() -> bool: - return sys.version_info < (3, 12) - - -def _dev_pkgs() -> AbstractSet[str]: - pkgs = {"pip"} - - if _should_suppress_build_backends(): - pkgs |= {"setuptools", "distribute", "wheel"} - - return pkgs - - -class FreezeCommand(Command): - """ - Output installed packages in requirements format. - - packages are listed in a case-insensitive sorted order. - """ - - ignore_require_venv = True - usage = """ - %prog [options]""" - log_streams = ("ext://sys.stderr", "ext://sys.stderr") - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-r", - "--requirement", - dest="requirements", - action="append", - default=[], - metavar="file", - help=( - "Use the order in the given requirements file and its " - "comments when generating output. This option can be " - "used multiple times." - ), - ) - self.cmd_opts.add_option( - "-l", - "--local", - dest="local", - action="store_true", - default=False, - help=( - "If in a virtualenv that has global access, do not output " - "globally-installed packages." - ), - ) - self.cmd_opts.add_option( - "--user", - dest="user", - action="store_true", - default=False, - help="Only output packages installed in user-site.", - ) - self.cmd_opts.add_option(cmdoptions.list_path()) - self.cmd_opts.add_option( - "--all", - dest="freeze_all", - action="store_true", - help=( - "Do not skip these packages in the output:" - " {}".format(", ".join(_dev_pkgs())) - ), - ) - self.cmd_opts.add_option( - "--exclude-editable", - dest="exclude_editable", - action="store_true", - help="Exclude editable package from output.", - ) - self.cmd_opts.add_option(cmdoptions.list_exclude()) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - skip = set(stdlib_pkgs) - if not options.freeze_all: - skip.update(_dev_pkgs()) - - if options.excludes: - skip.update(options.excludes) - - cmdoptions.check_list_path_option(options) - - for line in freeze( - requirement=options.requirements, - local_only=options.local, - user_only=options.user, - paths=options.path, - isolated=options.isolated_mode, - skip=skip, - exclude_editable=options.exclude_editable, - ): - sys.stdout.write(line + "\n") - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/hash.py b/.env/Lib/site-packages/pip/_internal/commands/hash.py deleted file mode 100644 index 042dac8..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/hash.py +++ /dev/null @@ -1,59 +0,0 @@ -import hashlib -import logging -import sys -from optparse import Values -from typing import List - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.utils.hashes import FAVORITE_HASH, STRONG_HASHES -from pip._internal.utils.misc import read_chunks, write_output - -logger = logging.getLogger(__name__) - - -class HashCommand(Command): - """ - Compute a hash of a local package archive. - - These can be used with --hash in a requirements file to do repeatable - installs. - """ - - usage = "%prog [options] ..." - ignore_require_venv = True - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-a", - "--algorithm", - dest="algorithm", - choices=STRONG_HASHES, - action="store", - default=FAVORITE_HASH, - help="The hash algorithm to use: one of {}".format( - ", ".join(STRONG_HASHES) - ), - ) - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - if not args: - self.parser.print_usage(sys.stderr) - return ERROR - - algorithm = options.algorithm - for path in args: - write_output( - "%s:\n--hash=%s:%s", path, algorithm, _hash_of_file(path, algorithm) - ) - return SUCCESS - - -def _hash_of_file(path: str, algorithm: str) -> str: - """Return the hash digest of a file.""" - with open(path, "rb") as archive: - hash = hashlib.new(algorithm) - for chunk in read_chunks(archive): - hash.update(chunk) - return hash.hexdigest() diff --git a/.env/Lib/site-packages/pip/_internal/commands/help.py b/.env/Lib/site-packages/pip/_internal/commands/help.py deleted file mode 100644 index 6206631..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/help.py +++ /dev/null @@ -1,41 +0,0 @@ -from optparse import Values -from typing import List - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.exceptions import CommandError - - -class HelpCommand(Command): - """Show help for commands""" - - usage = """ - %prog """ - ignore_require_venv = True - - def run(self, options: Values, args: List[str]) -> int: - from pip._internal.commands import ( - commands_dict, - create_command, - get_similar_commands, - ) - - try: - # 'pip help' with no args is handled by pip.__init__.parseopt() - cmd_name = args[0] # the command we need help for - except IndexError: - return SUCCESS - - if cmd_name not in commands_dict: - guess = get_similar_commands(cmd_name) - - msg = [f'unknown command "{cmd_name}"'] - if guess: - msg.append(f'maybe you meant "{guess}"') - - raise CommandError(" - ".join(msg)) - - command = create_command(cmd_name) - command.parser.print_help() - - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/index.py b/.env/Lib/site-packages/pip/_internal/commands/index.py deleted file mode 100644 index 2e2661b..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/index.py +++ /dev/null @@ -1,139 +0,0 @@ -import logging -from optparse import Values -from typing import Any, Iterable, List, Optional - -from pip._vendor.packaging.version import Version - -from pip._internal.cli import cmdoptions -from pip._internal.cli.req_command import IndexGroupCommand -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.commands.search import print_dist_installation_info -from pip._internal.exceptions import CommandError, DistributionNotFound, PipError -from pip._internal.index.collector import LinkCollector -from pip._internal.index.package_finder import PackageFinder -from pip._internal.models.selection_prefs import SelectionPreferences -from pip._internal.models.target_python import TargetPython -from pip._internal.network.session import PipSession -from pip._internal.utils.misc import write_output - -logger = logging.getLogger(__name__) - - -class IndexCommand(IndexGroupCommand): - """ - Inspect information available from package indexes. - """ - - ignore_require_venv = True - usage = """ - %prog versions - """ - - def add_options(self) -> None: - cmdoptions.add_target_python_options(self.cmd_opts) - - self.cmd_opts.add_option(cmdoptions.ignore_requires_python()) - self.cmd_opts.add_option(cmdoptions.pre()) - self.cmd_opts.add_option(cmdoptions.no_binary()) - self.cmd_opts.add_option(cmdoptions.only_binary()) - - index_opts = cmdoptions.make_option_group( - cmdoptions.index_group, - self.parser, - ) - - self.parser.insert_option_group(0, index_opts) - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - handlers = { - "versions": self.get_available_package_versions, - } - - logger.warning( - "pip index is currently an experimental command. " - "It may be removed/changed in a future release " - "without prior warning." - ) - - # Determine action - if not args or args[0] not in handlers: - logger.error( - "Need an action (%s) to perform.", - ", ".join(sorted(handlers)), - ) - return ERROR - - action = args[0] - - # Error handling happens here, not in the action-handlers. - try: - handlers[action](options, args[1:]) - except PipError as e: - logger.error(e.args[0]) - return ERROR - - return SUCCESS - - def _build_package_finder( - self, - options: Values, - session: PipSession, - target_python: Optional[TargetPython] = None, - ignore_requires_python: Optional[bool] = None, - ) -> PackageFinder: - """ - Create a package finder appropriate to the index command. - """ - link_collector = LinkCollector.create(session, options=options) - - # Pass allow_yanked=False to ignore yanked versions. - selection_prefs = SelectionPreferences( - allow_yanked=False, - allow_all_prereleases=options.pre, - ignore_requires_python=ignore_requires_python, - ) - - return PackageFinder.create( - link_collector=link_collector, - selection_prefs=selection_prefs, - target_python=target_python, - ) - - def get_available_package_versions(self, options: Values, args: List[Any]) -> None: - if len(args) != 1: - raise CommandError("You need to specify exactly one argument") - - target_python = cmdoptions.make_target_python(options) - query = args[0] - - with self._build_session(options) as session: - finder = self._build_package_finder( - options=options, - session=session, - target_python=target_python, - ignore_requires_python=options.ignore_requires_python, - ) - - versions: Iterable[Version] = ( - candidate.version for candidate in finder.find_all_candidates(query) - ) - - if not options.pre: - # Remove prereleases - versions = ( - version for version in versions if not version.is_prerelease - ) - versions = set(versions) - - if not versions: - raise DistributionNotFound( - f"No matching distribution found for {query}" - ) - - formatted_versions = [str(ver) for ver in sorted(versions, reverse=True)] - latest = formatted_versions[0] - - write_output(f"{query} ({latest})") - write_output("Available versions: {}".format(", ".join(formatted_versions))) - print_dist_installation_info(query, latest) diff --git a/.env/Lib/site-packages/pip/_internal/commands/inspect.py b/.env/Lib/site-packages/pip/_internal/commands/inspect.py deleted file mode 100644 index e810c13..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/inspect.py +++ /dev/null @@ -1,92 +0,0 @@ -import logging -from optparse import Values -from typing import Any, Dict, List - -from pip._vendor.packaging.markers import default_environment -from pip._vendor.rich import print_json - -from pip import __version__ -from pip._internal.cli import cmdoptions -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.metadata import BaseDistribution, get_environment -from pip._internal.utils.compat import stdlib_pkgs -from pip._internal.utils.urls import path_to_url - -logger = logging.getLogger(__name__) - - -class InspectCommand(Command): - """ - Inspect the content of a Python environment and produce a report in JSON format. - """ - - ignore_require_venv = True - usage = """ - %prog [options]""" - - def add_options(self) -> None: - self.cmd_opts.add_option( - "--local", - action="store_true", - default=False, - help=( - "If in a virtualenv that has global access, do not list " - "globally-installed packages." - ), - ) - self.cmd_opts.add_option( - "--user", - dest="user", - action="store_true", - default=False, - help="Only output packages installed in user-site.", - ) - self.cmd_opts.add_option(cmdoptions.list_path()) - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - cmdoptions.check_list_path_option(options) - dists = get_environment(options.path).iter_installed_distributions( - local_only=options.local, - user_only=options.user, - skip=set(stdlib_pkgs), - ) - output = { - "version": "1", - "pip_version": __version__, - "installed": [self._dist_to_dict(dist) for dist in dists], - "environment": default_environment(), - # TODO tags? scheme? - } - print_json(data=output) - return SUCCESS - - def _dist_to_dict(self, dist: BaseDistribution) -> Dict[str, Any]: - res: Dict[str, Any] = { - "metadata": dist.metadata_dict, - "metadata_location": dist.info_location, - } - # direct_url. Note that we don't have download_info (as in the installation - # report) since it is not recorded in installed metadata. - direct_url = dist.direct_url - if direct_url is not None: - res["direct_url"] = direct_url.to_dict() - else: - # Emulate direct_url for legacy editable installs. - editable_project_location = dist.editable_project_location - if editable_project_location is not None: - res["direct_url"] = { - "url": path_to_url(editable_project_location), - "dir_info": { - "editable": True, - }, - } - # installer - installer = dist.installer - if dist.installer: - res["installer"] = installer - # requested - if dist.installed_with_dist_info: - res["requested"] = dist.requested - return res diff --git a/.env/Lib/site-packages/pip/_internal/commands/install.py b/.env/Lib/site-packages/pip/_internal/commands/install.py deleted file mode 100644 index ad45a2f..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/install.py +++ /dev/null @@ -1,783 +0,0 @@ -import errno -import json -import operator -import os -import shutil -import site -from optparse import SUPPRESS_HELP, Values -from typing import List, Optional - -from pip._vendor.packaging.utils import canonicalize_name -from pip._vendor.rich import print_json - -from pip._internal.cache import WheelCache -from pip._internal.cli import cmdoptions -from pip._internal.cli.cmdoptions import make_target_python -from pip._internal.cli.req_command import ( - RequirementCommand, - with_cleanup, -) -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.exceptions import CommandError, InstallationError -from pip._internal.locations import get_scheme -from pip._internal.metadata import get_environment -from pip._internal.models.installation_report import InstallationReport -from pip._internal.operations.build.build_tracker import get_build_tracker -from pip._internal.operations.check import ConflictDetails, check_install_conflicts -from pip._internal.req import install_given_reqs -from pip._internal.req.req_install import ( - InstallRequirement, - check_legacy_setup_py_options, -) -from pip._internal.utils.compat import WINDOWS -from pip._internal.utils.filesystem import test_writable_dir -from pip._internal.utils.logging import getLogger -from pip._internal.utils.misc import ( - check_externally_managed, - ensure_dir, - get_pip_version, - protect_pip_from_modification_on_windows, - warn_if_run_as_root, - write_output, -) -from pip._internal.utils.temp_dir import TempDirectory -from pip._internal.utils.virtualenv import ( - running_under_virtualenv, - virtualenv_no_global, -) -from pip._internal.wheel_builder import build, should_build_for_install_command - -logger = getLogger(__name__) - - -class InstallCommand(RequirementCommand): - """ - Install packages from: - - - PyPI (and other indexes) using requirement specifiers. - - VCS project urls. - - Local project directories. - - Local or remote source archives. - - pip also supports installing from "requirements files", which provide - an easy way to specify a whole environment to be installed. - """ - - usage = """ - %prog [options] [package-index-options] ... - %prog [options] -r [package-index-options] ... - %prog [options] [-e] ... - %prog [options] [-e] ... - %prog [options] ...""" - - def add_options(self) -> None: - self.cmd_opts.add_option(cmdoptions.requirements()) - self.cmd_opts.add_option(cmdoptions.constraints()) - self.cmd_opts.add_option(cmdoptions.no_deps()) - self.cmd_opts.add_option(cmdoptions.pre()) - - self.cmd_opts.add_option(cmdoptions.editable()) - self.cmd_opts.add_option( - "--dry-run", - action="store_true", - dest="dry_run", - default=False, - help=( - "Don't actually install anything, just print what would be. " - "Can be used in combination with --ignore-installed " - "to 'resolve' the requirements." - ), - ) - self.cmd_opts.add_option( - "-t", - "--target", - dest="target_dir", - metavar="dir", - default=None, - help=( - "Install packages into . " - "By default this will not replace existing files/folders in " - ". Use --upgrade to replace existing packages in " - "with new versions." - ), - ) - cmdoptions.add_target_python_options(self.cmd_opts) - - self.cmd_opts.add_option( - "--user", - dest="use_user_site", - action="store_true", - help=( - "Install to the Python user install directory for your " - "platform. Typically ~/.local/, or %APPDATA%\\Python on " - "Windows. (See the Python documentation for site.USER_BASE " - "for full details.)" - ), - ) - self.cmd_opts.add_option( - "--no-user", - dest="use_user_site", - action="store_false", - help=SUPPRESS_HELP, - ) - self.cmd_opts.add_option( - "--root", - dest="root_path", - metavar="dir", - default=None, - help="Install everything relative to this alternate root directory.", - ) - self.cmd_opts.add_option( - "--prefix", - dest="prefix_path", - metavar="dir", - default=None, - help=( - "Installation prefix where lib, bin and other top-level " - "folders are placed. Note that the resulting installation may " - "contain scripts and other resources which reference the " - "Python interpreter of pip, and not that of ``--prefix``. " - "See also the ``--python`` option if the intention is to " - "install packages into another (possibly pip-free) " - "environment." - ), - ) - - self.cmd_opts.add_option(cmdoptions.src()) - - self.cmd_opts.add_option( - "-U", - "--upgrade", - dest="upgrade", - action="store_true", - help=( - "Upgrade all specified packages to the newest available " - "version. The handling of dependencies depends on the " - "upgrade-strategy used." - ), - ) - - self.cmd_opts.add_option( - "--upgrade-strategy", - dest="upgrade_strategy", - default="only-if-needed", - choices=["only-if-needed", "eager"], - help=( - "Determines how dependency upgrading should be handled " - "[default: %default]. " - '"eager" - dependencies are upgraded regardless of ' - "whether the currently installed version satisfies the " - "requirements of the upgraded package(s). " - '"only-if-needed" - are upgraded only when they do not ' - "satisfy the requirements of the upgraded package(s)." - ), - ) - - self.cmd_opts.add_option( - "--force-reinstall", - dest="force_reinstall", - action="store_true", - help="Reinstall all packages even if they are already up-to-date.", - ) - - self.cmd_opts.add_option( - "-I", - "--ignore-installed", - dest="ignore_installed", - action="store_true", - help=( - "Ignore the installed packages, overwriting them. " - "This can break your system if the existing package " - "is of a different version or was installed " - "with a different package manager!" - ), - ) - - self.cmd_opts.add_option(cmdoptions.ignore_requires_python()) - self.cmd_opts.add_option(cmdoptions.no_build_isolation()) - self.cmd_opts.add_option(cmdoptions.use_pep517()) - self.cmd_opts.add_option(cmdoptions.no_use_pep517()) - self.cmd_opts.add_option(cmdoptions.check_build_deps()) - self.cmd_opts.add_option(cmdoptions.override_externally_managed()) - - self.cmd_opts.add_option(cmdoptions.config_settings()) - self.cmd_opts.add_option(cmdoptions.global_options()) - - self.cmd_opts.add_option( - "--compile", - action="store_true", - dest="compile", - default=True, - help="Compile Python source files to bytecode", - ) - - self.cmd_opts.add_option( - "--no-compile", - action="store_false", - dest="compile", - help="Do not compile Python source files to bytecode", - ) - - self.cmd_opts.add_option( - "--no-warn-script-location", - action="store_false", - dest="warn_script_location", - default=True, - help="Do not warn when installing scripts outside PATH", - ) - self.cmd_opts.add_option( - "--no-warn-conflicts", - action="store_false", - dest="warn_about_conflicts", - default=True, - help="Do not warn about broken dependencies", - ) - self.cmd_opts.add_option(cmdoptions.no_binary()) - self.cmd_opts.add_option(cmdoptions.only_binary()) - self.cmd_opts.add_option(cmdoptions.prefer_binary()) - self.cmd_opts.add_option(cmdoptions.require_hashes()) - self.cmd_opts.add_option(cmdoptions.progress_bar()) - self.cmd_opts.add_option(cmdoptions.root_user_action()) - - index_opts = cmdoptions.make_option_group( - cmdoptions.index_group, - self.parser, - ) - - self.parser.insert_option_group(0, index_opts) - self.parser.insert_option_group(0, self.cmd_opts) - - self.cmd_opts.add_option( - "--report", - dest="json_report_file", - metavar="file", - default=None, - help=( - "Generate a JSON file describing what pip did to install " - "the provided requirements. " - "Can be used in combination with --dry-run and --ignore-installed " - "to 'resolve' the requirements. " - "When - is used as file name it writes to stdout. " - "When writing to stdout, please combine with the --quiet option " - "to avoid mixing pip logging output with JSON output." - ), - ) - - @with_cleanup - def run(self, options: Values, args: List[str]) -> int: - if options.use_user_site and options.target_dir is not None: - raise CommandError("Can not combine '--user' and '--target'") - - # Check whether the environment we're installing into is externally - # managed, as specified in PEP 668. Specifying --root, --target, or - # --prefix disables the check, since there's no reliable way to locate - # the EXTERNALLY-MANAGED file for those cases. An exception is also - # made specifically for "--dry-run --report" for convenience. - installing_into_current_environment = ( - not (options.dry_run and options.json_report_file) - and options.root_path is None - and options.target_dir is None - and options.prefix_path is None - ) - if ( - installing_into_current_environment - and not options.override_externally_managed - ): - check_externally_managed() - - upgrade_strategy = "to-satisfy-only" - if options.upgrade: - upgrade_strategy = options.upgrade_strategy - - cmdoptions.check_dist_restriction(options, check_target=True) - - logger.verbose("Using %s", get_pip_version()) - options.use_user_site = decide_user_install( - options.use_user_site, - prefix_path=options.prefix_path, - target_dir=options.target_dir, - root_path=options.root_path, - isolated_mode=options.isolated_mode, - ) - - target_temp_dir: Optional[TempDirectory] = None - target_temp_dir_path: Optional[str] = None - if options.target_dir: - options.ignore_installed = True - options.target_dir = os.path.abspath(options.target_dir) - if ( - # fmt: off - os.path.exists(options.target_dir) and - not os.path.isdir(options.target_dir) - # fmt: on - ): - raise CommandError( - "Target path exists but is not a directory, will not continue." - ) - - # Create a target directory for using with the target option - target_temp_dir = TempDirectory(kind="target") - target_temp_dir_path = target_temp_dir.path - self.enter_context(target_temp_dir) - - global_options = options.global_options or [] - - session = self.get_default_session(options) - - target_python = make_target_python(options) - finder = self._build_package_finder( - options=options, - session=session, - target_python=target_python, - ignore_requires_python=options.ignore_requires_python, - ) - build_tracker = self.enter_context(get_build_tracker()) - - directory = TempDirectory( - delete=not options.no_clean, - kind="install", - globally_managed=True, - ) - - try: - reqs = self.get_requirements(args, options, finder, session) - check_legacy_setup_py_options(options, reqs) - - wheel_cache = WheelCache(options.cache_dir) - - # Only when installing is it permitted to use PEP 660. - # In other circumstances (pip wheel, pip download) we generate - # regular (i.e. non editable) metadata and wheels. - for req in reqs: - req.permit_editable_wheels = True - - preparer = self.make_requirement_preparer( - temp_build_dir=directory, - options=options, - build_tracker=build_tracker, - session=session, - finder=finder, - use_user_site=options.use_user_site, - verbosity=self.verbosity, - ) - resolver = self.make_resolver( - preparer=preparer, - finder=finder, - options=options, - wheel_cache=wheel_cache, - use_user_site=options.use_user_site, - ignore_installed=options.ignore_installed, - ignore_requires_python=options.ignore_requires_python, - force_reinstall=options.force_reinstall, - upgrade_strategy=upgrade_strategy, - use_pep517=options.use_pep517, - py_version_info=options.python_version, - ) - - self.trace_basic_info(finder) - - requirement_set = resolver.resolve( - reqs, check_supported_wheels=not options.target_dir - ) - - if options.json_report_file: - report = InstallationReport(requirement_set.requirements_to_install) - if options.json_report_file == "-": - print_json(data=report.to_dict()) - else: - with open(options.json_report_file, "w", encoding="utf-8") as f: - json.dump(report.to_dict(), f, indent=2, ensure_ascii=False) - - if options.dry_run: - would_install_items = sorted( - (r.metadata["name"], r.metadata["version"]) - for r in requirement_set.requirements_to_install - ) - if would_install_items: - write_output( - "Would install %s", - " ".join("-".join(item) for item in would_install_items), - ) - return SUCCESS - - try: - pip_req = requirement_set.get_requirement("pip") - except KeyError: - modifying_pip = False - else: - # If we're not replacing an already installed pip, - # we're not modifying it. - modifying_pip = pip_req.satisfied_by is None - if modifying_pip: - # Eagerly import this module to avoid crashes. Otherwise, this - # module would be imported *after* pip was replaced, resulting in - # crashes if the new self_outdated_check module was incompatible - # with the rest of pip that's already imported. - import pip._internal.self_outdated_check # noqa: F401 - protect_pip_from_modification_on_windows(modifying_pip=modifying_pip) - - reqs_to_build = [ - r - for r in requirement_set.requirements.values() - if should_build_for_install_command(r) - ] - - _, build_failures = build( - reqs_to_build, - wheel_cache=wheel_cache, - verify=True, - build_options=[], - global_options=global_options, - ) - - if build_failures: - raise InstallationError( - "ERROR: Failed to build installable wheels for some " - "pyproject.toml based projects ({})".format( - ", ".join(r.name for r in build_failures) # type: ignore - ) - ) - - to_install = resolver.get_installation_order(requirement_set) - - # Check for conflicts in the package set we're installing. - conflicts: Optional[ConflictDetails] = None - should_warn_about_conflicts = ( - not options.ignore_dependencies and options.warn_about_conflicts - ) - if should_warn_about_conflicts: - conflicts = self._determine_conflicts(to_install) - - # Don't warn about script install locations if - # --target or --prefix has been specified - warn_script_location = options.warn_script_location - if options.target_dir or options.prefix_path: - warn_script_location = False - - installed = install_given_reqs( - to_install, - global_options, - root=options.root_path, - home=target_temp_dir_path, - prefix=options.prefix_path, - warn_script_location=warn_script_location, - use_user_site=options.use_user_site, - pycompile=options.compile, - ) - - lib_locations = get_lib_location_guesses( - user=options.use_user_site, - home=target_temp_dir_path, - root=options.root_path, - prefix=options.prefix_path, - isolated=options.isolated_mode, - ) - env = get_environment(lib_locations) - - # Display a summary of installed packages, with extra care to - # display a package name as it was requested by the user. - installed.sort(key=operator.attrgetter("name")) - summary = [] - installed_versions = {} - for distribution in env.iter_all_distributions(): - installed_versions[distribution.canonical_name] = distribution.version - for package in installed: - display_name = package.name - version = installed_versions.get(canonicalize_name(display_name), None) - if version: - text = f"{display_name}-{version}" - else: - text = display_name - summary.append(text) - - if conflicts is not None: - self._warn_about_conflicts( - conflicts, - resolver_variant=self.determine_resolver_variant(options), - ) - - installed_desc = " ".join(summary) - if installed_desc: - write_output( - "Successfully installed %s", - installed_desc, - ) - except OSError as error: - show_traceback = self.verbosity >= 1 - - message = create_os_error_message( - error, - show_traceback, - options.use_user_site, - ) - logger.error(message, exc_info=show_traceback) - - return ERROR - - if options.target_dir: - assert target_temp_dir - self._handle_target_dir( - options.target_dir, target_temp_dir, options.upgrade - ) - if options.root_user_action == "warn": - warn_if_run_as_root() - return SUCCESS - - def _handle_target_dir( - self, target_dir: str, target_temp_dir: TempDirectory, upgrade: bool - ) -> None: - ensure_dir(target_dir) - - # Checking both purelib and platlib directories for installed - # packages to be moved to target directory - lib_dir_list = [] - - # Checking both purelib and platlib directories for installed - # packages to be moved to target directory - scheme = get_scheme("", home=target_temp_dir.path) - purelib_dir = scheme.purelib - platlib_dir = scheme.platlib - data_dir = scheme.data - - if os.path.exists(purelib_dir): - lib_dir_list.append(purelib_dir) - if os.path.exists(platlib_dir) and platlib_dir != purelib_dir: - lib_dir_list.append(platlib_dir) - if os.path.exists(data_dir): - lib_dir_list.append(data_dir) - - for lib_dir in lib_dir_list: - for item in os.listdir(lib_dir): - if lib_dir == data_dir: - ddir = os.path.join(data_dir, item) - if any(s.startswith(ddir) for s in lib_dir_list[:-1]): - continue - target_item_dir = os.path.join(target_dir, item) - if os.path.exists(target_item_dir): - if not upgrade: - logger.warning( - "Target directory %s already exists. Specify " - "--upgrade to force replacement.", - target_item_dir, - ) - continue - if os.path.islink(target_item_dir): - logger.warning( - "Target directory %s already exists and is " - "a link. pip will not automatically replace " - "links, please remove if replacement is " - "desired.", - target_item_dir, - ) - continue - if os.path.isdir(target_item_dir): - shutil.rmtree(target_item_dir) - else: - os.remove(target_item_dir) - - shutil.move(os.path.join(lib_dir, item), target_item_dir) - - def _determine_conflicts( - self, to_install: List[InstallRequirement] - ) -> Optional[ConflictDetails]: - try: - return check_install_conflicts(to_install) - except Exception: - logger.exception( - "Error while checking for conflicts. Please file an issue on " - "pip's issue tracker: https://github.com/pypa/pip/issues/new" - ) - return None - - def _warn_about_conflicts( - self, conflict_details: ConflictDetails, resolver_variant: str - ) -> None: - package_set, (missing, conflicting) = conflict_details - if not missing and not conflicting: - return - - parts: List[str] = [] - if resolver_variant == "legacy": - parts.append( - "pip's legacy dependency resolver does not consider dependency " - "conflicts when selecting packages. This behaviour is the " - "source of the following dependency conflicts." - ) - else: - assert resolver_variant == "resolvelib" - parts.append( - "pip's dependency resolver does not currently take into account " - "all the packages that are installed. This behaviour is the " - "source of the following dependency conflicts." - ) - - # NOTE: There is some duplication here, with commands/check.py - for project_name in missing: - version = package_set[project_name][0] - for dependency in missing[project_name]: - message = ( - f"{project_name} {version} requires {dependency[1]}, " - "which is not installed." - ) - parts.append(message) - - for project_name in conflicting: - version = package_set[project_name][0] - for dep_name, dep_version, req in conflicting[project_name]: - message = ( - "{name} {version} requires {requirement}, but {you} have " - "{dep_name} {dep_version} which is incompatible." - ).format( - name=project_name, - version=version, - requirement=req, - dep_name=dep_name, - dep_version=dep_version, - you=("you" if resolver_variant == "resolvelib" else "you'll"), - ) - parts.append(message) - - logger.critical("\n".join(parts)) - - -def get_lib_location_guesses( - user: bool = False, - home: Optional[str] = None, - root: Optional[str] = None, - isolated: bool = False, - prefix: Optional[str] = None, -) -> List[str]: - scheme = get_scheme( - "", - user=user, - home=home, - root=root, - isolated=isolated, - prefix=prefix, - ) - return [scheme.purelib, scheme.platlib] - - -def site_packages_writable(root: Optional[str], isolated: bool) -> bool: - return all( - test_writable_dir(d) - for d in set(get_lib_location_guesses(root=root, isolated=isolated)) - ) - - -def decide_user_install( - use_user_site: Optional[bool], - prefix_path: Optional[str] = None, - target_dir: Optional[str] = None, - root_path: Optional[str] = None, - isolated_mode: bool = False, -) -> bool: - """Determine whether to do a user install based on the input options. - - If use_user_site is False, no additional checks are done. - If use_user_site is True, it is checked for compatibility with other - options. - If use_user_site is None, the default behaviour depends on the environment, - which is provided by the other arguments. - """ - # In some cases (config from tox), use_user_site can be set to an integer - # rather than a bool, which 'use_user_site is False' wouldn't catch. - if (use_user_site is not None) and (not use_user_site): - logger.debug("Non-user install by explicit request") - return False - - if use_user_site: - if prefix_path: - raise CommandError( - "Can not combine '--user' and '--prefix' as they imply " - "different installation locations" - ) - if virtualenv_no_global(): - raise InstallationError( - "Can not perform a '--user' install. User site-packages " - "are not visible in this virtualenv." - ) - logger.debug("User install by explicit request") - return True - - # If we are here, user installs have not been explicitly requested/avoided - assert use_user_site is None - - # user install incompatible with --prefix/--target - if prefix_path or target_dir: - logger.debug("Non-user install due to --prefix or --target option") - return False - - # If user installs are not enabled, choose a non-user install - if not site.ENABLE_USER_SITE: - logger.debug("Non-user install because user site-packages disabled") - return False - - # If we have permission for a non-user install, do that, - # otherwise do a user install. - if site_packages_writable(root=root_path, isolated=isolated_mode): - logger.debug("Non-user install because site-packages writeable") - return False - - logger.info( - "Defaulting to user installation because normal site-packages " - "is not writeable" - ) - return True - - -def create_os_error_message( - error: OSError, show_traceback: bool, using_user_site: bool -) -> str: - """Format an error message for an OSError - - It may occur anytime during the execution of the install command. - """ - parts = [] - - # Mention the error if we are not going to show a traceback - parts.append("Could not install packages due to an OSError") - if not show_traceback: - parts.append(": ") - parts.append(str(error)) - else: - parts.append(".") - - # Spilt the error indication from a helper message (if any) - parts[-1] += "\n" - - # Suggest useful actions to the user: - # (1) using user site-packages or (2) verifying the permissions - if error.errno == errno.EACCES: - user_option_part = "Consider using the `--user` option" - permissions_part = "Check the permissions" - - if not running_under_virtualenv() and not using_user_site: - parts.extend( - [ - user_option_part, - " or ", - permissions_part.lower(), - ] - ) - else: - parts.append(permissions_part) - parts.append(".\n") - - # Suggest the user to enable Long Paths if path length is - # more than 260 - if ( - WINDOWS - and error.errno == errno.ENOENT - and error.filename - and len(error.filename) > 260 - ): - parts.append( - "HINT: This error might have occurred since " - "this system does not have Windows Long Path " - "support enabled. You can find information on " - "how to enable this at " - "https://pip.pypa.io/warnings/enable-long-paths\n" - ) - - return "".join(parts).strip() + "\n" diff --git a/.env/Lib/site-packages/pip/_internal/commands/list.py b/.env/Lib/site-packages/pip/_internal/commands/list.py deleted file mode 100644 index 82fc46a..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/list.py +++ /dev/null @@ -1,375 +0,0 @@ -import json -import logging -from optparse import Values -from typing import TYPE_CHECKING, Generator, List, Optional, Sequence, Tuple, cast - -from pip._vendor.packaging.utils import canonicalize_name -from pip._vendor.packaging.version import Version - -from pip._internal.cli import cmdoptions -from pip._internal.cli.index_command import IndexGroupCommand -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.exceptions import CommandError -from pip._internal.metadata import BaseDistribution, get_environment -from pip._internal.models.selection_prefs import SelectionPreferences -from pip._internal.utils.compat import stdlib_pkgs -from pip._internal.utils.misc import tabulate, write_output - -if TYPE_CHECKING: - from pip._internal.index.package_finder import PackageFinder - from pip._internal.network.session import PipSession - - class _DistWithLatestInfo(BaseDistribution): - """Give the distribution object a couple of extra fields. - - These will be populated during ``get_outdated()``. This is dirty but - makes the rest of the code much cleaner. - """ - - latest_version: Version - latest_filetype: str - - _ProcessedDists = Sequence[_DistWithLatestInfo] - - -logger = logging.getLogger(__name__) - - -class ListCommand(IndexGroupCommand): - """ - List installed packages, including editables. - - Packages are listed in a case-insensitive sorted order. - """ - - ignore_require_venv = True - usage = """ - %prog [options]""" - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-o", - "--outdated", - action="store_true", - default=False, - help="List outdated packages", - ) - self.cmd_opts.add_option( - "-u", - "--uptodate", - action="store_true", - default=False, - help="List uptodate packages", - ) - self.cmd_opts.add_option( - "-e", - "--editable", - action="store_true", - default=False, - help="List editable projects.", - ) - self.cmd_opts.add_option( - "-l", - "--local", - action="store_true", - default=False, - help=( - "If in a virtualenv that has global access, do not list " - "globally-installed packages." - ), - ) - self.cmd_opts.add_option( - "--user", - dest="user", - action="store_true", - default=False, - help="Only output packages installed in user-site.", - ) - self.cmd_opts.add_option(cmdoptions.list_path()) - self.cmd_opts.add_option( - "--pre", - action="store_true", - default=False, - help=( - "Include pre-release and development versions. By default, " - "pip only finds stable versions." - ), - ) - - self.cmd_opts.add_option( - "--format", - action="store", - dest="list_format", - default="columns", - choices=("columns", "freeze", "json"), - help=( - "Select the output format among: columns (default), freeze, or json. " - "The 'freeze' format cannot be used with the --outdated option." - ), - ) - - self.cmd_opts.add_option( - "--not-required", - action="store_true", - dest="not_required", - help="List packages that are not dependencies of installed packages.", - ) - - self.cmd_opts.add_option( - "--exclude-editable", - action="store_false", - dest="include_editable", - help="Exclude editable package from output.", - ) - self.cmd_opts.add_option( - "--include-editable", - action="store_true", - dest="include_editable", - help="Include editable package from output.", - default=True, - ) - self.cmd_opts.add_option(cmdoptions.list_exclude()) - index_opts = cmdoptions.make_option_group(cmdoptions.index_group, self.parser) - - self.parser.insert_option_group(0, index_opts) - self.parser.insert_option_group(0, self.cmd_opts) - - def handle_pip_version_check(self, options: Values) -> None: - if options.outdated or options.uptodate: - super().handle_pip_version_check(options) - - def _build_package_finder( - self, options: Values, session: "PipSession" - ) -> "PackageFinder": - """ - Create a package finder appropriate to this list command. - """ - # Lazy import the heavy index modules as most list invocations won't need 'em. - from pip._internal.index.collector import LinkCollector - from pip._internal.index.package_finder import PackageFinder - - link_collector = LinkCollector.create(session, options=options) - - # Pass allow_yanked=False to ignore yanked versions. - selection_prefs = SelectionPreferences( - allow_yanked=False, - allow_all_prereleases=options.pre, - ) - - return PackageFinder.create( - link_collector=link_collector, - selection_prefs=selection_prefs, - ) - - def run(self, options: Values, args: List[str]) -> int: - if options.outdated and options.uptodate: - raise CommandError("Options --outdated and --uptodate cannot be combined.") - - if options.outdated and options.list_format == "freeze": - raise CommandError( - "List format 'freeze' cannot be used with the --outdated option." - ) - - cmdoptions.check_list_path_option(options) - - skip = set(stdlib_pkgs) - if options.excludes: - skip.update(canonicalize_name(n) for n in options.excludes) - - packages: "_ProcessedDists" = [ - cast("_DistWithLatestInfo", d) - for d in get_environment(options.path).iter_installed_distributions( - local_only=options.local, - user_only=options.user, - editables_only=options.editable, - include_editables=options.include_editable, - skip=skip, - ) - ] - - # get_not_required must be called firstly in order to find and - # filter out all dependencies correctly. Otherwise a package - # can't be identified as requirement because some parent packages - # could be filtered out before. - if options.not_required: - packages = self.get_not_required(packages, options) - - if options.outdated: - packages = self.get_outdated(packages, options) - elif options.uptodate: - packages = self.get_uptodate(packages, options) - - self.output_package_listing(packages, options) - return SUCCESS - - def get_outdated( - self, packages: "_ProcessedDists", options: Values - ) -> "_ProcessedDists": - return [ - dist - for dist in self.iter_packages_latest_infos(packages, options) - if dist.latest_version > dist.version - ] - - def get_uptodate( - self, packages: "_ProcessedDists", options: Values - ) -> "_ProcessedDists": - return [ - dist - for dist in self.iter_packages_latest_infos(packages, options) - if dist.latest_version == dist.version - ] - - def get_not_required( - self, packages: "_ProcessedDists", options: Values - ) -> "_ProcessedDists": - dep_keys = { - canonicalize_name(dep.name) - for dist in packages - for dep in (dist.iter_dependencies() or ()) - } - - # Create a set to remove duplicate packages, and cast it to a list - # to keep the return type consistent with get_outdated and - # get_uptodate - return list({pkg for pkg in packages if pkg.canonical_name not in dep_keys}) - - def iter_packages_latest_infos( - self, packages: "_ProcessedDists", options: Values - ) -> Generator["_DistWithLatestInfo", None, None]: - with self._build_session(options) as session: - finder = self._build_package_finder(options, session) - - def latest_info( - dist: "_DistWithLatestInfo", - ) -> Optional["_DistWithLatestInfo"]: - all_candidates = finder.find_all_candidates(dist.canonical_name) - if not options.pre: - # Remove prereleases - all_candidates = [ - candidate - for candidate in all_candidates - if not candidate.version.is_prerelease - ] - - evaluator = finder.make_candidate_evaluator( - project_name=dist.canonical_name, - ) - best_candidate = evaluator.sort_best_candidate(all_candidates) - if best_candidate is None: - return None - - remote_version = best_candidate.version - if best_candidate.link.is_wheel: - typ = "wheel" - else: - typ = "sdist" - dist.latest_version = remote_version - dist.latest_filetype = typ - return dist - - for dist in map(latest_info, packages): - if dist is not None: - yield dist - - def output_package_listing( - self, packages: "_ProcessedDists", options: Values - ) -> None: - packages = sorted( - packages, - key=lambda dist: dist.canonical_name, - ) - if options.list_format == "columns" and packages: - data, header = format_for_columns(packages, options) - self.output_package_listing_columns(data, header) - elif options.list_format == "freeze": - for dist in packages: - if options.verbose >= 1: - write_output( - "%s==%s (%s)", dist.raw_name, dist.version, dist.location - ) - else: - write_output("%s==%s", dist.raw_name, dist.version) - elif options.list_format == "json": - write_output(format_for_json(packages, options)) - - def output_package_listing_columns( - self, data: List[List[str]], header: List[str] - ) -> None: - # insert the header first: we need to know the size of column names - if len(data) > 0: - data.insert(0, header) - - pkg_strings, sizes = tabulate(data) - - # Create and add a separator. - if len(data) > 0: - pkg_strings.insert(1, " ".join("-" * x for x in sizes)) - - for val in pkg_strings: - write_output(val) - - -def format_for_columns( - pkgs: "_ProcessedDists", options: Values -) -> Tuple[List[List[str]], List[str]]: - """ - Convert the package data into something usable - by output_package_listing_columns. - """ - header = ["Package", "Version"] - - running_outdated = options.outdated - if running_outdated: - header.extend(["Latest", "Type"]) - - has_editables = any(x.editable for x in pkgs) - if has_editables: - header.append("Editable project location") - - if options.verbose >= 1: - header.append("Location") - if options.verbose >= 1: - header.append("Installer") - - data = [] - for proj in pkgs: - # if we're working on the 'outdated' list, separate out the - # latest_version and type - row = [proj.raw_name, proj.raw_version] - - if running_outdated: - row.append(str(proj.latest_version)) - row.append(proj.latest_filetype) - - if has_editables: - row.append(proj.editable_project_location or "") - - if options.verbose >= 1: - row.append(proj.location or "") - if options.verbose >= 1: - row.append(proj.installer) - - data.append(row) - - return data, header - - -def format_for_json(packages: "_ProcessedDists", options: Values) -> str: - data = [] - for dist in packages: - info = { - "name": dist.raw_name, - "version": str(dist.version), - } - if options.verbose >= 1: - info["location"] = dist.location or "" - info["installer"] = dist.installer - if options.outdated: - info["latest_version"] = str(dist.latest_version) - info["latest_filetype"] = dist.latest_filetype - editable_project_location = dist.editable_project_location - if editable_project_location: - info["editable_project_location"] = editable_project_location - data.append(info) - return json.dumps(data) diff --git a/.env/Lib/site-packages/pip/_internal/commands/search.py b/.env/Lib/site-packages/pip/_internal/commands/search.py deleted file mode 100644 index e0d329d..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/search.py +++ /dev/null @@ -1,172 +0,0 @@ -import logging -import shutil -import sys -import textwrap -import xmlrpc.client -from collections import OrderedDict -from optparse import Values -from typing import TYPE_CHECKING, Dict, List, Optional, TypedDict - -from pip._vendor.packaging.version import parse as parse_version - -from pip._internal.cli.base_command import Command -from pip._internal.cli.req_command import SessionCommandMixin -from pip._internal.cli.status_codes import NO_MATCHES_FOUND, SUCCESS -from pip._internal.exceptions import CommandError -from pip._internal.metadata import get_default_environment -from pip._internal.models.index import PyPI -from pip._internal.network.xmlrpc import PipXmlrpcTransport -from pip._internal.utils.logging import indent_log -from pip._internal.utils.misc import write_output - -if TYPE_CHECKING: - - class TransformedHit(TypedDict): - name: str - summary: str - versions: List[str] - - -logger = logging.getLogger(__name__) - - -class SearchCommand(Command, SessionCommandMixin): - """Search for PyPI packages whose name or summary contains .""" - - usage = """ - %prog [options] """ - ignore_require_venv = True - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-i", - "--index", - dest="index", - metavar="URL", - default=PyPI.pypi_url, - help="Base URL of Python Package Index (default %default)", - ) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - if not args: - raise CommandError("Missing required argument (search query).") - query = args - pypi_hits = self.search(query, options) - hits = transform_hits(pypi_hits) - - terminal_width = None - if sys.stdout.isatty(): - terminal_width = shutil.get_terminal_size()[0] - - print_results(hits, terminal_width=terminal_width) - if pypi_hits: - return SUCCESS - return NO_MATCHES_FOUND - - def search(self, query: List[str], options: Values) -> List[Dict[str, str]]: - index_url = options.index - - session = self.get_default_session(options) - - transport = PipXmlrpcTransport(index_url, session) - pypi = xmlrpc.client.ServerProxy(index_url, transport) - try: - hits = pypi.search({"name": query, "summary": query}, "or") - except xmlrpc.client.Fault as fault: - message = ( - f"XMLRPC request failed [code: {fault.faultCode}]\n{fault.faultString}" - ) - raise CommandError(message) - assert isinstance(hits, list) - return hits - - -def transform_hits(hits: List[Dict[str, str]]) -> List["TransformedHit"]: - """ - The list from pypi is really a list of versions. We want a list of - packages with the list of versions stored inline. This converts the - list from pypi into one we can use. - """ - packages: Dict[str, "TransformedHit"] = OrderedDict() - for hit in hits: - name = hit["name"] - summary = hit["summary"] - version = hit["version"] - - if name not in packages.keys(): - packages[name] = { - "name": name, - "summary": summary, - "versions": [version], - } - else: - packages[name]["versions"].append(version) - - # if this is the highest version, replace summary and score - if version == highest_version(packages[name]["versions"]): - packages[name]["summary"] = summary - - return list(packages.values()) - - -def print_dist_installation_info(name: str, latest: str) -> None: - env = get_default_environment() - dist = env.get_distribution(name) - if dist is not None: - with indent_log(): - if dist.version == latest: - write_output("INSTALLED: %s (latest)", dist.version) - else: - write_output("INSTALLED: %s", dist.version) - if parse_version(latest).pre: - write_output( - "LATEST: %s (pre-release; install" - " with `pip install --pre`)", - latest, - ) - else: - write_output("LATEST: %s", latest) - - -def print_results( - hits: List["TransformedHit"], - name_column_width: Optional[int] = None, - terminal_width: Optional[int] = None, -) -> None: - if not hits: - return - if name_column_width is None: - name_column_width = ( - max( - [ - len(hit["name"]) + len(highest_version(hit.get("versions", ["-"]))) - for hit in hits - ] - ) - + 4 - ) - - for hit in hits: - name = hit["name"] - summary = hit["summary"] or "" - latest = highest_version(hit.get("versions", ["-"])) - if terminal_width is not None: - target_width = terminal_width - name_column_width - 5 - if target_width > 10: - # wrap and indent summary to fit terminal - summary_lines = textwrap.wrap(summary, target_width) - summary = ("\n" + " " * (name_column_width + 3)).join(summary_lines) - - name_latest = f"{name} ({latest})" - line = f"{name_latest:{name_column_width}} - {summary}" - try: - write_output(line) - print_dist_installation_info(name, latest) - except UnicodeEncodeError: - pass - - -def highest_version(versions: List[str]) -> str: - return max(versions, key=parse_version) diff --git a/.env/Lib/site-packages/pip/_internal/commands/show.py b/.env/Lib/site-packages/pip/_internal/commands/show.py deleted file mode 100644 index c54d548..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/show.py +++ /dev/null @@ -1,217 +0,0 @@ -import logging -from optparse import Values -from typing import Generator, Iterable, Iterator, List, NamedTuple, Optional - -from pip._vendor.packaging.requirements import InvalidRequirement -from pip._vendor.packaging.utils import canonicalize_name - -from pip._internal.cli.base_command import Command -from pip._internal.cli.status_codes import ERROR, SUCCESS -from pip._internal.metadata import BaseDistribution, get_default_environment -from pip._internal.utils.misc import write_output - -logger = logging.getLogger(__name__) - - -class ShowCommand(Command): - """ - Show information about one or more installed packages. - - The output is in RFC-compliant mail header format. - """ - - usage = """ - %prog [options] ...""" - ignore_require_venv = True - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-f", - "--files", - dest="files", - action="store_true", - default=False, - help="Show the full list of installed files for each package.", - ) - - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - if not args: - logger.warning("ERROR: Please provide a package name or names.") - return ERROR - query = args - - results = search_packages_info(query) - if not print_results( - results, list_files=options.files, verbose=options.verbose - ): - return ERROR - return SUCCESS - - -class _PackageInfo(NamedTuple): - name: str - version: str - location: str - editable_project_location: Optional[str] - requires: List[str] - required_by: List[str] - installer: str - metadata_version: str - classifiers: List[str] - summary: str - homepage: str - project_urls: List[str] - author: str - author_email: str - license: str - entry_points: List[str] - files: Optional[List[str]] - - -def search_packages_info(query: List[str]) -> Generator[_PackageInfo, None, None]: - """ - Gather details from installed distributions. Print distribution name, - version, location, and installed files. Installed files requires a - pip generated 'installed-files.txt' in the distributions '.egg-info' - directory. - """ - env = get_default_environment() - - installed = {dist.canonical_name: dist for dist in env.iter_all_distributions()} - query_names = [canonicalize_name(name) for name in query] - missing = sorted( - [name for name, pkg in zip(query, query_names) if pkg not in installed] - ) - if missing: - logger.warning("Package(s) not found: %s", ", ".join(missing)) - - def _get_requiring_packages(current_dist: BaseDistribution) -> Iterator[str]: - return ( - dist.metadata["Name"] or "UNKNOWN" - for dist in installed.values() - if current_dist.canonical_name - in {canonicalize_name(d.name) for d in dist.iter_dependencies()} - ) - - for query_name in query_names: - try: - dist = installed[query_name] - except KeyError: - continue - - try: - requires = sorted( - # Avoid duplicates in requirements (e.g. due to environment markers). - {req.name for req in dist.iter_dependencies()}, - key=str.lower, - ) - except InvalidRequirement: - requires = sorted(dist.iter_raw_dependencies(), key=str.lower) - - try: - required_by = sorted(_get_requiring_packages(dist), key=str.lower) - except InvalidRequirement: - required_by = ["#N/A"] - - try: - entry_points_text = dist.read_text("entry_points.txt") - entry_points = entry_points_text.splitlines(keepends=False) - except FileNotFoundError: - entry_points = [] - - files_iter = dist.iter_declared_entries() - if files_iter is None: - files: Optional[List[str]] = None - else: - files = sorted(files_iter) - - metadata = dist.metadata - - project_urls = metadata.get_all("Project-URL", []) - homepage = metadata.get("Home-page", "") - if not homepage: - # It's common that there is a "homepage" Project-URL, but Home-page - # remains unset (especially as PEP 621 doesn't surface the field). - # - # This logic was taken from PyPI's codebase. - for url in project_urls: - url_label, url = url.split(",", maxsplit=1) - normalized_label = ( - url_label.casefold().replace("-", "").replace("_", "").strip() - ) - if normalized_label == "homepage": - homepage = url.strip() - break - - yield _PackageInfo( - name=dist.raw_name, - version=dist.raw_version, - location=dist.location or "", - editable_project_location=dist.editable_project_location, - requires=requires, - required_by=required_by, - installer=dist.installer, - metadata_version=dist.metadata_version or "", - classifiers=metadata.get_all("Classifier", []), - summary=metadata.get("Summary", ""), - homepage=homepage, - project_urls=project_urls, - author=metadata.get("Author", ""), - author_email=metadata.get("Author-email", ""), - license=metadata.get("License", ""), - entry_points=entry_points, - files=files, - ) - - -def print_results( - distributions: Iterable[_PackageInfo], - list_files: bool, - verbose: bool, -) -> bool: - """ - Print the information from installed distributions found. - """ - results_printed = False - for i, dist in enumerate(distributions): - results_printed = True - if i > 0: - write_output("---") - - write_output("Name: %s", dist.name) - write_output("Version: %s", dist.version) - write_output("Summary: %s", dist.summary) - write_output("Home-page: %s", dist.homepage) - write_output("Author: %s", dist.author) - write_output("Author-email: %s", dist.author_email) - write_output("License: %s", dist.license) - write_output("Location: %s", dist.location) - if dist.editable_project_location is not None: - write_output( - "Editable project location: %s", dist.editable_project_location - ) - write_output("Requires: %s", ", ".join(dist.requires)) - write_output("Required-by: %s", ", ".join(dist.required_by)) - - if verbose: - write_output("Metadata-Version: %s", dist.metadata_version) - write_output("Installer: %s", dist.installer) - write_output("Classifiers:") - for classifier in dist.classifiers: - write_output(" %s", classifier) - write_output("Entry-points:") - for entry in dist.entry_points: - write_output(" %s", entry.strip()) - write_output("Project-URLs:") - for project_url in dist.project_urls: - write_output(" %s", project_url) - if list_files: - write_output("Files:") - if dist.files is None: - write_output("Cannot locate RECORD or installed-files.txt") - else: - for line in dist.files: - write_output(" %s", line.strip()) - return results_printed diff --git a/.env/Lib/site-packages/pip/_internal/commands/uninstall.py b/.env/Lib/site-packages/pip/_internal/commands/uninstall.py deleted file mode 100644 index bc0edea..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/uninstall.py +++ /dev/null @@ -1,114 +0,0 @@ -import logging -from optparse import Values -from typing import List - -from pip._vendor.packaging.utils import canonicalize_name - -from pip._internal.cli import cmdoptions -from pip._internal.cli.base_command import Command -from pip._internal.cli.index_command import SessionCommandMixin -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.exceptions import InstallationError -from pip._internal.req import parse_requirements -from pip._internal.req.constructors import ( - install_req_from_line, - install_req_from_parsed_requirement, -) -from pip._internal.utils.misc import ( - check_externally_managed, - protect_pip_from_modification_on_windows, - warn_if_run_as_root, -) - -logger = logging.getLogger(__name__) - - -class UninstallCommand(Command, SessionCommandMixin): - """ - Uninstall packages. - - pip is able to uninstall most installed packages. Known exceptions are: - - - Pure distutils packages installed with ``python setup.py install``, which - leave behind no metadata to determine what files were installed. - - Script wrappers installed by ``python setup.py develop``. - """ - - usage = """ - %prog [options] ... - %prog [options] -r ...""" - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-r", - "--requirement", - dest="requirements", - action="append", - default=[], - metavar="file", - help=( - "Uninstall all the packages listed in the given requirements " - "file. This option can be used multiple times." - ), - ) - self.cmd_opts.add_option( - "-y", - "--yes", - dest="yes", - action="store_true", - help="Don't ask for confirmation of uninstall deletions.", - ) - self.cmd_opts.add_option(cmdoptions.root_user_action()) - self.cmd_opts.add_option(cmdoptions.override_externally_managed()) - self.parser.insert_option_group(0, self.cmd_opts) - - def run(self, options: Values, args: List[str]) -> int: - session = self.get_default_session(options) - - reqs_to_uninstall = {} - for name in args: - req = install_req_from_line( - name, - isolated=options.isolated_mode, - ) - if req.name: - reqs_to_uninstall[canonicalize_name(req.name)] = req - else: - logger.warning( - "Invalid requirement: %r ignored -" - " the uninstall command expects named" - " requirements.", - name, - ) - for filename in options.requirements: - for parsed_req in parse_requirements( - filename, options=options, session=session - ): - req = install_req_from_parsed_requirement( - parsed_req, isolated=options.isolated_mode - ) - if req.name: - reqs_to_uninstall[canonicalize_name(req.name)] = req - if not reqs_to_uninstall: - raise InstallationError( - f"You must give at least one requirement to {self.name} (see " - f'"pip help {self.name}")' - ) - - if not options.override_externally_managed: - check_externally_managed() - - protect_pip_from_modification_on_windows( - modifying_pip="pip" in reqs_to_uninstall - ) - - for req in reqs_to_uninstall.values(): - uninstall_pathset = req.uninstall( - auto_confirm=options.yes, - verbose=self.verbosity > 0, - ) - if uninstall_pathset: - uninstall_pathset.commit() - if options.root_user_action == "warn": - warn_if_run_as_root() - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/commands/wheel.py b/.env/Lib/site-packages/pip/_internal/commands/wheel.py deleted file mode 100644 index 278719f..0000000 --- a/.env/Lib/site-packages/pip/_internal/commands/wheel.py +++ /dev/null @@ -1,182 +0,0 @@ -import logging -import os -import shutil -from optparse import Values -from typing import List - -from pip._internal.cache import WheelCache -from pip._internal.cli import cmdoptions -from pip._internal.cli.req_command import RequirementCommand, with_cleanup -from pip._internal.cli.status_codes import SUCCESS -from pip._internal.exceptions import CommandError -from pip._internal.operations.build.build_tracker import get_build_tracker -from pip._internal.req.req_install import ( - InstallRequirement, - check_legacy_setup_py_options, -) -from pip._internal.utils.misc import ensure_dir, normalize_path -from pip._internal.utils.temp_dir import TempDirectory -from pip._internal.wheel_builder import build, should_build_for_wheel_command - -logger = logging.getLogger(__name__) - - -class WheelCommand(RequirementCommand): - """ - Build Wheel archives for your requirements and dependencies. - - Wheel is a built-package format, and offers the advantage of not - recompiling your software during every install. For more details, see the - wheel docs: https://wheel.readthedocs.io/en/latest/ - - 'pip wheel' uses the build system interface as described here: - https://pip.pypa.io/en/stable/reference/build-system/ - - """ - - usage = """ - %prog [options] ... - %prog [options] -r ... - %prog [options] [-e] ... - %prog [options] [-e] ... - %prog [options] ...""" - - def add_options(self) -> None: - self.cmd_opts.add_option( - "-w", - "--wheel-dir", - dest="wheel_dir", - metavar="dir", - default=os.curdir, - help=( - "Build wheels into , where the default is the " - "current working directory." - ), - ) - self.cmd_opts.add_option(cmdoptions.no_binary()) - self.cmd_opts.add_option(cmdoptions.only_binary()) - self.cmd_opts.add_option(cmdoptions.prefer_binary()) - self.cmd_opts.add_option(cmdoptions.no_build_isolation()) - self.cmd_opts.add_option(cmdoptions.use_pep517()) - self.cmd_opts.add_option(cmdoptions.no_use_pep517()) - self.cmd_opts.add_option(cmdoptions.check_build_deps()) - self.cmd_opts.add_option(cmdoptions.constraints()) - self.cmd_opts.add_option(cmdoptions.editable()) - self.cmd_opts.add_option(cmdoptions.requirements()) - self.cmd_opts.add_option(cmdoptions.src()) - self.cmd_opts.add_option(cmdoptions.ignore_requires_python()) - self.cmd_opts.add_option(cmdoptions.no_deps()) - self.cmd_opts.add_option(cmdoptions.progress_bar()) - - self.cmd_opts.add_option( - "--no-verify", - dest="no_verify", - action="store_true", - default=False, - help="Don't verify if built wheel is valid.", - ) - - self.cmd_opts.add_option(cmdoptions.config_settings()) - self.cmd_opts.add_option(cmdoptions.build_options()) - self.cmd_opts.add_option(cmdoptions.global_options()) - - self.cmd_opts.add_option( - "--pre", - action="store_true", - default=False, - help=( - "Include pre-release and development versions. By default, " - "pip only finds stable versions." - ), - ) - - self.cmd_opts.add_option(cmdoptions.require_hashes()) - - index_opts = cmdoptions.make_option_group( - cmdoptions.index_group, - self.parser, - ) - - self.parser.insert_option_group(0, index_opts) - self.parser.insert_option_group(0, self.cmd_opts) - - @with_cleanup - def run(self, options: Values, args: List[str]) -> int: - session = self.get_default_session(options) - - finder = self._build_package_finder(options, session) - - options.wheel_dir = normalize_path(options.wheel_dir) - ensure_dir(options.wheel_dir) - - build_tracker = self.enter_context(get_build_tracker()) - - directory = TempDirectory( - delete=not options.no_clean, - kind="wheel", - globally_managed=True, - ) - - reqs = self.get_requirements(args, options, finder, session) - check_legacy_setup_py_options(options, reqs) - - wheel_cache = WheelCache(options.cache_dir) - - preparer = self.make_requirement_preparer( - temp_build_dir=directory, - options=options, - build_tracker=build_tracker, - session=session, - finder=finder, - download_dir=options.wheel_dir, - use_user_site=False, - verbosity=self.verbosity, - ) - - resolver = self.make_resolver( - preparer=preparer, - finder=finder, - options=options, - wheel_cache=wheel_cache, - ignore_requires_python=options.ignore_requires_python, - use_pep517=options.use_pep517, - ) - - self.trace_basic_info(finder) - - requirement_set = resolver.resolve(reqs, check_supported_wheels=True) - - reqs_to_build: List[InstallRequirement] = [] - for req in requirement_set.requirements.values(): - if req.is_wheel: - preparer.save_linked_requirement(req) - elif should_build_for_wheel_command(req): - reqs_to_build.append(req) - - preparer.prepare_linked_requirements_more(requirement_set.requirements.values()) - - # build wheels - build_successes, build_failures = build( - reqs_to_build, - wheel_cache=wheel_cache, - verify=(not options.no_verify), - build_options=options.build_options or [], - global_options=options.global_options or [], - ) - for req in build_successes: - assert req.link and req.link.is_wheel - assert req.local_file_path - # copy from cache to target directory - try: - shutil.copy(req.local_file_path, options.wheel_dir) - except OSError as e: - logger.warning( - "Building wheel for %s failed: %s", - req.name, - e, - ) - build_failures.append(req) - if len(build_failures) != 0: - raise CommandError("Failed to build one or more wheels") - - return SUCCESS diff --git a/.env/Lib/site-packages/pip/_internal/configuration.py b/.env/Lib/site-packages/pip/_internal/configuration.py deleted file mode 100644 index c25273d..0000000 --- a/.env/Lib/site-packages/pip/_internal/configuration.py +++ /dev/null @@ -1,383 +0,0 @@ -"""Configuration management setup - -Some terminology: -- name - As written in config files. -- value - Value associated with a name -- key - Name combined with it's section (section.name) -- variant - A single word describing where the configuration key-value pair came from -""" - -import configparser -import locale -import os -import sys -from typing import Any, Dict, Iterable, List, NewType, Optional, Tuple - -from pip._internal.exceptions import ( - ConfigurationError, - ConfigurationFileCouldNotBeLoaded, -) -from pip._internal.utils import appdirs -from pip._internal.utils.compat import WINDOWS -from pip._internal.utils.logging import getLogger -from pip._internal.utils.misc import ensure_dir, enum - -RawConfigParser = configparser.RawConfigParser # Shorthand -Kind = NewType("Kind", str) - -CONFIG_BASENAME = "pip.ini" if WINDOWS else "pip.conf" -ENV_NAMES_IGNORED = "version", "help" - -# The kinds of configurations there are. -kinds = enum( - USER="user", # User Specific - GLOBAL="global", # System Wide - SITE="site", # [Virtual] Environment Specific - ENV="env", # from PIP_CONFIG_FILE - ENV_VAR="env-var", # from Environment Variables -) -OVERRIDE_ORDER = kinds.GLOBAL, kinds.USER, kinds.SITE, kinds.ENV, kinds.ENV_VAR -VALID_LOAD_ONLY = kinds.USER, kinds.GLOBAL, kinds.SITE - -logger = getLogger(__name__) - - -# NOTE: Maybe use the optionx attribute to normalize keynames. -def _normalize_name(name: str) -> str: - """Make a name consistent regardless of source (environment or file)""" - name = name.lower().replace("_", "-") - if name.startswith("--"): - name = name[2:] # only prefer long opts - return name - - -def _disassemble_key(name: str) -> List[str]: - if "." not in name: - error_message = ( - "Key does not contain dot separated section and key. " - f"Perhaps you wanted to use 'global.{name}' instead?" - ) - raise ConfigurationError(error_message) - return name.split(".", 1) - - -def get_configuration_files() -> Dict[Kind, List[str]]: - global_config_files = [ - os.path.join(path, CONFIG_BASENAME) for path in appdirs.site_config_dirs("pip") - ] - - site_config_file = os.path.join(sys.prefix, CONFIG_BASENAME) - legacy_config_file = os.path.join( - os.path.expanduser("~"), - "pip" if WINDOWS else ".pip", - CONFIG_BASENAME, - ) - new_config_file = os.path.join(appdirs.user_config_dir("pip"), CONFIG_BASENAME) - return { - kinds.GLOBAL: global_config_files, - kinds.SITE: [site_config_file], - kinds.USER: [legacy_config_file, new_config_file], - } - - -class Configuration: - """Handles management of configuration. - - Provides an interface to accessing and managing configuration files. - - This class converts provides an API that takes "section.key-name" style - keys and stores the value associated with it as "key-name" under the - section "section". - - This allows for a clean interface wherein the both the section and the - key-name are preserved in an easy to manage form in the configuration files - and the data stored is also nice. - """ - - def __init__(self, isolated: bool, load_only: Optional[Kind] = None) -> None: - super().__init__() - - if load_only is not None and load_only not in VALID_LOAD_ONLY: - raise ConfigurationError( - "Got invalid value for load_only - should be one of {}".format( - ", ".join(map(repr, VALID_LOAD_ONLY)) - ) - ) - self.isolated = isolated - self.load_only = load_only - - # Because we keep track of where we got the data from - self._parsers: Dict[Kind, List[Tuple[str, RawConfigParser]]] = { - variant: [] for variant in OVERRIDE_ORDER - } - self._config: Dict[Kind, Dict[str, Any]] = { - variant: {} for variant in OVERRIDE_ORDER - } - self._modified_parsers: List[Tuple[str, RawConfigParser]] = [] - - def load(self) -> None: - """Loads configuration from configuration files and environment""" - self._load_config_files() - if not self.isolated: - self._load_environment_vars() - - def get_file_to_edit(self) -> Optional[str]: - """Returns the file with highest priority in configuration""" - assert self.load_only is not None, "Need to be specified a file to be editing" - - try: - return self._get_parser_to_modify()[0] - except IndexError: - return None - - def items(self) -> Iterable[Tuple[str, Any]]: - """Returns key-value pairs like dict.items() representing the loaded - configuration - """ - return self._dictionary.items() - - def get_value(self, key: str) -> Any: - """Get a value from the configuration.""" - orig_key = key - key = _normalize_name(key) - try: - return self._dictionary[key] - except KeyError: - # disassembling triggers a more useful error message than simply - # "No such key" in the case that the key isn't in the form command.option - _disassemble_key(key) - raise ConfigurationError(f"No such key - {orig_key}") - - def set_value(self, key: str, value: Any) -> None: - """Modify a value in the configuration.""" - key = _normalize_name(key) - self._ensure_have_load_only() - - assert self.load_only - fname, parser = self._get_parser_to_modify() - - if parser is not None: - section, name = _disassemble_key(key) - - # Modify the parser and the configuration - if not parser.has_section(section): - parser.add_section(section) - parser.set(section, name, value) - - self._config[self.load_only][key] = value - self._mark_as_modified(fname, parser) - - def unset_value(self, key: str) -> None: - """Unset a value in the configuration.""" - orig_key = key - key = _normalize_name(key) - self._ensure_have_load_only() - - assert self.load_only - if key not in self._config[self.load_only]: - raise ConfigurationError(f"No such key - {orig_key}") - - fname, parser = self._get_parser_to_modify() - - if parser is not None: - section, name = _disassemble_key(key) - if not ( - parser.has_section(section) and parser.remove_option(section, name) - ): - # The option was not removed. - raise ConfigurationError( - "Fatal Internal error [id=1]. Please report as a bug." - ) - - # The section may be empty after the option was removed. - if not parser.items(section): - parser.remove_section(section) - self._mark_as_modified(fname, parser) - - del self._config[self.load_only][key] - - def save(self) -> None: - """Save the current in-memory state.""" - self._ensure_have_load_only() - - for fname, parser in self._modified_parsers: - logger.info("Writing to %s", fname) - - # Ensure directory exists. - ensure_dir(os.path.dirname(fname)) - - # Ensure directory's permission(need to be writeable) - try: - with open(fname, "w") as f: - parser.write(f) - except OSError as error: - raise ConfigurationError( - f"An error occurred while writing to the configuration file " - f"{fname}: {error}" - ) - - # - # Private routines - # - - def _ensure_have_load_only(self) -> None: - if self.load_only is None: - raise ConfigurationError("Needed a specific file to be modifying.") - logger.debug("Will be working with %s variant only", self.load_only) - - @property - def _dictionary(self) -> Dict[str, Any]: - """A dictionary representing the loaded configuration.""" - # NOTE: Dictionaries are not populated if not loaded. So, conditionals - # are not needed here. - retval = {} - - for variant in OVERRIDE_ORDER: - retval.update(self._config[variant]) - - return retval - - def _load_config_files(self) -> None: - """Loads configuration from configuration files""" - config_files = dict(self.iter_config_files()) - if config_files[kinds.ENV][0:1] == [os.devnull]: - logger.debug( - "Skipping loading configuration files due to " - "environment's PIP_CONFIG_FILE being os.devnull" - ) - return - - for variant, files in config_files.items(): - for fname in files: - # If there's specific variant set in `load_only`, load only - # that variant, not the others. - if self.load_only is not None and variant != self.load_only: - logger.debug("Skipping file '%s' (variant: %s)", fname, variant) - continue - - parser = self._load_file(variant, fname) - - # Keeping track of the parsers used - self._parsers[variant].append((fname, parser)) - - def _load_file(self, variant: Kind, fname: str) -> RawConfigParser: - logger.verbose("For variant '%s', will try loading '%s'", variant, fname) - parser = self._construct_parser(fname) - - for section in parser.sections(): - items = parser.items(section) - self._config[variant].update(self._normalized_keys(section, items)) - - return parser - - def _construct_parser(self, fname: str) -> RawConfigParser: - parser = configparser.RawConfigParser() - # If there is no such file, don't bother reading it but create the - # parser anyway, to hold the data. - # Doing this is useful when modifying and saving files, where we don't - # need to construct a parser. - if os.path.exists(fname): - locale_encoding = locale.getpreferredencoding(False) - try: - parser.read(fname, encoding=locale_encoding) - except UnicodeDecodeError: - # See https://github.com/pypa/pip/issues/4963 - raise ConfigurationFileCouldNotBeLoaded( - reason=f"contains invalid {locale_encoding} characters", - fname=fname, - ) - except configparser.Error as error: - # See https://github.com/pypa/pip/issues/4893 - raise ConfigurationFileCouldNotBeLoaded(error=error) - return parser - - def _load_environment_vars(self) -> None: - """Loads configuration from environment variables""" - self._config[kinds.ENV_VAR].update( - self._normalized_keys(":env:", self.get_environ_vars()) - ) - - def _normalized_keys( - self, section: str, items: Iterable[Tuple[str, Any]] - ) -> Dict[str, Any]: - """Normalizes items to construct a dictionary with normalized keys. - - This routine is where the names become keys and are made the same - regardless of source - configuration files or environment. - """ - normalized = {} - for name, val in items: - key = section + "." + _normalize_name(name) - normalized[key] = val - return normalized - - def get_environ_vars(self) -> Iterable[Tuple[str, str]]: - """Returns a generator with all environmental vars with prefix PIP_""" - for key, val in os.environ.items(): - if key.startswith("PIP_"): - name = key[4:].lower() - if name not in ENV_NAMES_IGNORED: - yield name, val - - # XXX: This is patched in the tests. - def iter_config_files(self) -> Iterable[Tuple[Kind, List[str]]]: - """Yields variant and configuration files associated with it. - - This should be treated like items of a dictionary. The order - here doesn't affect what gets overridden. That is controlled - by OVERRIDE_ORDER. However this does control the order they are - displayed to the user. It's probably most ergononmic to display - things in the same order as OVERRIDE_ORDER - """ - # SMELL: Move the conditions out of this function - - env_config_file = os.environ.get("PIP_CONFIG_FILE", None) - config_files = get_configuration_files() - - yield kinds.GLOBAL, config_files[kinds.GLOBAL] - - # per-user config is not loaded when env_config_file exists - should_load_user_config = not self.isolated and not ( - env_config_file and os.path.exists(env_config_file) - ) - if should_load_user_config: - # The legacy config file is overridden by the new config file - yield kinds.USER, config_files[kinds.USER] - - # virtualenv config - yield kinds.SITE, config_files[kinds.SITE] - - if env_config_file is not None: - yield kinds.ENV, [env_config_file] - else: - yield kinds.ENV, [] - - def get_values_in_config(self, variant: Kind) -> Dict[str, Any]: - """Get values present in a config file""" - return self._config[variant] - - def _get_parser_to_modify(self) -> Tuple[str, RawConfigParser]: - # Determine which parser to modify - assert self.load_only - parsers = self._parsers[self.load_only] - if not parsers: - # This should not happen if everything works correctly. - raise ConfigurationError( - "Fatal Internal error [id=2]. Please report as a bug." - ) - - # Use the highest priority parser. - return parsers[-1] - - # XXX: This is patched in the tests. - def _mark_as_modified(self, fname: str, parser: RawConfigParser) -> None: - file_parser_tuple = (fname, parser) - if file_parser_tuple not in self._modified_parsers: - self._modified_parsers.append(file_parser_tuple) - - def __repr__(self) -> str: - return f"{self.__class__.__name__}({self._dictionary!r})" diff --git a/.env/Lib/site-packages/pip/_internal/distributions/__init__.py b/.env/Lib/site-packages/pip/_internal/distributions/__init__.py deleted file mode 100644 index 9a89a83..0000000 --- a/.env/Lib/site-packages/pip/_internal/distributions/__init__.py +++ /dev/null @@ -1,21 +0,0 @@ -from pip._internal.distributions.base import AbstractDistribution -from pip._internal.distributions.sdist import SourceDistribution -from pip._internal.distributions.wheel import WheelDistribution -from pip._internal.req.req_install import InstallRequirement - - -def make_distribution_for_install_requirement( - install_req: InstallRequirement, -) -> AbstractDistribution: - """Returns a Distribution for the given InstallRequirement""" - # Editable requirements will always be source distributions. They use the - # legacy logic until we create a modern standard for them. - if install_req.editable: - return SourceDistribution(install_req) - - # If it's a wheel, it's a WheelDistribution - if install_req.is_wheel: - return WheelDistribution(install_req) - - # Otherwise, a SourceDistribution - return SourceDistribution(install_req) diff --git a/.env/Lib/site-packages/pip/_internal/distributions/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/distributions/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 097c86d86c744a998baff1873c9f31cf90b52811..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 941 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 I6Kp5`7fCIpOaK4? diff --git a/.env/Lib/site-packages/pip/_internal/distributions/__pycache__/wheel.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/distributions/__pycache__/wheel.cpython-312.pyc deleted file mode 100644 index 345914c1633ea73a5b3e9e161463633fdf15be6d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 2281 zcmaJ?&2JM&6rWwM*K5bdPKgj7g^W-UH4x{}OK2-pN`Zb65Fe_#vb0@$#$?HUF|!V_ z)kvs{D%uN2uFy-1@(+|t`2%`!n@UtGsv=b!dNVauB~E>BH%{#U9cgFYynXXNfAi+Y zQmH^-41W01Iglmf58Mm~?2|EQ!E%c*!W5Tyim7<2sVcZv-HfN1T0&>stf!lLLThf$ zGfczFn|ZHb78Jr2p@V$Zo$!igQ6VZhO_;t)SdQy!nbeF~>KN7hpSY>2ap~fx=T2HD z-amQbqYvg!$Hv((asuCWt4ciCwEe($nzrk7x#im)2j$yco@JR12`DfFK9%KL1}gx6 zqLZ&q#FL3{qmFm4p68+9lKYXQHccF+qYl>!1q$Qhf+HylxTJm%QJ-^0ZNEd|XlM%? z(?V^T9+wnSNgEv02;Pj`4sH7(gv?njQ5EZgM{VIZm5l&~3zlr4!E}ps$X#XrE`btg zlDjR5qVyR%@n-#!!B0QT8q(J(rAHGk4HAMBb&BF;3b?C zi!HcU6iOC=gcpboz>d)rKmgbTP>9b_Ov7nAqHPIab)q8)UOgGj z5d3NzAnUugFI^s)XIWuz6+6bcP2mqzXXRnvcv^IA6p5l z>Tw3t$7MHw4r5_ZR%k~Hv4+0|-g|)w76_rLDkfoxvx%D$6Go|c1zEeWs|-^$)>yiS z6WgPCsJ&EpH|UZLn68t4fmEh$E`PQBpnUM>^1+^QsaL(U4s-e7ujTFdKQa0nEoNsPL&{%H}Wl$q3FjciARbme^ zUd3rIOcGCH@zSL2O}i8y%hzTWyR*+H?tc%wh8FNHLGp?uUrHjZ7GuLoo(;>23zp>t ztPM4`;-Y0;ZQCxE8Fo3#VnGuKBNRc%Mbr`KYn*}SMQjI3pkjgbiM2)$xMCXj7;0iB z2}l%iixcwYXE3djCu$~ZKr!`geO1X$-G2SMH}1^*MSvV+hRnWhWX~$S!*2};68$MX zTY6NP?q^}`>ty%7ehw*v?3wQ8kt&eFwx<(7rN+_Ze05u_N1YIGFR&=7|9hab)~qG& zvq03+b_)VEcrRU~gqW z_^>8eOH(zTzGueUX?2c?BVZCcoScEFpHUR$cXH@4Ir^9!|3}pn^$EeLNVLh{w0EBB QyOmOJ=aIh&tVu%t1tIfXxc~qF diff --git a/.env/Lib/site-packages/pip/_internal/distributions/base.py b/.env/Lib/site-packages/pip/_internal/distributions/base.py deleted file mode 100644 index 6e4d0c9..0000000 --- a/.env/Lib/site-packages/pip/_internal/distributions/base.py +++ /dev/null @@ -1,53 +0,0 @@ -import abc -from typing import TYPE_CHECKING, Optional - -from pip._internal.metadata.base import BaseDistribution -from pip._internal.req import InstallRequirement - -if TYPE_CHECKING: - from pip._internal.index.package_finder import PackageFinder - - -class AbstractDistribution(metaclass=abc.ABCMeta): - """A base class for handling installable artifacts. - - The requirements for anything installable are as follows: - - - we must be able to determine the requirement name - (or we can't correctly handle the non-upgrade case). - - - for packages with setup requirements, we must also be able - to determine their requirements without installing additional - packages (for the same reason as run-time dependencies) - - - we must be able to create a Distribution object exposing the - above metadata. - - - if we need to do work in the build tracker, we must be able to generate a unique - string to identify the requirement in the build tracker. - """ - - def __init__(self, req: InstallRequirement) -> None: - super().__init__() - self.req = req - - @abc.abstractproperty - def build_tracker_id(self) -> Optional[str]: - """A string that uniquely identifies this requirement to the build tracker. - - If None, then this dist has no work to do in the build tracker, and - ``.prepare_distribution_metadata()`` will not be called.""" - raise NotImplementedError() - - @abc.abstractmethod - def get_metadata_distribution(self) -> BaseDistribution: - raise NotImplementedError() - - @abc.abstractmethod - def prepare_distribution_metadata( - self, - finder: "PackageFinder", - build_isolation: bool, - check_build_deps: bool, - ) -> None: - raise NotImplementedError() diff --git a/.env/Lib/site-packages/pip/_internal/distributions/installed.py b/.env/Lib/site-packages/pip/_internal/distributions/installed.py deleted file mode 100644 index ab8d53b..0000000 --- a/.env/Lib/site-packages/pip/_internal/distributions/installed.py +++ /dev/null @@ -1,29 +0,0 @@ -from typing import Optional - -from pip._internal.distributions.base import AbstractDistribution -from pip._internal.index.package_finder import PackageFinder -from pip._internal.metadata import BaseDistribution - - -class InstalledDistribution(AbstractDistribution): - """Represents an installed package. - - This does not need any preparation as the required information has already - been computed. - """ - - @property - def build_tracker_id(self) -> Optional[str]: - return None - - def get_metadata_distribution(self) -> BaseDistribution: - assert self.req.satisfied_by is not None, "not actually installed" - return self.req.satisfied_by - - def prepare_distribution_metadata( - self, - finder: PackageFinder, - build_isolation: bool, - check_build_deps: bool, - ) -> None: - pass diff --git a/.env/Lib/site-packages/pip/_internal/distributions/sdist.py b/.env/Lib/site-packages/pip/_internal/distributions/sdist.py deleted file mode 100644 index 28ea5ce..0000000 --- a/.env/Lib/site-packages/pip/_internal/distributions/sdist.py +++ /dev/null @@ -1,158 +0,0 @@ -import logging -from typing import TYPE_CHECKING, Iterable, Optional, Set, Tuple - -from pip._internal.build_env import BuildEnvironment -from pip._internal.distributions.base import AbstractDistribution -from pip._internal.exceptions import InstallationError -from pip._internal.metadata import BaseDistribution -from pip._internal.utils.subprocess import runner_with_spinner_message - -if TYPE_CHECKING: - from pip._internal.index.package_finder import PackageFinder - -logger = logging.getLogger(__name__) - - -class SourceDistribution(AbstractDistribution): - """Represents a source distribution. - - The preparation step for these needs metadata for the packages to be - generated, either using PEP 517 or using the legacy `setup.py egg_info`. - """ - - @property - def build_tracker_id(self) -> Optional[str]: - """Identify this requirement uniquely by its link.""" - assert self.req.link - return self.req.link.url_without_fragment - - def get_metadata_distribution(self) -> BaseDistribution: - return self.req.get_dist() - - def prepare_distribution_metadata( - self, - finder: "PackageFinder", - build_isolation: bool, - check_build_deps: bool, - ) -> None: - # Load pyproject.toml, to determine whether PEP 517 is to be used - self.req.load_pyproject_toml() - - # Set up the build isolation, if this requirement should be isolated - should_isolate = self.req.use_pep517 and build_isolation - if should_isolate: - # Setup an isolated environment and install the build backend static - # requirements in it. - self._prepare_build_backend(finder) - # Check that if the requirement is editable, it either supports PEP 660 or - # has a setup.py or a setup.cfg. This cannot be done earlier because we need - # to setup the build backend to verify it supports build_editable, nor can - # it be done later, because we want to avoid installing build requirements - # needlessly. Doing it here also works around setuptools generating - # UNKNOWN.egg-info when running get_requires_for_build_wheel on a directory - # without setup.py nor setup.cfg. - self.req.isolated_editable_sanity_check() - # Install the dynamic build requirements. - self._install_build_reqs(finder) - # Check if the current environment provides build dependencies - should_check_deps = self.req.use_pep517 and check_build_deps - if should_check_deps: - pyproject_requires = self.req.pyproject_requires - assert pyproject_requires is not None - conflicting, missing = self.req.build_env.check_requirements( - pyproject_requires - ) - if conflicting: - self._raise_conflicts("the backend dependencies", conflicting) - if missing: - self._raise_missing_reqs(missing) - self.req.prepare_metadata() - - def _prepare_build_backend(self, finder: "PackageFinder") -> None: - # Isolate in a BuildEnvironment and install the build-time - # requirements. - pyproject_requires = self.req.pyproject_requires - assert pyproject_requires is not None - - self.req.build_env = BuildEnvironment() - self.req.build_env.install_requirements( - finder, pyproject_requires, "overlay", kind="build dependencies" - ) - conflicting, missing = self.req.build_env.check_requirements( - self.req.requirements_to_check - ) - if conflicting: - self._raise_conflicts("PEP 517/518 supported requirements", conflicting) - if missing: - logger.warning( - "Missing build requirements in pyproject.toml for %s.", - self.req, - ) - logger.warning( - "The project does not specify a build backend, and " - "pip cannot fall back to setuptools without %s.", - " and ".join(map(repr, sorted(missing))), - ) - - def _get_build_requires_wheel(self) -> Iterable[str]: - with self.req.build_env: - runner = runner_with_spinner_message("Getting requirements to build wheel") - backend = self.req.pep517_backend - assert backend is not None - with backend.subprocess_runner(runner): - return backend.get_requires_for_build_wheel() - - def _get_build_requires_editable(self) -> Iterable[str]: - with self.req.build_env: - runner = runner_with_spinner_message( - "Getting requirements to build editable" - ) - backend = self.req.pep517_backend - assert backend is not None - with backend.subprocess_runner(runner): - return backend.get_requires_for_build_editable() - - def _install_build_reqs(self, finder: "PackageFinder") -> None: - # Install any extra build dependencies that the backend requests. - # This must be done in a second pass, as the pyproject.toml - # dependencies must be installed before we can call the backend. - if ( - self.req.editable - and self.req.permit_editable_wheels - and self.req.supports_pyproject_editable - ): - build_reqs = self._get_build_requires_editable() - else: - build_reqs = self._get_build_requires_wheel() - conflicting, missing = self.req.build_env.check_requirements(build_reqs) - if conflicting: - self._raise_conflicts("the backend dependencies", conflicting) - self.req.build_env.install_requirements( - finder, missing, "normal", kind="backend dependencies" - ) - - def _raise_conflicts( - self, conflicting_with: str, conflicting_reqs: Set[Tuple[str, str]] - ) -> None: - format_string = ( - "Some build dependencies for {requirement} " - "conflict with {conflicting_with}: {description}." - ) - error_message = format_string.format( - requirement=self.req, - conflicting_with=conflicting_with, - description=", ".join( - f"{installed} is incompatible with {wanted}" - for installed, wanted in sorted(conflicting_reqs) - ), - ) - raise InstallationError(error_message) - - def _raise_missing_reqs(self, missing: Set[str]) -> None: - format_string = ( - "Some build dependencies for {requirement} are missing: {missing}." - ) - error_message = format_string.format( - requirement=self.req, missing=", ".join(map(repr, sorted(missing))) - ) - raise InstallationError(error_message) diff --git a/.env/Lib/site-packages/pip/_internal/distributions/wheel.py b/.env/Lib/site-packages/pip/_internal/distributions/wheel.py deleted file mode 100644 index bfadd39..0000000 --- a/.env/Lib/site-packages/pip/_internal/distributions/wheel.py +++ /dev/null @@ -1,42 +0,0 @@ -from typing import TYPE_CHECKING, Optional - -from pip._vendor.packaging.utils import canonicalize_name - -from pip._internal.distributions.base import AbstractDistribution -from pip._internal.metadata import ( - BaseDistribution, - FilesystemWheel, - get_wheel_distribution, -) - -if TYPE_CHECKING: - from pip._internal.index.package_finder import PackageFinder - - -class WheelDistribution(AbstractDistribution): - """Represents a wheel distribution. - - This does not need any preparation as wheels can be directly unpacked. - """ - - @property - def build_tracker_id(self) -> Optional[str]: - return None - - def get_metadata_distribution(self) -> BaseDistribution: - """Loads the metadata from the wheel file into memory and returns a - Distribution that uses it, not relying on the wheel file or - requirement. - """ - assert self.req.local_file_path, "Set as part of preparation during download" - assert self.req.name, "Wheels are never unnamed" - wheel = FilesystemWheel(self.req.local_file_path) - return get_wheel_distribution(wheel, canonicalize_name(self.req.name)) - - def prepare_distribution_metadata( - self, - finder: "PackageFinder", - build_isolation: bool, - check_build_deps: bool, - ) -> None: - pass diff --git a/.env/Lib/site-packages/pip/_internal/exceptions.py b/.env/Lib/site-packages/pip/_internal/exceptions.py deleted file mode 100644 index 2587740..0000000 --- a/.env/Lib/site-packages/pip/_internal/exceptions.py +++ /dev/null @@ -1,777 +0,0 @@ -"""Exceptions used throughout package. - -This module MUST NOT try to import from anything within `pip._internal` to -operate. This is expected to be importable from any/all files within the -subpackage and, thus, should not depend on them. -""" - -import configparser -import contextlib -import locale -import logging -import pathlib -import re -import sys -from itertools import chain, groupby, repeat -from typing import TYPE_CHECKING, Dict, Iterator, List, Literal, Optional, Union - -from pip._vendor.rich.console import Console, ConsoleOptions, RenderResult -from pip._vendor.rich.markup import escape -from pip._vendor.rich.text import Text - -if TYPE_CHECKING: - from hashlib import _Hash - - from pip._vendor.requests.models import Request, Response - - from pip._internal.metadata import BaseDistribution - from pip._internal.req.req_install import InstallRequirement - -logger = logging.getLogger(__name__) - - -# -# Scaffolding -# -def _is_kebab_case(s: str) -> bool: - return re.match(r"^[a-z]+(-[a-z]+)*$", s) is not None - - -def _prefix_with_indent( - s: Union[Text, str], - console: Console, - *, - prefix: str, - indent: str, -) -> Text: - if isinstance(s, Text): - text = s - else: - text = console.render_str(s) - - return console.render_str(prefix, overflow="ignore") + console.render_str( - f"\n{indent}", overflow="ignore" - ).join(text.split(allow_blank=True)) - - -class PipError(Exception): - """The base pip error.""" - - -class DiagnosticPipError(PipError): - """An error, that presents diagnostic information to the user. - - This contains a bunch of logic, to enable pretty presentation of our error - messages. Each error gets a unique reference. Each error can also include - additional context, a hint and/or a note -- which are presented with the - main error message in a consistent style. - - This is adapted from the error output styling in `sphinx-theme-builder`. - """ - - reference: str - - def __init__( - self, - *, - kind: 'Literal["error", "warning"]' = "error", - reference: Optional[str] = None, - message: Union[str, Text], - context: Optional[Union[str, Text]], - hint_stmt: Optional[Union[str, Text]], - note_stmt: Optional[Union[str, Text]] = None, - link: Optional[str] = None, - ) -> None: - # Ensure a proper reference is provided. - if reference is None: - assert hasattr(self, "reference"), "error reference not provided!" - reference = self.reference - assert _is_kebab_case(reference), "error reference must be kebab-case!" - - self.kind = kind - self.reference = reference - - self.message = message - self.context = context - - self.note_stmt = note_stmt - self.hint_stmt = hint_stmt - - self.link = link - - super().__init__(f"<{self.__class__.__name__}: {self.reference}>") - - def __repr__(self) -> str: - return ( - f"<{self.__class__.__name__}(" - f"reference={self.reference!r}, " - f"message={self.message!r}, " - f"context={self.context!r}, " - f"note_stmt={self.note_stmt!r}, " - f"hint_stmt={self.hint_stmt!r}" - ")>" - ) - - def __rich_console__( - self, - console: Console, - options: ConsoleOptions, - ) -> RenderResult: - colour = "red" if self.kind == "error" else "yellow" - - yield f"[{colour} bold]{self.kind}[/]: [bold]{self.reference}[/]" - yield "" - - if not options.ascii_only: - # Present the main message, with relevant context indented. - if self.context is not None: - yield _prefix_with_indent( - self.message, - console, - prefix=f"[{colour}]×[/] ", - indent=f"[{colour}]│[/] ", - ) - yield _prefix_with_indent( - self.context, - console, - prefix=f"[{colour}]╰─>[/] ", - indent=f"[{colour}] [/] ", - ) - else: - yield _prefix_with_indent( - self.message, - console, - prefix="[red]×[/] ", - indent=" ", - ) - else: - yield self.message - if self.context is not None: - yield "" - yield self.context - - if self.note_stmt is not None or self.hint_stmt is not None: - yield "" - - if self.note_stmt is not None: - yield _prefix_with_indent( - self.note_stmt, - console, - prefix="[magenta bold]note[/]: ", - indent=" ", - ) - if self.hint_stmt is not None: - yield _prefix_with_indent( - self.hint_stmt, - console, - prefix="[cyan bold]hint[/]: ", - indent=" ", - ) - - if self.link is not None: - yield "" - yield f"Link: {self.link}" - - -# -# Actual Errors -# -class ConfigurationError(PipError): - """General exception in configuration""" - - -class InstallationError(PipError): - """General exception during installation""" - - -class MissingPyProjectBuildRequires(DiagnosticPipError): - """Raised when pyproject.toml has `build-system`, but no `build-system.requires`.""" - - reference = "missing-pyproject-build-system-requires" - - def __init__(self, *, package: str) -> None: - super().__init__( - message=f"Can not process {escape(package)}", - context=Text( - "This package has an invalid pyproject.toml file.\n" - "The [build-system] table is missing the mandatory `requires` key." - ), - note_stmt="This is an issue with the package mentioned above, not pip.", - hint_stmt=Text("See PEP 518 for the detailed specification."), - ) - - -class InvalidPyProjectBuildRequires(DiagnosticPipError): - """Raised when pyproject.toml an invalid `build-system.requires`.""" - - reference = "invalid-pyproject-build-system-requires" - - def __init__(self, *, package: str, reason: str) -> None: - super().__init__( - message=f"Can not process {escape(package)}", - context=Text( - "This package has an invalid `build-system.requires` key in " - f"pyproject.toml.\n{reason}" - ), - note_stmt="This is an issue with the package mentioned above, not pip.", - hint_stmt=Text("See PEP 518 for the detailed specification."), - ) - - -class NoneMetadataError(PipError): - """Raised when accessing a Distribution's "METADATA" or "PKG-INFO". - - This signifies an inconsistency, when the Distribution claims to have - the metadata file (if not, raise ``FileNotFoundError`` instead), but is - not actually able to produce its content. This may be due to permission - errors. - """ - - def __init__( - self, - dist: "BaseDistribution", - metadata_name: str, - ) -> None: - """ - :param dist: A Distribution object. - :param metadata_name: The name of the metadata being accessed - (can be "METADATA" or "PKG-INFO"). - """ - self.dist = dist - self.metadata_name = metadata_name - - def __str__(self) -> str: - # Use `dist` in the error message because its stringification - # includes more information, like the version and location. - return f"None {self.metadata_name} metadata found for distribution: {self.dist}" - - -class UserInstallationInvalid(InstallationError): - """A --user install is requested on an environment without user site.""" - - def __str__(self) -> str: - return "User base directory is not specified" - - -class InvalidSchemeCombination(InstallationError): - def __str__(self) -> str: - before = ", ".join(str(a) for a in self.args[:-1]) - return f"Cannot set {before} and {self.args[-1]} together" - - -class DistributionNotFound(InstallationError): - """Raised when a distribution cannot be found to satisfy a requirement""" - - -class RequirementsFileParseError(InstallationError): - """Raised when a general error occurs parsing a requirements file line.""" - - -class BestVersionAlreadyInstalled(PipError): - """Raised when the most up-to-date version of a package is already - installed.""" - - -class BadCommand(PipError): - """Raised when virtualenv or a command is not found""" - - -class CommandError(PipError): - """Raised when there is an error in command-line arguments""" - - -class PreviousBuildDirError(PipError): - """Raised when there's a previous conflicting build directory""" - - -class NetworkConnectionError(PipError): - """HTTP connection error""" - - def __init__( - self, - error_msg: str, - response: Optional["Response"] = None, - request: Optional["Request"] = None, - ) -> None: - """ - Initialize NetworkConnectionError with `request` and `response` - objects. - """ - self.response = response - self.request = request - self.error_msg = error_msg - if ( - self.response is not None - and not self.request - and hasattr(response, "request") - ): - self.request = self.response.request - super().__init__(error_msg, response, request) - - def __str__(self) -> str: - return str(self.error_msg) - - -class InvalidWheelFilename(InstallationError): - """Invalid wheel filename.""" - - -class UnsupportedWheel(InstallationError): - """Unsupported wheel.""" - - -class InvalidWheel(InstallationError): - """Invalid (e.g. corrupt) wheel.""" - - def __init__(self, location: str, name: str): - self.location = location - self.name = name - - def __str__(self) -> str: - return f"Wheel '{self.name}' located at {self.location} is invalid." - - -class MetadataInconsistent(InstallationError): - """Built metadata contains inconsistent information. - - This is raised when the metadata contains values (e.g. name and version) - that do not match the information previously obtained from sdist filename, - user-supplied ``#egg=`` value, or an install requirement name. - """ - - def __init__( - self, ireq: "InstallRequirement", field: str, f_val: str, m_val: str - ) -> None: - self.ireq = ireq - self.field = field - self.f_val = f_val - self.m_val = m_val - - def __str__(self) -> str: - return ( - f"Requested {self.ireq} has inconsistent {self.field}: " - f"expected {self.f_val!r}, but metadata has {self.m_val!r}" - ) - - -class MetadataInvalid(InstallationError): - """Metadata is invalid.""" - - def __init__(self, ireq: "InstallRequirement", error: str) -> None: - self.ireq = ireq - self.error = error - - def __str__(self) -> str: - return f"Requested {self.ireq} has invalid metadata: {self.error}" - - -class InstallationSubprocessError(DiagnosticPipError, InstallationError): - """A subprocess call failed.""" - - reference = "subprocess-exited-with-error" - - def __init__( - self, - *, - command_description: str, - exit_code: int, - output_lines: Optional[List[str]], - ) -> None: - if output_lines is None: - output_prompt = Text("See above for output.") - else: - output_prompt = ( - Text.from_markup(f"[red][{len(output_lines)} lines of output][/]\n") - + Text("".join(output_lines)) - + Text.from_markup(R"[red]\[end of output][/]") - ) - - super().__init__( - message=( - f"[green]{escape(command_description)}[/] did not run successfully.\n" - f"exit code: {exit_code}" - ), - context=output_prompt, - hint_stmt=None, - note_stmt=( - "This error originates from a subprocess, and is likely not a " - "problem with pip." - ), - ) - - self.command_description = command_description - self.exit_code = exit_code - - def __str__(self) -> str: - return f"{self.command_description} exited with {self.exit_code}" - - -class MetadataGenerationFailed(InstallationSubprocessError, InstallationError): - reference = "metadata-generation-failed" - - def __init__( - self, - *, - package_details: str, - ) -> None: - super(InstallationSubprocessError, self).__init__( - message="Encountered error while generating package metadata.", - context=escape(package_details), - hint_stmt="See above for details.", - note_stmt="This is an issue with the package mentioned above, not pip.", - ) - - def __str__(self) -> str: - return "metadata generation failed" - - -class HashErrors(InstallationError): - """Multiple HashError instances rolled into one for reporting""" - - def __init__(self) -> None: - self.errors: List["HashError"] = [] - - def append(self, error: "HashError") -> None: - self.errors.append(error) - - def __str__(self) -> str: - lines = [] - self.errors.sort(key=lambda e: e.order) - for cls, errors_of_cls in groupby(self.errors, lambda e: e.__class__): - lines.append(cls.head) - lines.extend(e.body() for e in errors_of_cls) - if lines: - return "\n".join(lines) - return "" - - def __bool__(self) -> bool: - return bool(self.errors) - - -class HashError(InstallationError): - """ - A failure to verify a package against known-good hashes - - :cvar order: An int sorting hash exception classes by difficulty of - recovery (lower being harder), so the user doesn't bother fretting - about unpinned packages when he has deeper issues, like VCS - dependencies, to deal with. Also keeps error reports in a - deterministic order. - :cvar head: A section heading for display above potentially many - exceptions of this kind - :ivar req: The InstallRequirement that triggered this error. This is - pasted on after the exception is instantiated, because it's not - typically available earlier. - - """ - - req: Optional["InstallRequirement"] = None - head = "" - order: int = -1 - - def body(self) -> str: - """Return a summary of me for display under the heading. - - This default implementation simply prints a description of the - triggering requirement. - - :param req: The InstallRequirement that provoked this error, with - its link already populated by the resolver's _populate_link(). - - """ - return f" {self._requirement_name()}" - - def __str__(self) -> str: - return f"{self.head}\n{self.body()}" - - def _requirement_name(self) -> str: - """Return a description of the requirement that triggered me. - - This default implementation returns long description of the req, with - line numbers - - """ - return str(self.req) if self.req else "unknown package" - - -class VcsHashUnsupported(HashError): - """A hash was provided for a version-control-system-based requirement, but - we don't have a method for hashing those.""" - - order = 0 - head = ( - "Can't verify hashes for these requirements because we don't " - "have a way to hash version control repositories:" - ) - - -class DirectoryUrlHashUnsupported(HashError): - """A hash was provided for a version-control-system-based requirement, but - we don't have a method for hashing those.""" - - order = 1 - head = ( - "Can't verify hashes for these file:// requirements because they " - "point to directories:" - ) - - -class HashMissing(HashError): - """A hash was needed for a requirement but is absent.""" - - order = 2 - head = ( - "Hashes are required in --require-hashes mode, but they are " - "missing from some requirements. Here is a list of those " - "requirements along with the hashes their downloaded archives " - "actually had. Add lines like these to your requirements files to " - "prevent tampering. (If you did not enable --require-hashes " - "manually, note that it turns on automatically when any package " - "has a hash.)" - ) - - def __init__(self, gotten_hash: str) -> None: - """ - :param gotten_hash: The hash of the (possibly malicious) archive we - just downloaded - """ - self.gotten_hash = gotten_hash - - def body(self) -> str: - # Dodge circular import. - from pip._internal.utils.hashes import FAVORITE_HASH - - package = None - if self.req: - # In the case of URL-based requirements, display the original URL - # seen in the requirements file rather than the package name, - # so the output can be directly copied into the requirements file. - package = ( - self.req.original_link - if self.req.is_direct - # In case someone feeds something downright stupid - # to InstallRequirement's constructor. - else getattr(self.req, "req", None) - ) - return " {} --hash={}:{}".format( - package or "unknown package", FAVORITE_HASH, self.gotten_hash - ) - - -class HashUnpinned(HashError): - """A requirement had a hash specified but was not pinned to a specific - version.""" - - order = 3 - head = ( - "In --require-hashes mode, all requirements must have their " - "versions pinned with ==. These do not:" - ) - - -class HashMismatch(HashError): - """ - Distribution file hash values don't match. - - :ivar package_name: The name of the package that triggered the hash - mismatch. Feel free to write to this after the exception is raise to - improve its error message. - - """ - - order = 4 - head = ( - "THESE PACKAGES DO NOT MATCH THE HASHES FROM THE REQUIREMENTS " - "FILE. If you have updated the package versions, please update " - "the hashes. Otherwise, examine the package contents carefully; " - "someone may have tampered with them." - ) - - def __init__(self, allowed: Dict[str, List[str]], gots: Dict[str, "_Hash"]) -> None: - """ - :param allowed: A dict of algorithm names pointing to lists of allowed - hex digests - :param gots: A dict of algorithm names pointing to hashes we - actually got from the files under suspicion - """ - self.allowed = allowed - self.gots = gots - - def body(self) -> str: - return f" {self._requirement_name()}:\n{self._hash_comparison()}" - - def _hash_comparison(self) -> str: - """ - Return a comparison of actual and expected hash values. - - Example:: - - Expected sha256 abcdeabcdeabcdeabcdeabcdeabcdeabcdeabcdeabcde - or 123451234512345123451234512345123451234512345 - Got bcdefbcdefbcdefbcdefbcdefbcdefbcdefbcdefbcdef - - """ - - def hash_then_or(hash_name: str) -> "chain[str]": - # For now, all the decent hashes have 6-char names, so we can get - # away with hard-coding space literals. - return chain([hash_name], repeat(" or")) - - lines: List[str] = [] - for hash_name, expecteds in self.allowed.items(): - prefix = hash_then_or(hash_name) - lines.extend((f" Expected {next(prefix)} {e}") for e in expecteds) - lines.append( - f" Got {self.gots[hash_name].hexdigest()}\n" - ) - return "\n".join(lines) - - -class UnsupportedPythonVersion(InstallationError): - """Unsupported python version according to Requires-Python package - metadata.""" - - -class ConfigurationFileCouldNotBeLoaded(ConfigurationError): - """When there are errors while loading a configuration file""" - - def __init__( - self, - reason: str = "could not be loaded", - fname: Optional[str] = None, - error: Optional[configparser.Error] = None, - ) -> None: - super().__init__(error) - self.reason = reason - self.fname = fname - self.error = error - - def __str__(self) -> str: - if self.fname is not None: - message_part = f" in {self.fname}." - else: - assert self.error is not None - message_part = f".\n{self.error}\n" - return f"Configuration file {self.reason}{message_part}" - - -_DEFAULT_EXTERNALLY_MANAGED_ERROR = f"""\ -The Python environment under {sys.prefix} is managed externally, and may not be -manipulated by the user. Please use specific tooling from the distributor of -the Python installation to interact with this environment instead. -""" - - -class ExternallyManagedEnvironment(DiagnosticPipError): - """The current environment is externally managed. - - This is raised when the current environment is externally managed, as - defined by `PEP 668`_. The ``EXTERNALLY-MANAGED`` configuration is checked - and displayed when the error is bubbled up to the user. - - :param error: The error message read from ``EXTERNALLY-MANAGED``. - """ - - reference = "externally-managed-environment" - - def __init__(self, error: Optional[str]) -> None: - if error is None: - context = Text(_DEFAULT_EXTERNALLY_MANAGED_ERROR) - else: - context = Text(error) - super().__init__( - message="This environment is externally managed", - context=context, - note_stmt=( - "If you believe this is a mistake, please contact your " - "Python installation or OS distribution provider. " - "You can override this, at the risk of breaking your Python " - "installation or OS, by passing --break-system-packages." - ), - hint_stmt=Text("See PEP 668 for the detailed specification."), - ) - - @staticmethod - def _iter_externally_managed_error_keys() -> Iterator[str]: - # LC_MESSAGES is in POSIX, but not the C standard. The most common - # platform that does not implement this category is Windows, where - # using other categories for console message localization is equally - # unreliable, so we fall back to the locale-less vendor message. This - # can always be re-evaluated when a vendor proposes a new alternative. - try: - category = locale.LC_MESSAGES - except AttributeError: - lang: Optional[str] = None - else: - lang, _ = locale.getlocale(category) - if lang is not None: - yield f"Error-{lang}" - for sep in ("-", "_"): - before, found, _ = lang.partition(sep) - if not found: - continue - yield f"Error-{before}" - yield "Error" - - @classmethod - def from_config( - cls, - config: Union[pathlib.Path, str], - ) -> "ExternallyManagedEnvironment": - parser = configparser.ConfigParser(interpolation=None) - try: - parser.read(config, encoding="utf-8") - section = parser["externally-managed"] - for key in cls._iter_externally_managed_error_keys(): - with contextlib.suppress(KeyError): - return cls(section[key]) - except KeyError: - pass - except (OSError, UnicodeDecodeError, configparser.ParsingError): - from pip._internal.utils._log import VERBOSE - - exc_info = logger.isEnabledFor(VERBOSE) - logger.warning("Failed to read %s", config, exc_info=exc_info) - return cls(None) - - -class UninstallMissingRecord(DiagnosticPipError): - reference = "uninstall-no-record-file" - - def __init__(self, *, distribution: "BaseDistribution") -> None: - installer = distribution.installer - if not installer or installer == "pip": - dep = f"{distribution.raw_name}=={distribution.version}" - hint = Text.assemble( - "You might be able to recover from this via: ", - (f"pip install --force-reinstall --no-deps {dep}", "green"), - ) - else: - hint = Text( - f"The package was installed by {installer}. " - "You should check if it can uninstall the package." - ) - - super().__init__( - message=Text(f"Cannot uninstall {distribution}"), - context=( - "The package's contents are unknown: " - f"no RECORD file was found for {distribution.raw_name}." - ), - hint_stmt=hint, - ) - - -class LegacyDistutilsInstall(DiagnosticPipError): - reference = "uninstall-distutils-installed-package" - - def __init__(self, *, distribution: "BaseDistribution") -> None: - super().__init__( - message=Text(f"Cannot uninstall {distribution}"), - context=( - "It is a distutils installed project and thus we cannot accurately " - "determine which files belong to it which would lead to only a partial " - "uninstall." - ), - hint_stmt=None, - ) diff --git a/.env/Lib/site-packages/pip/_internal/index/__init__.py b/.env/Lib/site-packages/pip/_internal/index/__init__.py deleted file mode 100644 index 7a17b7b..0000000 --- a/.env/Lib/site-packages/pip/_internal/index/__init__.py +++ /dev/null @@ -1,2 +0,0 @@ -"""Index interaction code -""" diff --git a/.env/Lib/site-packages/pip/_internal/index/__pycache__/__init__.cpython-312.pyc b/.env/Lib/site-packages/pip/_internal/index/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index b3149c0288eca1bb21b52064766697a075b290a5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 232 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