diff --git a/notes/manifests/internal.sh b/notes/manifests/internal.sh new file mode 100644 index 000000000..462e6d9a2 --- /dev/null +++ b/notes/manifests/internal.sh @@ -0,0 +1,86 @@ +# notes/docker.sh Start docker compose +# notes/postgres.sh Reset database + +INSTANCE=manifests/internal +DATASET=$INSTANCE +BASEDIR=var/instances/$INSTANCE + +mkdir -p $BASEDIR + +export SPINTA_CONFIG=$BASEDIR/config.yml + +cat $BASEDIR/config.yml +cat > $BASEDIR/config.yml < $BASEDIR/manifest.txt <=4.0,<5.0", markers = "python_version < \"3.11\""} +attrs = ">=17.3.0" +frozenlist = ">=1.1.1" +multidict = ">=4.5,<7.0" +yarl = ">=1.0,<2.0" + +[package.extras] +speedups = ["Brotli", "aiodns", "brotlicffi"] + +[[package]] +name = "aiosignal" +version = "1.3.1" +description = "aiosignal: a list of registered asynchronous callbacks" +optional = false +python-versions = ">=3.7" +files = [ + {file = "aiosignal-1.3.1-py3-none-any.whl", hash = "sha256:f8376fb07dd1e86a584e4fcdec80b36b7f81aac666ebc724e2c090300dd83b17"}, + {file = "aiosignal-1.3.1.tar.gz", hash = "sha256:54cd96e15e1649b75d6c87526a6ff0b6c1b0dd3459f43d9ca11d48c339b68cfc"}, +] + +[package.dependencies] +frozenlist = ">=1.1.0" + +[[package]] +name = "alabaster" +version = "0.7.13" +description = "A configurable sidebar-enabled Sphinx theme" +optional = false +python-versions = ">=3.6" +files = [ + {file = "alabaster-0.7.13-py3-none-any.whl", hash = "sha256:1ee19aca801bbabb5ba3f5f258e4422dfa86f82f3e9cefb0859b283cdd7f62a3"}, + {file = "alabaster-0.7.13.tar.gz", hash = "sha256:a27a4a084d5e690e16e01e03ad2b2e552c61a65469419b907243193de1a84ae2"}, +] + +[[package]] +name = "alembic" +version = "1.13.0" +description = "A database migration tool for SQLAlchemy." +optional = false +python-versions = ">=3.8" +files = [ + {file = "alembic-1.13.0-py3-none-any.whl", hash = "sha256:a23974ea301c3ee52705db809c7413cecd165290c6679b9998dd6c74342ca23a"}, + {file = "alembic-1.13.0.tar.gz", hash = "sha256:ab4b3b94d2e1e5f81e34be8a9b7b7575fc9dd5398fccb0bef351ec9b14872623"}, +] + +[package.dependencies] +importlib-metadata = {version = "*", markers = "python_version < \"3.9\""} +importlib-resources = {version = "*", markers = "python_version < \"3.9\""} +Mako = "*" +SQLAlchemy = ">=1.3.0" +typing-extensions = ">=4" + +[package.extras] +tz = ["backports.zoneinfo"] + +[[package]] +name = "anyio" +version = "4.1.0" +description = "High level compatibility layer for multiple asynchronous event loop implementations" +optional = false +python-versions = ">=3.8" +files = [ + {file = "anyio-4.1.0-py3-none-any.whl", hash = "sha256:56a415fbc462291813a94528a779597226619c8e78af7de0507333f700011e5f"}, + {file = "anyio-4.1.0.tar.gz", hash = "sha256:5a0bec7085176715be77df87fc66d6c9d70626bd752fcc85f57cdbee5b3760da"}, +] + +[package.dependencies] +exceptiongroup = {version = ">=1.0.2", markers = "python_version < \"3.11\""} +idna = ">=2.8" +sniffio = ">=1.1" + +[package.extras] +doc = ["Sphinx (>=7)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme"] +test = ["anyio[trio]", "coverage[toml] (>=7)", "exceptiongroup (>=1.2.0)", "hypothesis (>=4.0)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.17)"] +trio = ["trio (>=0.23)"] + +[[package]] +name = "appnope" +version = "0.1.3" +description = "Disable App Nap on macOS >= 10.9" +optional = false +python-versions = "*" +files = [ + {file = "appnope-0.1.3-py2.py3-none-any.whl", hash = "sha256:265a455292d0bd8a72453494fa24df5a11eb18373a60c7c0430889f22548605e"}, + {file = "appnope-0.1.3.tar.gz", hash = "sha256:02bd91c4de869fbb1e1c50aafc4098827a7a54ab2f39d9dcba6c9547ed920e24"}, +] + +[[package]] +name = "asttokens" +version = "2.4.1" +description = "Annotate AST trees with source code positions" +optional = false +python-versions = "*" +files = [ + {file = "asttokens-2.4.1-py2.py3-none-any.whl", hash = "sha256:051ed49c3dcae8913ea7cd08e46a606dba30b79993209636c4875bc1d637bc24"}, + {file = "asttokens-2.4.1.tar.gz", hash = "sha256:b03869718ba9a6eb027e134bfdf69f38a236d681c83c160d510768af11254ba0"}, +] + +[package.dependencies] +six = ">=1.12.0" + +[package.extras] +astroid = ["astroid (>=1,<2)", "astroid (>=2,<4)"] +test = ["astroid (>=1,<2)", "astroid (>=2,<4)", "pytest"] + +[[package]] +name = "async-timeout" +version = "4.0.3" +description = "Timeout context manager for asyncio programs" +optional = false +python-versions = ">=3.7" +files = [ + {file = "async-timeout-4.0.3.tar.gz", hash = "sha256:4640d96be84d82d02ed59ea2b7105a0f7b33abe8703703cd0ab0bf87c427522f"}, + {file = "async_timeout-4.0.3-py3-none-any.whl", hash = "sha256:7405140ff1230c310e51dc27b3145b9092d659ce68ff733fb0cefe3ee42be028"}, +] + +[[package]] +name = "asyncpg" +version = "0.22.0" +description = "An asyncio PostgreSQL driver" +optional = false +python-versions = ">=3.5.0" +files = [ + {file = "asyncpg-0.22.0-cp35-cp35m-macosx_10_14_x86_64.whl", hash = "sha256:ccd75cfb4710c7e8debc19516e2e1d4c9863cce3f7a45a3822980d04b16f4fdd"}, + {file = "asyncpg-0.22.0-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:3af9a8511569983481b5cf94db17b7cbecd06b5398aac9c82e4acb69bb1f4090"}, + {file = "asyncpg-0.22.0-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:d1cb6e5b58a4e017335f2a1886e153a32bd213ffa9f7129ee5aced2a7210fa3c"}, + {file = "asyncpg-0.22.0-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:0f4604a88386d68c46bf7b50c201a9718515b0d2df6d5e9ce024d78ed0f7189c"}, + {file = "asyncpg-0.22.0-cp36-cp36m-win_amd64.whl", hash = "sha256:b37efafbbec505287bd1499a88f4b59ff2b470709a1d8f7e4db198d3e2c5a2c4"}, + {file = "asyncpg-0.22.0-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:1d3efdec14f3fbcc665b77619f8b420564f98b89632a21694be2101dafa6bcf2"}, + {file = "asyncpg-0.22.0-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:f1df7cfd12ef484210717e7827cc2d4d550b16a1b4dd4566c93914c7a2259352"}, + {file = "asyncpg-0.22.0-cp37-cp37m-win_amd64.whl", hash = "sha256:1f514b13bc54bde65db6cd1d0832ae27f21093e3cb66f741e078fab77768971c"}, + {file = "asyncpg-0.22.0-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:82e23ba5b37c0c7ee96f290a95cbf9815b2d29b302e8b9c4af1de9b7759fd27b"}, + {file = "asyncpg-0.22.0-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:062e4ff80e68fe56066c44a8c51989a98785904bf86f49058a242a5887be6ce3"}, + {file = "asyncpg-0.22.0-cp38-cp38-win_amd64.whl", hash = "sha256:e7a67fb0244e4a5b3baaa40092d0efd642da032b5e891d75947dab993b47d925"}, + {file = "asyncpg-0.22.0-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:1bbe5e829de506c743cbd5240b3722e487c53669a5f1e159abcc3b92a64a985e"}, + {file = "asyncpg-0.22.0-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:2cb730241dfe650b9626eae00490cca4cfeb00871ed8b8f389f3a4507b328683"}, + {file = "asyncpg-0.22.0-cp39-cp39-win_amd64.whl", hash = "sha256:2e3875c82ae609b21e562e6befdc35e52c4290e49d03e7529275d59a0595ca97"}, + {file = "asyncpg-0.22.0.tar.gz", hash = "sha256:348ad471d9bdd77f0609a00c860142f47c81c9123f4064d13d65c8569415d802"}, +] + +[package.extras] +dev = ["Cython (>=0.29.20,<0.30.0)", "Sphinx (>=1.7.3,<1.8.0)", "flake8 (>=3.7.9,<3.8.0)", "pycodestyle (>=2.5.0,<2.6.0)", "pytest (>=3.6.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)", "uvloop (>=0.14.0,<0.15.0)"] +docs = ["Sphinx (>=1.7.3,<1.8.0)", "sphinx-rtd-theme (>=0.2.4,<0.3.0)", "sphinxcontrib-asyncio (>=0.2.0,<0.3.0)"] +test = ["flake8 (>=3.7.9,<3.8.0)", "pycodestyle (>=2.5.0,<2.6.0)", "uvloop (>=0.14.0,<0.15.0)"] + +[[package]] +name = "attrs" +version = "23.1.0" +description = "Classes Without Boilerplate" +optional = false +python-versions = ">=3.7" +files = [ + {file = "attrs-23.1.0-py3-none-any.whl", hash = "sha256:1f28b4522cdc2fb4256ac1a020c78acf9cba2c6b461ccd2c126f3aa8e8335d04"}, + {file = "attrs-23.1.0.tar.gz", hash = "sha256:6279836d581513a26f1bf235f9acd333bc9115683f14f7e8fae46c98fc50e015"}, +] + +[package.extras] +cov = ["attrs[tests]", "coverage[toml] (>=5.3)"] +dev = ["attrs[docs,tests]", "pre-commit"] +docs = ["furo", "myst-parser", "sphinx", "sphinx-notfound-page", "sphinxcontrib-towncrier", "towncrier", "zope-interface"] +tests = ["attrs[tests-no-zope]", "zope-interface"] +tests-no-zope = ["cloudpickle", "hypothesis", "mypy (>=1.1.1)", "pympler", "pytest (>=4.3.0)", "pytest-mypy-plugins", "pytest-xdist[psutil]"] + +[[package]] +name = "authlib" +version = "0.11" +description = "The ultimate Python library in building OAuth and OpenID Connect servers." +optional = false +python-versions = "*" +files = [ + {file = "Authlib-0.11-py2.py3-none-any.whl", hash = "sha256:3a226f231e962a16dd5f6fcf0c113235805ba206e294717a64fa8e04ae3ad9c4"}, + {file = "Authlib-0.11.tar.gz", hash = "sha256:9741db6de2950a0a5cefbdb72ec7ab12f7e9fd530ff47219f1530e79183cbaaf"}, +] + +[package.dependencies] +cryptography = "*" +requests = "*" + +[[package]] +name = "babel" +version = "2.14.0" +description = "Internationalization utilities" +optional = false +python-versions = ">=3.7" +files = [ + {file = "Babel-2.14.0-py3-none-any.whl", hash = "sha256:efb1a25b7118e67ce3a259bed20545c29cb68be8ad2c784c83689981b7a57287"}, + {file = "Babel-2.14.0.tar.gz", hash = "sha256:6919867db036398ba21eb5c7a0f6b28ab8cbc3ae7a73a44ebe34ae74a4e7d363"}, +] + +[package.dependencies] +pytz = {version = ">=2015.7", markers = "python_version < \"3.9\""} + +[package.extras] +dev = ["freezegun (>=1.0,<2.0)", "pytest (>=6.0)", "pytest-cov"] + +[[package]] +name = "backcall" +version = "0.2.0" +description = "Specifications for callback functions passed in to an API" +optional = false +python-versions = "*" +files = [ + {file = "backcall-0.2.0-py2.py3-none-any.whl", hash = "sha256:fbbce6a29f263178a1f7915c1940bde0ec2b2a967566fe1c65c1dfb7422bd255"}, + {file = "backcall-0.2.0.tar.gz", hash = "sha256:5cbdbf27be5e7cfadb448baf0aa95508f91f2bbc6c6437cd9cd06e2a4c215e1e"}, +] + +[[package]] +name = "certifi" +version = "2023.11.17" +description = "Python package for providing Mozilla's CA Bundle." +optional = false +python-versions = ">=3.6" +files = [ + {file = "certifi-2023.11.17-py3-none-any.whl", hash = "sha256:e036ab49d5b79556f99cfc2d9320b34cfbe5be05c5871b51de9329f0603b0474"}, + {file = "certifi-2023.11.17.tar.gz", hash = "sha256:9b469f3a900bf28dc19b8cfbf8019bf47f7fdd1a65a1d4ffb98fc14166beb4d1"}, +] + +[[package]] +name = "cffi" +version = "1.16.0" +description = "Foreign Function Interface for Python calling C code." +optional = false +python-versions = ">=3.8" +files = [ + {file = "cffi-1.16.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6b3d6606d369fc1da4fd8c357d026317fbb9c9b75d36dc16e90e84c26854b088"}, + {file = "cffi-1.16.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ac0f5edd2360eea2f1daa9e26a41db02dd4b0451b48f7c318e217ee092a213e9"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7e61e3e4fa664a8588aa25c883eab612a188c725755afff6289454d6362b9673"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a72e8961a86d19bdb45851d8f1f08b041ea37d2bd8d4fd19903bc3083d80c896"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5b50bf3f55561dac5438f8e70bfcdfd74543fd60df5fa5f62d94e5867deca684"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7651c50c8c5ef7bdb41108b7b8c5a83013bfaa8a935590c5d74627c047a583c7"}, + {file = "cffi-1.16.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e4108df7fe9b707191e55f33efbcb2d81928e10cea45527879a4749cbe472614"}, + {file = "cffi-1.16.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:32c68ef735dbe5857c810328cb2481e24722a59a2003018885514d4c09af9743"}, + {file = "cffi-1.16.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:673739cb539f8cdaa07d92d02efa93c9ccf87e345b9a0b556e3ecc666718468d"}, + {file = "cffi-1.16.0-cp310-cp310-win32.whl", hash = "sha256:9f90389693731ff1f659e55c7d1640e2ec43ff725cc61b04b2f9c6d8d017df6a"}, + {file = "cffi-1.16.0-cp310-cp310-win_amd64.whl", hash = "sha256:e6024675e67af929088fda399b2094574609396b1decb609c55fa58b028a32a1"}, + {file = "cffi-1.16.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b84834d0cf97e7d27dd5b7f3aca7b6e9263c56308ab9dc8aae9784abb774d404"}, + {file = "cffi-1.16.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1b8ebc27c014c59692bb2664c7d13ce7a6e9a629be20e54e7271fa696ff2b417"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ee07e47c12890ef248766a6e55bd38ebfb2bb8edd4142d56db91b21ea68b7627"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d8a9d3ebe49f084ad71f9269834ceccbf398253c9fac910c4fd7053ff1386936"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e70f54f1796669ef691ca07d046cd81a29cb4deb1e5f942003f401c0c4a2695d"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5bf44d66cdf9e893637896c7faa22298baebcd18d1ddb6d2626a6e39793a1d56"}, + {file = "cffi-1.16.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7b78010e7b97fef4bee1e896df8a4bbb6712b7f05b7ef630f9d1da00f6444d2e"}, + {file = "cffi-1.16.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:c6a164aa47843fb1b01e941d385aab7215563bb8816d80ff3a363a9f8448a8dc"}, + {file = "cffi-1.16.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e09f3ff613345df5e8c3667da1d918f9149bd623cd9070c983c013792a9a62eb"}, + {file = "cffi-1.16.0-cp311-cp311-win32.whl", hash = "sha256:2c56b361916f390cd758a57f2e16233eb4f64bcbeee88a4881ea90fca14dc6ab"}, + {file = "cffi-1.16.0-cp311-cp311-win_amd64.whl", hash = "sha256:db8e577c19c0fda0beb7e0d4e09e0ba74b1e4c092e0e40bfa12fe05b6f6d75ba"}, + {file = "cffi-1.16.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:fa3a0128b152627161ce47201262d3140edb5a5c3da88d73a1b790a959126956"}, + {file = "cffi-1.16.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:68e7c44931cc171c54ccb702482e9fc723192e88d25a0e133edd7aff8fcd1f6e"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:abd808f9c129ba2beda4cfc53bde801e5bcf9d6e0f22f095e45327c038bfe68e"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:88e2b3c14bdb32e440be531ade29d3c50a1a59cd4e51b1dd8b0865c54ea5d2e2"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fcc8eb6d5902bb1cf6dc4f187ee3ea80a1eba0a89aba40a5cb20a5087d961357"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b7be2d771cdba2942e13215c4e340bfd76398e9227ad10402a8767ab1865d2e6"}, + {file = "cffi-1.16.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e715596e683d2ce000574bae5d07bd522c781a822866c20495e52520564f0969"}, + {file = "cffi-1.16.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:2d92b25dbf6cae33f65005baf472d2c245c050b1ce709cc4588cdcdd5495b520"}, + {file = "cffi-1.16.0-cp312-cp312-win32.whl", hash = "sha256:b2ca4e77f9f47c55c194982e10f058db063937845bb2b7a86c84a6cfe0aefa8b"}, + {file = "cffi-1.16.0-cp312-cp312-win_amd64.whl", hash = "sha256:68678abf380b42ce21a5f2abde8efee05c114c2fdb2e9eef2efdb0257fba1235"}, + {file = "cffi-1.16.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0c9ef6ff37e974b73c25eecc13952c55bceed9112be2d9d938ded8e856138bcc"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a09582f178759ee8128d9270cd1344154fd473bb77d94ce0aeb2a93ebf0feaf0"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e760191dd42581e023a68b758769e2da259b5d52e3103c6060ddc02c9edb8d7b"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:80876338e19c951fdfed6198e70bc88f1c9758b94578d5a7c4c91a87af3cf31c"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a6a14b17d7e17fa0d207ac08642c8820f84f25ce17a442fd15e27ea18d67c59b"}, + {file = "cffi-1.16.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6602bc8dc6f3a9e02b6c22c4fc1e47aa50f8f8e6d3f78a5e16ac33ef5fefa324"}, + {file = "cffi-1.16.0-cp38-cp38-win32.whl", hash = "sha256:131fd094d1065b19540c3d72594260f118b231090295d8c34e19a7bbcf2e860a"}, + {file = "cffi-1.16.0-cp38-cp38-win_amd64.whl", hash = "sha256:31d13b0f99e0836b7ff893d37af07366ebc90b678b6664c955b54561fc36ef36"}, + {file = "cffi-1.16.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:582215a0e9adbe0e379761260553ba11c58943e4bbe9c36430c4ca6ac74b15ed"}, + {file = "cffi-1.16.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:b29ebffcf550f9da55bec9e02ad430c992a87e5f512cd63388abb76f1036d8d2"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dc9b18bf40cc75f66f40a7379f6a9513244fe33c0e8aa72e2d56b0196a7ef872"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9cb4a35b3642fc5c005a6755a5d17c6c8b6bcb6981baf81cea8bfbc8903e8ba8"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b86851a328eedc692acf81fb05444bdf1891747c25af7529e39ddafaf68a4f3f"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c0f31130ebc2d37cdd8e44605fb5fa7ad59049298b3f745c74fa74c62fbfcfc4"}, + {file = "cffi-1.16.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8f8e709127c6c77446a8c0a8c8bf3c8ee706a06cd44b1e827c3e6a2ee6b8c098"}, + {file = "cffi-1.16.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:748dcd1e3d3d7cd5443ef03ce8685043294ad6bd7c02a38d1bd367cfd968e000"}, + {file = "cffi-1.16.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:8895613bcc094d4a1b2dbe179d88d7fb4a15cee43c052e8885783fac397d91fe"}, + {file = "cffi-1.16.0-cp39-cp39-win32.whl", hash = "sha256:ed86a35631f7bfbb28e108dd96773b9d5a6ce4811cf6ea468bb6a359b256b1e4"}, + {file = "cffi-1.16.0-cp39-cp39-win_amd64.whl", hash = "sha256:3686dffb02459559c74dd3d81748269ffb0eb027c39a6fc99502de37d501faa8"}, + {file = "cffi-1.16.0.tar.gz", hash = "sha256:bcb3ef43e58665bbda2fb198698fcae6776483e0c4a631aa5647806c25e02cc0"}, +] + +[package.dependencies] +pycparser = "*" + +[[package]] +name = "chardet" +version = "5.2.0" +description = "Universal encoding detector for Python 3" +optional = false +python-versions = ">=3.7" +files = [ + {file = "chardet-5.2.0-py3-none-any.whl", hash = "sha256:e1cf59446890a00105fe7b7912492ea04b6e6f06d4b742b2c788469e34c82970"}, + {file = "chardet-5.2.0.tar.gz", hash = "sha256:1b3b6ff479a8c414bc3fa2c0852995695c4a026dcd6d0633b2dd092ca39c1cf7"}, +] + +[[package]] +name = "charset-normalizer" +version = "3.3.2" +description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "charset-normalizer-3.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win32.whl", hash = "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win32.whl", hash = "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win32.whl", hash = "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win32.whl", hash = "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win32.whl", hash = "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d"}, + {file = "charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc"}, +] + +[[package]] +name = "cheap-repr" +version = "0.5.1" +description = "Better version of repr/reprlib for short, cheap string representations." +optional = false +python-versions = "*" +files = [ + {file = "cheap_repr-0.5.1-py2.py3-none-any.whl", hash = "sha256:30096998aeb49367a4a153988d7a99dce9dc59bbdd4b19740da6b4f3f97cf2ff"}, + {file = "cheap_repr-0.5.1.tar.gz", hash = "sha256:31ec63b9d8394aa23d746c8376c8307f75f9fca0b983566b8bcf13cc661fe6dd"}, +] + +[package.extras] +tests = ["Django", "Django (<2)", "Django (<3)", "chainmap", "numpy (>=1.16.3)", "numpy (>=1.16.3,<1.17)", "numpy (>=1.16.3,<1.19)", "pandas (>=0.24.2)", "pandas (>=0.24.2,<0.25)", "pandas (>=0.24.2,<0.26)", "pytest"] + +[[package]] +name = "click" +version = "8.1.7" +description = "Composable command line interface toolkit" +optional = false +python-versions = ">=3.7" +files = [ + {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, + {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[[package]] +name = "cloudpickle" +version = "3.0.0" +description = "Pickler class to extend the standard pickle.Pickler functionality" +optional = false +python-versions = ">=3.8" +files = [ + {file = "cloudpickle-3.0.0-py3-none-any.whl", hash = "sha256:246ee7d0c295602a036e86369c77fecda4ab17b506496730f2f576d9016fd9c7"}, + {file = "cloudpickle-3.0.0.tar.gz", hash = "sha256:996d9a482c6fb4f33c1a35335cf8afd065d2a56e973270364840712d9131a882"}, +] + +[[package]] +name = "colorama" +version = "0.4.6" +description = "Cross-platform colored terminal text." +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" +files = [ + {file = "colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6"}, + {file = "colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44"}, +] + +[[package]] +name = "commonmark" +version = "0.9.1" +description = "Python parser for the CommonMark Markdown spec" +optional = false +python-versions = "*" +files = [ + {file = "commonmark-0.9.1-py2.py3-none-any.whl", hash = "sha256:da2f38c92590f83de410ba1a3cbceafbc74fee9def35f9251ba9a971d6d66fd9"}, + {file = "commonmark-0.9.1.tar.gz", hash = "sha256:452f9dc859be7f06631ddcb328b6919c67984aca654e5fefb3914d54691aed60"}, +] + +[package.extras] +test = ["flake8 (==3.7.8)", "hypothesis (==3.55.3)"] + +[[package]] +name = "coverage" +version = "7.3.2" +description = "Code coverage measurement for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "coverage-7.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d872145f3a3231a5f20fd48500274d7df222e291d90baa2026cc5152b7ce86bf"}, + {file = "coverage-7.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:310b3bb9c91ea66d59c53fa4989f57d2436e08f18fb2f421a1b0b6b8cc7fffda"}, + {file = "coverage-7.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f47d39359e2c3779c5331fc740cf4bce6d9d680a7b4b4ead97056a0ae07cb49a"}, + {file = "coverage-7.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:aa72dbaf2c2068404b9870d93436e6d23addd8bbe9295f49cbca83f6e278179c"}, + {file = "coverage-7.3.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:beaa5c1b4777f03fc63dfd2a6bd820f73f036bfb10e925fce067b00a340d0f3f"}, + {file = "coverage-7.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:dbc1b46b92186cc8074fee9d9fbb97a9dd06c6cbbef391c2f59d80eabdf0faa6"}, + {file = "coverage-7.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:315a989e861031334d7bee1f9113c8770472db2ac484e5b8c3173428360a9148"}, + {file = "coverage-7.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:d1bc430677773397f64a5c88cb522ea43175ff16f8bfcc89d467d974cb2274f9"}, + {file = "coverage-7.3.2-cp310-cp310-win32.whl", hash = "sha256:a889ae02f43aa45032afe364c8ae84ad3c54828c2faa44f3bfcafecb5c96b02f"}, + {file = "coverage-7.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:c0ba320de3fb8c6ec16e0be17ee1d3d69adcda99406c43c0409cb5c41788a611"}, + {file = "coverage-7.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ac8c802fa29843a72d32ec56d0ca792ad15a302b28ca6203389afe21f8fa062c"}, + {file = "coverage-7.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:89a937174104339e3a3ffcf9f446c00e3a806c28b1841c63edb2b369310fd074"}, + {file = "coverage-7.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e267e9e2b574a176ddb983399dec325a80dbe161f1a32715c780b5d14b5f583a"}, + {file = "coverage-7.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2443cbda35df0d35dcfb9bf8f3c02c57c1d6111169e3c85fc1fcc05e0c9f39a3"}, + {file = "coverage-7.3.2-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4175e10cc8dda0265653e8714b3174430b07c1dca8957f4966cbd6c2b1b8065a"}, + {file = "coverage-7.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0cbf38419fb1a347aaf63481c00f0bdc86889d9fbf3f25109cf96c26b403fda1"}, + {file = "coverage-7.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:5c913b556a116b8d5f6ef834038ba983834d887d82187c8f73dec21049abd65c"}, + {file = "coverage-7.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:1981f785239e4e39e6444c63a98da3a1db8e971cb9ceb50a945ba6296b43f312"}, + {file = "coverage-7.3.2-cp311-cp311-win32.whl", hash = "sha256:43668cabd5ca8258f5954f27a3aaf78757e6acf13c17604d89648ecc0cc66640"}, + {file = "coverage-7.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:e10c39c0452bf6e694511c901426d6b5ac005acc0f78ff265dbe36bf81f808a2"}, + {file = "coverage-7.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:4cbae1051ab791debecc4a5dcc4a1ff45fc27b91b9aee165c8a27514dd160836"}, + {file = "coverage-7.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:12d15ab5833a997716d76f2ac1e4b4d536814fc213c85ca72756c19e5a6b3d63"}, + {file = "coverage-7.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3c7bba973ebee5e56fe9251300c00f1579652587a9f4a5ed8404b15a0471f216"}, + {file = "coverage-7.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:fe494faa90ce6381770746077243231e0b83ff3f17069d748f645617cefe19d4"}, + {file = "coverage-7.3.2-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f6e9589bd04d0461a417562649522575d8752904d35c12907d8c9dfeba588faf"}, + {file = "coverage-7.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d51ac2a26f71da1b57f2dc81d0e108b6ab177e7d30e774db90675467c847bbdf"}, + {file = "coverage-7.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:99b89d9f76070237975b315b3d5f4d6956ae354a4c92ac2388a5695516e47c84"}, + {file = "coverage-7.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:fa28e909776dc69efb6ed975a63691bc8172b64ff357e663a1bb06ff3c9b589a"}, + {file = "coverage-7.3.2-cp312-cp312-win32.whl", hash = "sha256:289fe43bf45a575e3ab10b26d7b6f2ddb9ee2dba447499f5401cfb5ecb8196bb"}, + {file = "coverage-7.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:7dbc3ed60e8659bc59b6b304b43ff9c3ed858da2839c78b804973f613d3e92ed"}, + {file = "coverage-7.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:f94b734214ea6a36fe16e96a70d941af80ff3bfd716c141300d95ebc85339738"}, + {file = "coverage-7.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:af3d828d2c1cbae52d34bdbb22fcd94d1ce715d95f1a012354a75e5913f1bda2"}, + {file = "coverage-7.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:630b13e3036e13c7adc480ca42fa7afc2a5d938081d28e20903cf7fd687872e2"}, + {file = "coverage-7.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c9eacf273e885b02a0273bb3a2170f30e2d53a6d53b72dbe02d6701b5296101c"}, + {file = "coverage-7.3.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d8f17966e861ff97305e0801134e69db33b143bbfb36436efb9cfff6ec7b2fd9"}, + {file = "coverage-7.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:b4275802d16882cf9c8b3d057a0839acb07ee9379fa2749eca54efbce1535b82"}, + {file = "coverage-7.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:72c0cfa5250f483181e677ebc97133ea1ab3eb68645e494775deb6a7f6f83901"}, + {file = "coverage-7.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:cb536f0dcd14149425996821a168f6e269d7dcd2c273a8bff8201e79f5104e76"}, + {file = "coverage-7.3.2-cp38-cp38-win32.whl", hash = "sha256:307adb8bd3abe389a471e649038a71b4eb13bfd6b7dd9a129fa856f5c695cf92"}, + {file = "coverage-7.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:88ed2c30a49ea81ea3b7f172e0269c182a44c236eb394718f976239892c0a27a"}, + {file = "coverage-7.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:b631c92dfe601adf8f5ebc7fc13ced6bb6e9609b19d9a8cd59fa47c4186ad1ce"}, + {file = "coverage-7.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:d3d9df4051c4a7d13036524b66ecf7a7537d14c18a384043f30a303b146164e9"}, + {file = "coverage-7.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5f7363d3b6a1119ef05015959ca24a9afc0ea8a02c687fe7e2d557705375c01f"}, + {file = "coverage-7.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2f11cc3c967a09d3695d2a6f03fb3e6236622b93be7a4b5dc09166a861be6d25"}, + {file = "coverage-7.3.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:149de1d2401ae4655c436a3dced6dd153f4c3309f599c3d4bd97ab172eaf02d9"}, + {file = "coverage-7.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:3a4006916aa6fee7cd38db3bfc95aa9c54ebb4ffbfc47c677c8bba949ceba0a6"}, + {file = "coverage-7.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9028a3871280110d6e1aa2df1afd5ef003bab5fb1ef421d6dc748ae1c8ef2ebc"}, + {file = "coverage-7.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:9f805d62aec8eb92bab5b61c0f07329275b6f41c97d80e847b03eb894f38d083"}, + {file = "coverage-7.3.2-cp39-cp39-win32.whl", hash = "sha256:d1c88ec1a7ff4ebca0219f5b1ef863451d828cccf889c173e1253aa84b1e07ce"}, + {file = "coverage-7.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b4767da59464bb593c07afceaddea61b154136300881844768037fd5e859353f"}, + {file = "coverage-7.3.2-pp38.pp39.pp310-none-any.whl", hash = "sha256:ae97af89f0fbf373400970c0a21eef5aa941ffeed90aee43650b81f7d7f47637"}, + {file = "coverage-7.3.2.tar.gz", hash = "sha256:be32ad29341b0170e795ca590e1c07e81fc061cb5b10c74ce7203491484404ef"}, +] + +[package.dependencies] +tomli = {version = "*", optional = true, markers = "python_full_version <= \"3.11.0a6\" and extra == \"toml\""} + +[package.extras] +toml = ["tomli"] + +[[package]] +name = "cryptography" +version = "41.0.7" +description = "cryptography is a package which provides cryptographic recipes and primitives to Python developers." +optional = false +python-versions = ">=3.7" +files = [ + {file = "cryptography-41.0.7-cp37-abi3-macosx_10_12_universal2.whl", hash = "sha256:3c78451b78313fa81607fa1b3f1ae0a5ddd8014c38a02d9db0616133987b9cdf"}, + {file = "cryptography-41.0.7-cp37-abi3-macosx_10_12_x86_64.whl", hash = "sha256:928258ba5d6f8ae644e764d0f996d61a8777559f72dfeb2eea7e2fe0ad6e782d"}, + {file = "cryptography-41.0.7-cp37-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5a1b41bc97f1ad230a41657d9155113c7521953869ae57ac39ac7f1bb471469a"}, + {file = "cryptography-41.0.7-cp37-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:841df4caa01008bad253bce2a6f7b47f86dc9f08df4b433c404def869f590a15"}, + {file = "cryptography-41.0.7-cp37-abi3-manylinux_2_28_aarch64.whl", hash = "sha256:5429ec739a29df2e29e15d082f1d9ad683701f0ec7709ca479b3ff2708dae65a"}, + {file = "cryptography-41.0.7-cp37-abi3-manylinux_2_28_x86_64.whl", hash = "sha256:43f2552a2378b44869fe8827aa19e69512e3245a219104438692385b0ee119d1"}, + {file = "cryptography-41.0.7-cp37-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:af03b32695b24d85a75d40e1ba39ffe7db7ffcb099fe507b39fd41a565f1b157"}, + {file = "cryptography-41.0.7-cp37-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:49f0805fc0b2ac8d4882dd52f4a3b935b210935d500b6b805f321addc8177406"}, + {file = "cryptography-41.0.7-cp37-abi3-win32.whl", hash = "sha256:f983596065a18a2183e7f79ab3fd4c475205b839e02cbc0efbbf9666c4b3083d"}, + {file = "cryptography-41.0.7-cp37-abi3-win_amd64.whl", hash = "sha256:90452ba79b8788fa380dfb587cca692976ef4e757b194b093d845e8d99f612f2"}, + {file = "cryptography-41.0.7-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:079b85658ea2f59c4f43b70f8119a52414cdb7be34da5d019a77bf96d473b960"}, + {file = "cryptography-41.0.7-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:b640981bf64a3e978a56167594a0e97db71c89a479da8e175d8bb5be5178c003"}, + {file = "cryptography-41.0.7-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:e3114da6d7f95d2dee7d3f4eec16dacff819740bbab931aff8648cb13c5ff5e7"}, + {file = "cryptography-41.0.7-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:d5ec85080cce7b0513cfd233914eb8b7bbd0633f1d1703aa28d1dd5a72f678ec"}, + {file = "cryptography-41.0.7-pp38-pypy38_pp73-macosx_10_12_x86_64.whl", hash = "sha256:7a698cb1dac82c35fcf8fe3417a3aaba97de16a01ac914b89a0889d364d2f6be"}, + {file = "cryptography-41.0.7-pp38-pypy38_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:37a138589b12069efb424220bf78eac59ca68b95696fc622b6ccc1c0a197204a"}, + {file = "cryptography-41.0.7-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:68a2dec79deebc5d26d617bfdf6e8aab065a4f34934b22d3b5010df3ba36612c"}, + {file = "cryptography-41.0.7-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:09616eeaef406f99046553b8a40fbf8b1e70795a91885ba4c96a70793de5504a"}, + {file = "cryptography-41.0.7-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:48a0476626da912a44cc078f9893f292f0b3e4c739caf289268168d8f4702a39"}, + {file = "cryptography-41.0.7-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:c7f3201ec47d5207841402594f1d7950879ef890c0c495052fa62f58283fde1a"}, + {file = "cryptography-41.0.7-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:c5ca78485a255e03c32b513f8c2bc39fedb7f5c5f8535545bdc223a03b24f248"}, + {file = "cryptography-41.0.7-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:d6c391c021ab1f7a82da5d8d0b3cee2f4b2c455ec86c8aebbc84837a631ff309"}, + {file = "cryptography-41.0.7.tar.gz", hash = "sha256:13f93ce9bea8016c253b34afc6bd6a75993e5c40672ed5405a9c832f0d4a00bc"}, +] + +[package.dependencies] +cffi = ">=1.12" + +[package.extras] +docs = ["sphinx (>=5.3.0)", "sphinx-rtd-theme (>=1.1.1)"] +docstest = ["pyenchant (>=1.6.11)", "sphinxcontrib-spelling (>=4.0.1)", "twine (>=1.12.0)"] +nox = ["nox"] +pep8test = ["black", "check-sdist", "mypy", "ruff"] +sdist = ["build"] +ssh = ["bcrypt (>=3.1.5)"] +test = ["pretend", "pytest (>=6.2.0)", "pytest-benchmark", "pytest-cov", "pytest-xdist"] +test-randomorder = ["pytest-randomly"] + +[[package]] +name = "cssselect" +version = "1.2.0" +description = "cssselect parses CSS3 Selectors and translates them to XPath 1.0" +optional = false +python-versions = ">=3.7" +files = [ + {file = "cssselect-1.2.0-py2.py3-none-any.whl", hash = "sha256:da1885f0c10b60c03ed5eccbb6b68d6eff248d91976fcde348f395d54c9fd35e"}, + {file = "cssselect-1.2.0.tar.gz", hash = "sha256:666b19839cfaddb9ce9d36bfe4c969132c647b92fc9088c4e23f786b30f1b3dc"}, +] + +[[package]] +name = "dask" +version = "2023.5.0" +description = "Parallel PyData with Task Scheduling" +optional = false +python-versions = ">=3.8" +files = [ + {file = "dask-2023.5.0-py3-none-any.whl", hash = "sha256:32b34986519b7ddc0947c8ca63c2fc81b964e4c208dfb5cbf9f4f8aec92d152b"}, + {file = "dask-2023.5.0.tar.gz", hash = "sha256:4f4c28ac406e81b8f21b5be4b31b21308808f3e0e7c7e2f4a914f16476d9941b"}, +] + +[package.dependencies] +click = ">=8.0" +cloudpickle = ">=1.5.0" +fsspec = ">=2021.09.0" +importlib-metadata = ">=4.13.0" +numpy = {version = ">=1.21", optional = true, markers = "extra == \"dataframe\""} +packaging = ">=20.0" +pandas = {version = ">=1.3", optional = true, markers = "extra == \"dataframe\""} +partd = ">=1.2.0" +pyyaml = ">=5.3.1" +toolz = ">=0.10.0" + +[package.extras] +array = ["numpy (>=1.21)"] +complete = ["dask[array,dataframe,diagnostics,distributed]", "lz4 (>=4.3.2)", "pyarrow (>=7.0)"] +dataframe = ["numpy (>=1.21)", "pandas (>=1.3)"] +diagnostics = ["bokeh (>=2.4.2)", "jinja2 (>=2.10.3)"] +distributed = ["distributed (==2023.5.0)"] +test = ["pandas[test]", "pre-commit", "pytest", "pytest-rerunfailures", "pytest-xdist"] + +[[package]] +name = "decorator" +version = "5.1.1" +description = "Decorators for Humans" +optional = false +python-versions = ">=3.5" +files = [ + {file = "decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186"}, + {file = "decorator-5.1.1.tar.gz", hash = "sha256:637996211036b6385ef91435e4fae22989472f9d571faba8927ba8253acbc330"}, +] + +[[package]] +name = "dnspython" +version = "2.4.2" +description = "DNS toolkit" +optional = false +python-versions = ">=3.8,<4.0" +files = [ + {file = "dnspython-2.4.2-py3-none-any.whl", hash = "sha256:57c6fbaaeaaf39c891292012060beb141791735dbb4004798328fc2c467402d8"}, + {file = "dnspython-2.4.2.tar.gz", hash = "sha256:8dcfae8c7460a2f84b4072e26f1c9f4101ca20c071649cb7c34e8b6a93d58984"}, +] + +[package.extras] +dnssec = ["cryptography (>=2.6,<42.0)"] +doh = ["h2 (>=4.1.0)", "httpcore (>=0.17.3)", "httpx (>=0.24.1)"] +doq = ["aioquic (>=0.9.20)"] +idna = ["idna (>=2.1,<4.0)"] +trio = ["trio (>=0.14,<0.23)"] +wmi = ["wmi (>=1.5.1,<2.0.0)"] + +[[package]] +name = "docutils" +version = "0.18.1" +description = "Docutils -- Python Documentation Utilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "docutils-0.18.1-py2.py3-none-any.whl", hash = "sha256:23010f129180089fbcd3bc08cfefccb3b890b0050e1ca00c867036e9d161b98c"}, + {file = "docutils-0.18.1.tar.gz", hash = "sha256:679987caf361a7539d76e584cbeddc311e3aee937877c87346f31debc63e9d06"}, +] + +[[package]] +name = "et-xmlfile" +version = "1.1.0" +description = "An implementation of lxml.xmlfile for the standard library" +optional = false +python-versions = ">=3.6" +files = [ + {file = "et_xmlfile-1.1.0-py3-none-any.whl", hash = "sha256:a2ba85d1d6a74ef63837eed693bcb89c3f752169b0e3e7ae5b16ca5e1b3deada"}, + {file = "et_xmlfile-1.1.0.tar.gz", hash = "sha256:8eb9e2bc2f8c97e37a2dc85a09ecdcdec9d8a396530a6d5a33b30b9a92da0c5c"}, +] + +[[package]] +name = "exceptiongroup" +version = "1.2.0" +description = "Backport of PEP 654 (exception groups)" +optional = false +python-versions = ">=3.7" +files = [ + {file = "exceptiongroup-1.2.0-py3-none-any.whl", hash = "sha256:4bfd3996ac73b41e9b9628b04e079f193850720ea5945fc96a08633c66912f14"}, + {file = "exceptiongroup-1.2.0.tar.gz", hash = "sha256:91f5c769735f051a4290d52edd0858999b57e5876e9f85937691bd4c9fa3ed68"}, +] + +[package.extras] +test = ["pytest (>=6)"] + +[[package]] +name = "executing" +version = "2.0.1" +description = "Get the currently executing AST node of a frame, and other information" +optional = false +python-versions = ">=3.5" +files = [ + {file = "executing-2.0.1-py2.py3-none-any.whl", hash = "sha256:eac49ca94516ccc753f9fb5ce82603156e590b27525a8bc32cce8ae302eb61bc"}, + {file = "executing-2.0.1.tar.gz", hash = "sha256:35afe2ce3affba8ee97f2d69927fa823b08b472b7b994e36a52a964b93d16147"}, +] + +[package.extras] +tests = ["asttokens (>=2.1.0)", "coverage", "coverage-enable-subprocess", "ipython", "littleutils", "pytest", "rich"] + +[[package]] +name = "frictionless" +version = "3.48.0" +description = "Frictionless is a framework to describe, extract, validate, and transform tabular data" +optional = false +python-versions = "*" +files = [ + {file = "frictionless-3.48.0-py2.py3-none-any.whl", hash = "sha256:8b772e6669c7d74b5800bfd3276f3729b7b3587e51f9ee78c04b439b34d10848"}, + {file = "frictionless-3.48.0.tar.gz", hash = "sha256:3a3331db5c2b4e22b65cc04dcdc86bfdcf874aa9f92db6601fb364735d08fb97"}, +] + +[package.dependencies] +chardet = ">=3.0" +isodate = ">=0.6" +jsonschema = ">=2.5" +petl = ">=1.6" +python-dateutil = ">=2.8" +python-slugify = ">=1.2" +pyyaml = ">=5.3" +requests = ">=2.10" +simpleeval = ">=0.9" +sqlalchemy = {version = ">=1.3", optional = true, markers = "extra == \"sql\""} +stringcase = ">=1.2" +typer = {version = ">=0.3", extras = ["all"]} +validators = ">=0.18" + +[package.extras] +bigquery = ["google-api-python-client (>=1.12.1)"] +ckan = ["ckanapi (>=4.3)"] +dataflows = ["dataflows (>=0.1)"] +dev = ["black", "docstring-parser", "ipykernel", "ipython", "jinja2", "moto", "mypy", "nbconvert", "notedown", "oauth2client", "psycopg2", "pylama", "pymysql", "pytest", "pytest-cov", "pytest-vcr", "python-dotenv", "requests-mock"] +excel = ["openpyxl (>=3.0)", "xlrd (>=1.2)", "xlwt (>=1.2)"] +gsheets = ["pygsheets (>=2.0)"] +html = ["pyquery (>=1.4)"] +json = ["ijson (>=3.0)", "jsonlines (>=1.2)"] +ods = ["ezodf (>=0.3)"] +pandas = ["pandas (>=1.0)"] +s3 = ["boto3 (>=1.9)"] +server = ["flask (>=1.1)", "gunicorn (>=20.0)"] +spss = ["savReaderWriter (>=3.0)"] +sql = ["sqlalchemy (>=1.3)"] + +[[package]] +name = "frozenlist" +version = "1.4.0" +description = "A list-like structure which implements collections.abc.MutableSequence" +optional = false +python-versions = ">=3.8" +files = [ + {file = "frozenlist-1.4.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:764226ceef3125e53ea2cb275000e309c0aa5464d43bd72abd661e27fffc26ab"}, + {file = "frozenlist-1.4.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d6484756b12f40003c6128bfcc3fa9f0d49a687e171186c2d85ec82e3758c559"}, + {file = "frozenlist-1.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9ac08e601308e41eb533f232dbf6b7e4cea762f9f84f6357136eed926c15d12c"}, + {file = "frozenlist-1.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d081f13b095d74b67d550de04df1c756831f3b83dc9881c38985834387487f1b"}, + {file = "frozenlist-1.4.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:71932b597f9895f011f47f17d6428252fc728ba2ae6024e13c3398a087c2cdea"}, + {file = "frozenlist-1.4.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:981b9ab5a0a3178ff413bca62526bb784249421c24ad7381e39d67981be2c326"}, + {file = "frozenlist-1.4.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e41f3de4df3e80de75845d3e743b3f1c4c8613c3997a912dbf0229fc61a8b963"}, + {file = "frozenlist-1.4.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6918d49b1f90821e93069682c06ffde41829c346c66b721e65a5c62b4bab0300"}, + {file = "frozenlist-1.4.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:0e5c8764c7829343d919cc2dfc587a8db01c4f70a4ebbc49abde5d4b158b007b"}, + {file = "frozenlist-1.4.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:8d0edd6b1c7fb94922bf569c9b092ee187a83f03fb1a63076e7774b60f9481a8"}, + {file = "frozenlist-1.4.0-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:e29cda763f752553fa14c68fb2195150bfab22b352572cb36c43c47bedba70eb"}, + {file = "frozenlist-1.4.0-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:0c7c1b47859ee2cac3846fde1c1dc0f15da6cec5a0e5c72d101e0f83dcb67ff9"}, + {file = "frozenlist-1.4.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:901289d524fdd571be1c7be054f48b1f88ce8dddcbdf1ec698b27d4b8b9e5d62"}, + {file = "frozenlist-1.4.0-cp310-cp310-win32.whl", hash = "sha256:1a0848b52815006ea6596c395f87449f693dc419061cc21e970f139d466dc0a0"}, + {file = "frozenlist-1.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:b206646d176a007466358aa21d85cd8600a415c67c9bd15403336c331a10d956"}, + {file = "frozenlist-1.4.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:de343e75f40e972bae1ef6090267f8260c1446a1695e77096db6cfa25e759a95"}, + {file = "frozenlist-1.4.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ad2a9eb6d9839ae241701d0918f54c51365a51407fd80f6b8289e2dfca977cc3"}, + {file = "frozenlist-1.4.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:bd7bd3b3830247580de99c99ea2a01416dfc3c34471ca1298bccabf86d0ff4dc"}, + {file = "frozenlist-1.4.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bdf1847068c362f16b353163391210269e4f0569a3c166bc6a9f74ccbfc7e839"}, + {file = "frozenlist-1.4.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:38461d02d66de17455072c9ba981d35f1d2a73024bee7790ac2f9e361ef1cd0c"}, + {file = "frozenlist-1.4.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d5a32087d720c608f42caed0ef36d2b3ea61a9d09ee59a5142d6070da9041b8f"}, + {file = "frozenlist-1.4.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dd65632acaf0d47608190a71bfe46b209719bf2beb59507db08ccdbe712f969b"}, + {file = "frozenlist-1.4.0-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:261b9f5d17cac914531331ff1b1d452125bf5daa05faf73b71d935485b0c510b"}, + {file = "frozenlist-1.4.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:b89ac9768b82205936771f8d2eb3ce88503b1556324c9f903e7156669f521472"}, + {file = "frozenlist-1.4.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:008eb8b31b3ea6896da16c38c1b136cb9fec9e249e77f6211d479db79a4eaf01"}, + {file = "frozenlist-1.4.0-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:e74b0506fa5aa5598ac6a975a12aa8928cbb58e1f5ac8360792ef15de1aa848f"}, + {file = "frozenlist-1.4.0-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:490132667476f6781b4c9458298b0c1cddf237488abd228b0b3650e5ecba7467"}, + {file = "frozenlist-1.4.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:76d4711f6f6d08551a7e9ef28c722f4a50dd0fc204c56b4bcd95c6cc05ce6fbb"}, + {file = "frozenlist-1.4.0-cp311-cp311-win32.whl", hash = "sha256:a02eb8ab2b8f200179b5f62b59757685ae9987996ae549ccf30f983f40602431"}, + {file = "frozenlist-1.4.0-cp311-cp311-win_amd64.whl", hash = "sha256:515e1abc578dd3b275d6a5114030b1330ba044ffba03f94091842852f806f1c1"}, + {file = "frozenlist-1.4.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:f0ed05f5079c708fe74bf9027e95125334b6978bf07fd5ab923e9e55e5fbb9d3"}, + {file = "frozenlist-1.4.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ca265542ca427bf97aed183c1676e2a9c66942e822b14dc6e5f42e038f92a503"}, + {file = "frozenlist-1.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:491e014f5c43656da08958808588cc6c016847b4360e327a62cb308c791bd2d9"}, + {file = "frozenlist-1.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:17ae5cd0f333f94f2e03aaf140bb762c64783935cc764ff9c82dff626089bebf"}, + {file = "frozenlist-1.4.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1e78fb68cf9c1a6aa4a9a12e960a5c9dfbdb89b3695197aa7064705662515de2"}, + {file = "frozenlist-1.4.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d5655a942f5f5d2c9ed93d72148226d75369b4f6952680211972a33e59b1dfdc"}, + {file = "frozenlist-1.4.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c11b0746f5d946fecf750428a95f3e9ebe792c1ee3b1e96eeba145dc631a9672"}, + {file = "frozenlist-1.4.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e66d2a64d44d50d2543405fb183a21f76b3b5fd16f130f5c99187c3fb4e64919"}, + {file = "frozenlist-1.4.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:88f7bc0fcca81f985f78dd0fa68d2c75abf8272b1f5c323ea4a01a4d7a614efc"}, + {file = "frozenlist-1.4.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:5833593c25ac59ede40ed4de6d67eb42928cca97f26feea219f21d0ed0959b79"}, + {file = "frozenlist-1.4.0-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:fec520865f42e5c7f050c2a79038897b1c7d1595e907a9e08e3353293ffc948e"}, + {file = "frozenlist-1.4.0-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:b826d97e4276750beca7c8f0f1a4938892697a6bcd8ec8217b3312dad6982781"}, + {file = "frozenlist-1.4.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:ceb6ec0a10c65540421e20ebd29083c50e6d1143278746a4ef6bcf6153171eb8"}, + {file = "frozenlist-1.4.0-cp38-cp38-win32.whl", hash = "sha256:2b8bcf994563466db019fab287ff390fffbfdb4f905fc77bc1c1d604b1c689cc"}, + {file = "frozenlist-1.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:a6c8097e01886188e5be3e6b14e94ab365f384736aa1fca6a0b9e35bd4a30bc7"}, + {file = "frozenlist-1.4.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:6c38721585f285203e4b4132a352eb3daa19121a035f3182e08e437cface44bf"}, + {file = "frozenlist-1.4.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:a0c6da9aee33ff0b1a451e867da0c1f47408112b3391dd43133838339e410963"}, + {file = "frozenlist-1.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:93ea75c050c5bb3d98016b4ba2497851eadf0ac154d88a67d7a6816206f6fa7f"}, + {file = "frozenlist-1.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f61e2dc5ad442c52b4887f1fdc112f97caeff4d9e6ebe78879364ac59f1663e1"}, + {file = "frozenlist-1.4.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:aa384489fefeb62321b238e64c07ef48398fe80f9e1e6afeff22e140e0850eef"}, + {file = "frozenlist-1.4.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:10ff5faaa22786315ef57097a279b833ecab1a0bfb07d604c9cbb1c4cdc2ed87"}, + {file = "frozenlist-1.4.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:007df07a6e3eb3e33e9a1fe6a9db7af152bbd8a185f9aaa6ece10a3529e3e1c6"}, + {file = "frozenlist-1.4.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f4f399d28478d1f604c2ff9119907af9726aed73680e5ed1ca634d377abb087"}, + {file = "frozenlist-1.4.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:c5374b80521d3d3f2ec5572e05adc94601985cc526fb276d0c8574a6d749f1b3"}, + {file = "frozenlist-1.4.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:ce31ae3e19f3c902de379cf1323d90c649425b86de7bbdf82871b8a2a0615f3d"}, + {file = "frozenlist-1.4.0-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7211ef110a9194b6042449431e08c4d80c0481e5891e58d429df5899690511c2"}, + {file = "frozenlist-1.4.0-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:556de4430ce324c836789fa4560ca62d1591d2538b8ceb0b4f68fb7b2384a27a"}, + {file = "frozenlist-1.4.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:7645a8e814a3ee34a89c4a372011dcd817964ce8cb273c8ed6119d706e9613e3"}, + {file = "frozenlist-1.4.0-cp39-cp39-win32.whl", hash = "sha256:19488c57c12d4e8095a922f328df3f179c820c212940a498623ed39160bc3c2f"}, + {file = "frozenlist-1.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:6221d84d463fb110bdd7619b69cb43878a11d51cbb9394ae3105d082d5199167"}, + {file = "frozenlist-1.4.0.tar.gz", hash = "sha256:09163bdf0b2907454042edb19f887c6d33806adc71fbd54afc14908bfdc22251"}, +] + +[[package]] +name = "fsspec" +version = "2022.11.0" +description = "File-system specification" +optional = false +python-versions = ">=3.7" +files = [ + {file = "fsspec-2022.11.0-py3-none-any.whl", hash = "sha256:d6e462003e3dcdcb8c7aa84c73a228f8227e72453cd22570e2363e8844edfe7b"}, + {file = "fsspec-2022.11.0.tar.gz", hash = "sha256:259d5fd5c8e756ff2ea72f42e7613c32667dc2049a4ac3d84364a7ca034acb8b"}, +] + +[package.extras] +abfs = ["adlfs"] +adl = ["adlfs"] +arrow = ["pyarrow (>=1)"] +dask = ["dask", "distributed"] +dropbox = ["dropbox", "dropboxdrivefs", "requests"] +entrypoints = ["importlib-metadata"] +fuse = ["fusepy"] +gcs = ["gcsfs"] +git = ["pygit2"] +github = ["requests"] +gs = ["gcsfs"] +gui = ["panel"] +hdfs = ["pyarrow (>=1)"] +http = ["aiohttp (!=4.0.0a0,!=4.0.0a1)", "requests"] +libarchive = ["libarchive-c"] +oci = ["ocifs"] +s3 = ["s3fs"] +sftp = ["paramiko"] +smb = ["smbprotocol"] +ssh = ["paramiko"] +tqdm = ["tqdm"] + +[[package]] +name = "geoalchemy2" +version = "0.9.4" +description = "Using SQLAlchemy with Spatial Databases" +optional = false +python-versions = "*" +files = [ + {file = "GeoAlchemy2-0.9.4-py2.py3-none-any.whl", hash = "sha256:2e3323a8442d4b1f8de69106315c17d8ce729fc127a38a5d7e2b53e1a19b9dd5"}, + {file = "GeoAlchemy2-0.9.4.tar.gz", hash = "sha256:b0e56d4a945bdc0f8fa9edd50ecc912889ea68e0e3558a19160dcb0d5b1b65fc"}, +] + +[package.dependencies] +packaging = "*" +SQLAlchemy = ">=1.1" + +[[package]] +name = "greenlet" +version = "3.0.2" +description = "Lightweight in-process concurrent programming" +optional = false +python-versions = ">=3.7" +files = [ + {file = "greenlet-3.0.2-cp310-cp310-macosx_11_0_universal2.whl", hash = "sha256:9acd8fd67c248b8537953cb3af8787c18a87c33d4dcf6830e410ee1f95a63fd4"}, + {file = "greenlet-3.0.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:339c0272a62fac7e602e4e6ec32a64ff9abadc638b72f17f6713556ed011d493"}, + {file = "greenlet-3.0.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:38878744926cec29b5cc3654ef47f3003f14bfbba7230e3c8492393fe29cc28b"}, + {file = "greenlet-3.0.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b3f0497db77cfd034f829678b28267eeeeaf2fc21b3f5041600f7617139e6773"}, + {file = "greenlet-3.0.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ed1a8a08de7f68506a38f9a2ddb26bbd1480689e66d788fcd4b5f77e2d9ecfcc"}, + {file = "greenlet-3.0.2-cp310-cp310-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:89a6f6ddcbef4000cda7e205c4c20d319488ff03db961d72d4e73519d2465309"}, + {file = "greenlet-3.0.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:c1f647fe5b94b51488b314c82fdda10a8756d650cee8d3cd29f657c6031bdf73"}, + {file = "greenlet-3.0.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:9560c580c896030ff9c311c603aaf2282234643c90d1dec738a1d93e3e53cd51"}, + {file = "greenlet-3.0.2-cp310-cp310-win_amd64.whl", hash = "sha256:2e9c5423046eec21f6651268cb674dfba97280701e04ef23d312776377313206"}, + {file = "greenlet-3.0.2-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:b1fd25dfc5879a82103b3d9e43fa952e3026c221996ff4d32a9c72052544835d"}, + {file = "greenlet-3.0.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cecfdc950dd25f25d6582952e58521bca749cf3eeb7a9bad69237024308c8196"}, + {file = "greenlet-3.0.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:edf7a1daba1f7c54326291a8cde58da86ab115b78c91d502be8744f0aa8e3ffa"}, + {file = "greenlet-3.0.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f4cf532bf3c58a862196b06947b1b5cc55503884f9b63bf18582a75228d9950e"}, + {file = "greenlet-3.0.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e79fb5a9fb2d0bd3b6573784f5e5adabc0b0566ad3180a028af99523ce8f6138"}, + {file = "greenlet-3.0.2-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:006c1028ac0cfcc4e772980cfe73f5476041c8c91d15d64f52482fc571149d46"}, + {file = "greenlet-3.0.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:fefd5eb2c0b1adffdf2802ff7df45bfe65988b15f6b972706a0e55d451bffaea"}, + {file = "greenlet-3.0.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0c0fdb8142742ee68e97c106eb81e7d3e883cc739d9c5f2b28bc38a7bafeb6d1"}, + {file = "greenlet-3.0.2-cp311-cp311-win_amd64.whl", hash = "sha256:8f8d14a0a4e8c670fbce633d8b9a1ee175673a695475acd838e372966845f764"}, + {file = "greenlet-3.0.2-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:654b84c9527182036747938b81938f1d03fb8321377510bc1854a9370418ab66"}, + {file = "greenlet-3.0.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cd5bc4fde0842ff2b9cf33382ad0b4db91c2582db836793d58d174c569637144"}, + {file = "greenlet-3.0.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c27b142a9080bdd5869a2fa7ebf407b3c0b24bd812db925de90e9afe3c417fd6"}, + {file = "greenlet-3.0.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0df7eed98ea23b20e9db64d46eb05671ba33147df9405330695bcd81a73bb0c9"}, + {file = "greenlet-3.0.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fb5d60805057d8948065338be6320d35e26b0a72f45db392eb32b70dd6dc9227"}, + {file = "greenlet-3.0.2-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:e0e28f5233d64c693382f66d47c362b72089ebf8ac77df7e12ac705c9fa1163d"}, + {file = "greenlet-3.0.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:3e4bfa752b3688d74ab1186e2159779ff4867644d2b1ebf16db14281f0445377"}, + {file = "greenlet-3.0.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:c42bb589e6e9f9d8bdd79f02f044dff020d30c1afa6e84c0b56d1ce8a324553c"}, + {file = "greenlet-3.0.2-cp312-cp312-win_amd64.whl", hash = "sha256:b2cedf279ca38ef3f4ed0d013a6a84a7fc3d9495a716b84a5fc5ff448965f251"}, + {file = "greenlet-3.0.2-cp37-cp37m-macosx_11_0_universal2.whl", hash = "sha256:6d65bec56a7bc352bcf11b275b838df618651109074d455a772d3afe25390b7d"}, + {file = "greenlet-3.0.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0acadbc3f72cb0ee85070e8d36bd2a4673d2abd10731ee73c10222cf2dd4713c"}, + {file = "greenlet-3.0.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:14b5d999aefe9ffd2049ad19079f733c3aaa426190ffecadb1d5feacef8fe397"}, + {file = "greenlet-3.0.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f27aa32466993c92d326df982c4acccd9530fe354e938d9e9deada563e71ce76"}, + {file = "greenlet-3.0.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8f34a765c5170c0673eb747213a0275ecc749ab3652bdbec324621ed5b2edaef"}, + {file = "greenlet-3.0.2-cp37-cp37m-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:520fcb53a39ef90f5021c77606952dbbc1da75d77114d69b8d7bded4a8e1a813"}, + {file = "greenlet-3.0.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d1fceb5351ab1601903e714c3028b37f6ea722be6873f46e349a960156c05650"}, + {file = "greenlet-3.0.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:7363756cc439a503505b67983237d1cc19139b66488263eb19f5719a32597836"}, + {file = "greenlet-3.0.2-cp37-cp37m-win32.whl", hash = "sha256:d5547b462b8099b84746461e882a3eb8a6e3f80be46cb6afb8524eeb191d1a30"}, + {file = "greenlet-3.0.2-cp37-cp37m-win_amd64.whl", hash = "sha256:950e21562818f9c771989b5b65f990e76f4ac27af66e1bb34634ae67886ede2a"}, + {file = "greenlet-3.0.2-cp38-cp38-macosx_11_0_universal2.whl", hash = "sha256:d64643317e76b4b41fdba659e7eca29634e5739b8bc394eda3a9127f697ed4b0"}, + {file = "greenlet-3.0.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5f9ea7c2c9795549653b6f7569f6bc75d2c7d1f6b2854eb8ce0bc6ec3cb2dd88"}, + {file = "greenlet-3.0.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:db4233358d3438369051a2f290f1311a360d25c49f255a6c5d10b5bcb3aa2b49"}, + {file = "greenlet-3.0.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ed9bf77b41798e8417657245b9f3649314218a4a17aefb02bb3992862df32495"}, + {file = "greenlet-3.0.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d4d0df07a38e41a10dfb62c6fc75ede196572b580f48ee49b9282c65639f3965"}, + {file = "greenlet-3.0.2-cp38-cp38-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:10d247260db20887ae8857c0cbc750b9170f0b067dd7d38fb68a3f2334393bd3"}, + {file = "greenlet-3.0.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:a37ae53cca36823597fd5f65341b6f7bac2dd69ecd6ca01334bb795460ab150b"}, + {file = "greenlet-3.0.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:80d068e4b6e2499847d916ef64176811ead6bf210a610859220d537d935ec6fd"}, + {file = "greenlet-3.0.2-cp38-cp38-win32.whl", hash = "sha256:b1405614692ac986490d10d3e1a05e9734f473750d4bee3cf7d1286ef7af7da6"}, + {file = "greenlet-3.0.2-cp38-cp38-win_amd64.whl", hash = "sha256:8756a94ed8f293450b0e91119eca2a36332deba69feb2f9ca410d35e74eae1e4"}, + {file = "greenlet-3.0.2-cp39-cp39-macosx_11_0_universal2.whl", hash = "sha256:2c93cd03acb1499ee4de675e1a4ed8eaaa7227f7949dc55b37182047b006a7aa"}, + {file = "greenlet-3.0.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1dac09e3c0b78265d2e6d3cbac2d7c48bd1aa4b04a8ffeda3adde9f1688df2c3"}, + {file = "greenlet-3.0.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2ee59c4627c8c4bb3e15949fbcd499abd6b7f4ad9e0bfcb62c65c5e2cabe0ec4"}, + {file = "greenlet-3.0.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18fe39d70d482b22f0014e84947c5aaa7211fb8e13dc4cc1c43ed2aa1db06d9a"}, + {file = "greenlet-3.0.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e84bef3cfb6b6bfe258c98c519811c240dbc5b33a523a14933a252e486797c90"}, + {file = "greenlet-3.0.2-cp39-cp39-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:aecea0442975741e7d69daff9b13c83caff8c13eeb17485afa65f6360a045765"}, + {file = "greenlet-3.0.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:f260e6c2337871a52161824058923df2bbddb38bc11a5cbe71f3474d877c5bd9"}, + {file = "greenlet-3.0.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:fc14dd9554f88c9c1fe04771589ae24db76cd56c8f1104e4381b383d6b71aff8"}, + {file = "greenlet-3.0.2-cp39-cp39-win32.whl", hash = "sha256:bfcecc984d60b20ffe30173b03bfe9ba6cb671b0be1e95c3e2056d4fe7006590"}, + {file = "greenlet-3.0.2-cp39-cp39-win_amd64.whl", hash = "sha256:c235131bf59d2546bb3ebaa8d436126267392f2e51b85ff45ac60f3a26549af0"}, + {file = "greenlet-3.0.2.tar.gz", hash = "sha256:1c1129bc47266d83444c85a8e990ae22688cf05fb20d7951fd2866007c2ba9bc"}, +] + +[package.extras] +docs = ["Sphinx"] +test = ["objgraph", "psutil"] + +[[package]] +name = "gunicorn" +version = "20.1.0" +description = "WSGI HTTP Server for UNIX" +optional = false +python-versions = ">=3.5" +files = [ + {file = "gunicorn-20.1.0-py3-none-any.whl", hash = "sha256:9dcc4547dbb1cb284accfb15ab5667a0e5d1881cc443e0677b4882a4067a807e"}, + {file = "gunicorn-20.1.0.tar.gz", hash = "sha256:e0a968b5ba15f8a328fdfd7ab1fcb5af4470c28aaf7e55df02a99bc13138e6e8"}, +] + +[package.dependencies] +setuptools = ">=3.0" + +[package.extras] +eventlet = ["eventlet (>=0.24.1)"] +gevent = ["gevent (>=1.4.0)"] +setproctitle = ["setproctitle"] +tornado = ["tornado (>=0.2)"] + +[[package]] +name = "h11" +version = "0.14.0" +description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" +optional = false +python-versions = ">=3.7" +files = [ + {file = "h11-0.14.0-py3-none-any.whl", hash = "sha256:e3fe4ac4b851c468cc8363d500db52c2ead036020723024a109d37346efaa761"}, + {file = "h11-0.14.0.tar.gz", hash = "sha256:8f19fbbe99e72420ff35c00b27a34cb9937e902a8b810e2c88300c6f0a3b699d"}, +] + +[[package]] +name = "httpcore" +version = "0.16.3" +description = "A minimal low-level HTTP client." +optional = false +python-versions = ">=3.7" +files = [ + {file = "httpcore-0.16.3-py3-none-any.whl", hash = "sha256:da1fb708784a938aa084bde4feb8317056c55037247c787bd7e19eb2c2949dc0"}, + {file = "httpcore-0.16.3.tar.gz", hash = "sha256:c5d6f04e2fc530f39e0c077e6a30caa53f1451096120f1f38b954afd0b17c0cb"}, +] + +[package.dependencies] +anyio = ">=3.0,<5.0" +certifi = "*" +h11 = ">=0.13,<0.15" +sniffio = "==1.*" + +[package.extras] +http2 = ["h2 (>=3,<5)"] +socks = ["socksio (==1.*)"] + +[[package]] +name = "httpx" +version = "0.23.3" +description = "The next generation HTTP client." +optional = false +python-versions = ">=3.7" +files = [ + {file = "httpx-0.23.3-py3-none-any.whl", hash = "sha256:a211fcce9b1254ea24f0cd6af9869b3d29aba40154e947d2a07bb499b3e310d6"}, + {file = "httpx-0.23.3.tar.gz", hash = "sha256:9818458eb565bb54898ccb9b8b251a28785dd4a55afbc23d0eb410754fe7d0f9"}, +] + +[package.dependencies] +certifi = "*" +httpcore = ">=0.15.0,<0.17.0" +rfc3986 = {version = ">=1.3,<2", extras = ["idna2008"]} +sniffio = "*" + +[package.extras] +brotli = ["brotli", "brotlicffi"] +cli = ["click (==8.*)", "pygments (==2.*)", "rich (>=10,<13)"] +http2 = ["h2 (>=3,<5)"] +socks = ["socksio (==1.*)"] + +[[package]] +name = "idna" +version = "3.6" +description = "Internationalized Domain Names in Applications (IDNA)" +optional = false +python-versions = ">=3.5" +files = [ + {file = "idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f"}, + {file = "idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca"}, +] + +[[package]] +name = "imagesize" +version = "1.4.1" +description = "Getting image size from png/jpeg/jpeg2000/gif file" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "imagesize-1.4.1-py2.py3-none-any.whl", hash = "sha256:0d8d18d08f840c19d0ee7ca1fd82490fdc3729b7ac93f49870406ddde8ef8d8b"}, + {file = "imagesize-1.4.1.tar.gz", hash = "sha256:69150444affb9cb0d5cc5a92b3676f0b2fb7cd9ae39e947a5e11a36b4497cd4a"}, +] + +[[package]] +name = "importlib-metadata" +version = "7.0.0" +description = "Read metadata from Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "importlib_metadata-7.0.0-py3-none-any.whl", hash = "sha256:d97503976bb81f40a193d41ee6570868479c69d5068651eb039c40d850c59d67"}, + {file = "importlib_metadata-7.0.0.tar.gz", hash = "sha256:7fc841f8b8332803464e5dc1c63a2e59121f46ca186c0e2e182e80bf8c1319f7"}, +] + +[package.dependencies] +zipp = ">=0.5" + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"] +perf = ["ipython"] +testing = ["flufl.flake8", "importlib-resources (>=1.3)", "packaging", "pyfakefs", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf (>=0.9.2)", "pytest-ruff"] + +[[package]] +name = "importlib-resources" +version = "6.1.1" +description = "Read resources from Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "importlib_resources-6.1.1-py3-none-any.whl", hash = "sha256:e8bf90d8213b486f428c9c39714b920041cb02c184686a3dee24905aaa8105d6"}, + {file = "importlib_resources-6.1.1.tar.gz", hash = "sha256:3893a00122eafde6894c59914446a512f728a0c1a45f9bb9b63721b6bacf0b4a"}, +] + +[package.dependencies] +zipp = {version = ">=3.1.0", markers = "python_version < \"3.10\""} + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"] +testing = ["pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-ruff", "zipp (>=3.17)"] + +[[package]] +name = "iniconfig" +version = "2.0.0" +description = "brain-dead simple config-ini parsing" +optional = false +python-versions = ">=3.7" +files = [ + {file = "iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374"}, + {file = "iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3"}, +] + +[[package]] +name = "ipdb" +version = "0.13.13" +description = "IPython-enabled pdb" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "ipdb-0.13.13-py3-none-any.whl", hash = "sha256:45529994741c4ab6d2388bfa5d7b725c2cf7fe9deffabdb8a6113aa5ed449ed4"}, + {file = "ipdb-0.13.13.tar.gz", hash = "sha256:e3ac6018ef05126d442af680aad863006ec19d02290561ac88b8b1c0b0cfc726"}, +] + +[package.dependencies] +decorator = {version = "*", markers = "python_version > \"3.6\""} +ipython = {version = ">=7.31.1", markers = "python_version > \"3.6\""} +tomli = {version = "*", markers = "python_version > \"3.6\" and python_version < \"3.11\""} + +[[package]] +name = "ipython" +version = "8.12.3" +description = "IPython: Productive Interactive Computing" +optional = false +python-versions = ">=3.8" +files = [ + {file = "ipython-8.12.3-py3-none-any.whl", hash = "sha256:b0340d46a933d27c657b211a329d0be23793c36595acf9e6ef4164bc01a1804c"}, + {file = "ipython-8.12.3.tar.gz", hash = "sha256:3910c4b54543c2ad73d06579aa771041b7d5707b033bd488669b4cf544e3b363"}, +] + +[package.dependencies] +appnope = {version = "*", markers = "sys_platform == \"darwin\""} +backcall = "*" +colorama = {version = "*", markers = "sys_platform == \"win32\""} +decorator = "*" +jedi = ">=0.16" +matplotlib-inline = "*" +pexpect = {version = ">4.3", markers = "sys_platform != \"win32\""} +pickleshare = "*" +prompt-toolkit = ">=3.0.30,<3.0.37 || >3.0.37,<3.1.0" +pygments = ">=2.4.0" +stack-data = "*" +traitlets = ">=5" +typing-extensions = {version = "*", markers = "python_version < \"3.10\""} + +[package.extras] +all = ["black", "curio", "docrepr", "ipykernel", "ipyparallel", "ipywidgets", "matplotlib", "matplotlib (!=3.2.0)", "nbconvert", "nbformat", "notebook", "numpy (>=1.21)", "pandas", "pytest (<7)", "pytest (<7.1)", "pytest-asyncio", "qtconsole", "setuptools (>=18.5)", "sphinx (>=1.3)", "sphinx-rtd-theme", "stack-data", "testpath", "trio", "typing-extensions"] +black = ["black"] +doc = ["docrepr", "ipykernel", "matplotlib", "pytest (<7)", "pytest (<7.1)", "pytest-asyncio", "setuptools (>=18.5)", "sphinx (>=1.3)", "sphinx-rtd-theme", "stack-data", "testpath", "typing-extensions"] +kernel = ["ipykernel"] +nbconvert = ["nbconvert"] +nbformat = ["nbformat"] +notebook = ["ipywidgets", "notebook"] +parallel = ["ipyparallel"] +qtconsole = ["qtconsole"] +test = ["pytest (<7.1)", "pytest-asyncio", "testpath"] +test-extra = ["curio", "matplotlib (!=3.2.0)", "nbformat", "numpy (>=1.21)", "pandas", "pytest (<7.1)", "pytest-asyncio", "testpath", "trio"] + +[[package]] +name = "isodate" +version = "0.6.1" +description = "An ISO 8601 date/time/duration parser and formatter" +optional = false +python-versions = "*" +files = [ + {file = "isodate-0.6.1-py2.py3-none-any.whl", hash = "sha256:0751eece944162659049d35f4f549ed815792b38793f07cf73381c1c87cbed96"}, + {file = "isodate-0.6.1.tar.gz", hash = "sha256:48c5881de7e8b0a0d648cb024c8062dc84e7b840ed81e864c7614fd3c127bde9"}, +] + +[package.dependencies] +six = "*" + +[[package]] +name = "jedi" +version = "0.19.1" +description = "An autocompletion tool for Python that can be used for text editors." +optional = false +python-versions = ">=3.6" +files = [ + {file = "jedi-0.19.1-py2.py3-none-any.whl", hash = "sha256:e983c654fe5c02867aef4cdfce5a2fbb4a50adc0af145f70504238f18ef5e7e0"}, + {file = "jedi-0.19.1.tar.gz", hash = "sha256:cf0496f3651bc65d7174ac1b7d043eff454892c708a87d1b683e57b569927ffd"}, +] + +[package.dependencies] +parso = ">=0.8.3,<0.9.0" + +[package.extras] +docs = ["Jinja2 (==2.11.3)", "MarkupSafe (==1.1.1)", "Pygments (==2.8.1)", "alabaster (==0.7.12)", "babel (==2.9.1)", "chardet (==4.0.0)", "commonmark (==0.8.1)", "docutils (==0.17.1)", "future (==0.18.2)", "idna (==2.10)", "imagesize (==1.2.0)", "mock (==1.0.1)", "packaging (==20.9)", "pyparsing (==2.4.7)", "pytz (==2021.1)", "readthedocs-sphinx-ext (==2.1.4)", "recommonmark (==0.5.0)", "requests (==2.25.1)", "six (==1.15.0)", "snowballstemmer (==2.1.0)", "sphinx (==1.8.5)", "sphinx-rtd-theme (==0.4.3)", "sphinxcontrib-serializinghtml (==1.1.4)", "sphinxcontrib-websupport (==1.2.4)", "urllib3 (==1.26.4)"] +qa = ["flake8 (==5.0.4)", "mypy (==0.971)", "types-setuptools (==67.2.0.1)"] +testing = ["Django", "attrs", "colorama", "docopt", "pytest (<7.0.0)"] + +[[package]] +name = "jinja2" +version = "3.1.2" +description = "A very fast and expressive template engine." +optional = false +python-versions = ">=3.7" +files = [ + {file = "Jinja2-3.1.2-py3-none-any.whl", hash = "sha256:6088930bfe239f0e6710546ab9c19c9ef35e29792895fed6e6e31a023a182a61"}, + {file = "Jinja2-3.1.2.tar.gz", hash = "sha256:31351a702a408a9e7595a8fc6150fc3f43bb6bf7e319770cbc0db9df9437e852"}, +] + +[package.dependencies] +MarkupSafe = ">=2.0" + +[package.extras] +i18n = ["Babel (>=2.7)"] + +[[package]] +name = "jsonpatch" +version = "1.33" +description = "Apply JSON-Patches (RFC 6902)" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*, !=3.6.*" +files = [ + {file = "jsonpatch-1.33-py2.py3-none-any.whl", hash = "sha256:0ae28c0cd062bbd8b8ecc26d7d164fbbea9652a1a3693f3b956c1eae5145dade"}, + {file = "jsonpatch-1.33.tar.gz", hash = "sha256:9fcd4009c41e6d12348b4a0ff2563ba56a2923a7dfee731d004e212e1ee5030c"}, +] + +[package.dependencies] +jsonpointer = ">=1.9" + +[[package]] +name = "jsonpointer" +version = "2.4" +description = "Identify specific nodes in a JSON document (RFC 6901)" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*, !=3.6.*" +files = [ + {file = "jsonpointer-2.4-py2.py3-none-any.whl", hash = "sha256:15d51bba20eea3165644553647711d150376234112651b4f1811022aecad7d7a"}, + {file = "jsonpointer-2.4.tar.gz", hash = "sha256:585cee82b70211fa9e6043b7bb89db6e1aa49524340dde8ad6b63206ea689d88"}, +] + +[[package]] +name = "jsonschema" +version = "4.20.0" +description = "An implementation of JSON Schema validation for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "jsonschema-4.20.0-py3-none-any.whl", hash = "sha256:ed6231f0429ecf966f5bc8dfef245998220549cbbcf140f913b7464c52c3b6b3"}, + {file = "jsonschema-4.20.0.tar.gz", hash = "sha256:4f614fd46d8d61258610998997743ec5492a648b33cf478c1ddc23ed4598a5fa"}, +] + +[package.dependencies] +attrs = ">=22.2.0" +importlib-resources = {version = ">=1.4.0", markers = "python_version < \"3.9\""} +jsonschema-specifications = ">=2023.03.6" +pkgutil-resolve-name = {version = ">=1.3.10", markers = "python_version < \"3.9\""} +referencing = ">=0.28.4" +rpds-py = ">=0.7.1" + +[package.extras] +format = ["fqdn", "idna", "isoduration", "jsonpointer (>1.13)", "rfc3339-validator", "rfc3987", "uri-template", "webcolors (>=1.11)"] +format-nongpl = ["fqdn", "idna", "isoduration", "jsonpointer (>1.13)", "rfc3339-validator", "rfc3986-validator (>0.1.0)", "uri-template", "webcolors (>=1.11)"] + +[[package]] +name = "jsonschema-specifications" +version = "2023.11.2" +description = "The JSON Schema meta-schemas and vocabularies, exposed as a Registry" +optional = false +python-versions = ">=3.8" +files = [ + {file = "jsonschema_specifications-2023.11.2-py3-none-any.whl", hash = "sha256:e74ba7c0a65e8cb49dc26837d6cfe576557084a8b423ed16a420984228104f93"}, + {file = "jsonschema_specifications-2023.11.2.tar.gz", hash = "sha256:9472fc4fea474cd74bea4a2b190daeccb5a9e4db2ea80efcf7a1b582fc9a81b8"}, +] + +[package.dependencies] +importlib-resources = {version = ">=1.4.0", markers = "python_version < \"3.9\""} +referencing = ">=0.31.0" + +[[package]] +name = "lark-parser" +version = "0.12.0" +description = "a modern parsing library" +optional = false +python-versions = "*" +files = [ + {file = "lark-parser-0.12.0.tar.gz", hash = "sha256:15967db1f1214013dca65b1180745047b9be457d73da224fcda3d9dd4e96a138"}, + {file = "lark_parser-0.12.0-py2.py3-none-any.whl", hash = "sha256:0eaf30cb5ba787fe404d73a7d6e61df97b21d5a63ac26c5008c78a494373c675"}, +] + +[package.extras] +atomic-cache = ["atomicwrites"] +nearley = ["js2py"] +regex = ["regex"] + +[[package]] +name = "livereload" +version = "2.6.3" +description = "Python LiveReload is an awesome tool for web developers" +optional = false +python-versions = "*" +files = [ + {file = "livereload-2.6.3-py2.py3-none-any.whl", hash = "sha256:ad4ac6f53b2d62bb6ce1a5e6e96f1f00976a32348afedcb4b6d68df2a1d346e4"}, + {file = "livereload-2.6.3.tar.gz", hash = "sha256:776f2f865e59fde56490a56bcc6773b6917366bce0c267c60ee8aaf1a0959869"}, +] + +[package.dependencies] +six = "*" +tornado = {version = "*", markers = "python_version > \"2.7\""} + +[[package]] +name = "locket" +version = "1.0.0" +description = "File-based locks for Python on Linux and Windows" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "locket-1.0.0-py2.py3-none-any.whl", hash = "sha256:b6c819a722f7b6bd955b80781788e4a66a55628b858d347536b7e81325a3a5e3"}, + {file = "locket-1.0.0.tar.gz", hash = "sha256:5c0d4c052a8bbbf750e056a8e65ccd309086f4f0f18a2eac306a8dfa4112a632"}, +] + +[[package]] +name = "lxml" +version = "4.9.3" +description = "Powerful and Pythonic XML processing library combining libxml2/libxslt with the ElementTree API." +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, != 3.4.*" +files = [ + {file = "lxml-4.9.3-cp27-cp27m-macosx_11_0_x86_64.whl", hash = "sha256:b0a545b46b526d418eb91754565ba5b63b1c0b12f9bd2f808c852d9b4b2f9b5c"}, + {file = "lxml-4.9.3-cp27-cp27m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:075b731ddd9e7f68ad24c635374211376aa05a281673ede86cbe1d1b3455279d"}, + {file = "lxml-4.9.3-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1e224d5755dba2f4a9498e150c43792392ac9b5380aa1b845f98a1618c94eeef"}, + {file = "lxml-4.9.3-cp27-cp27m-win32.whl", hash = "sha256:2c74524e179f2ad6d2a4f7caf70e2d96639c0954c943ad601a9e146c76408ed7"}, + {file = "lxml-4.9.3-cp27-cp27m-win_amd64.whl", hash = "sha256:4f1026bc732b6a7f96369f7bfe1a4f2290fb34dce00d8644bc3036fb351a4ca1"}, + {file = "lxml-4.9.3-cp27-cp27mu-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c0781a98ff5e6586926293e59480b64ddd46282953203c76ae15dbbbf302e8bb"}, + {file = "lxml-4.9.3-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:cef2502e7e8a96fe5ad686d60b49e1ab03e438bd9123987994528febd569868e"}, + {file = "lxml-4.9.3-cp310-cp310-macosx_11_0_x86_64.whl", hash = "sha256:b86164d2cff4d3aaa1f04a14685cbc072efd0b4f99ca5708b2ad1b9b5988a991"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:42871176e7896d5d45138f6d28751053c711ed4d48d8e30b498da155af39aebd"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:ae8b9c6deb1e634ba4f1930eb67ef6e6bf6a44b6eb5ad605642b2d6d5ed9ce3c"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:411007c0d88188d9f621b11d252cce90c4a2d1a49db6c068e3c16422f306eab8"}, + {file = "lxml-4.9.3-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:cd47b4a0d41d2afa3e58e5bf1f62069255aa2fd6ff5ee41604418ca925911d76"}, + {file = "lxml-4.9.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:0e2cb47860da1f7e9a5256254b74ae331687b9672dfa780eed355c4c9c3dbd23"}, + {file = "lxml-4.9.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:1247694b26342a7bf47c02e513d32225ededd18045264d40758abeb3c838a51f"}, + {file = "lxml-4.9.3-cp310-cp310-win32.whl", hash = "sha256:cdb650fc86227eba20de1a29d4b2c1bfe139dc75a0669270033cb2ea3d391b85"}, + {file = "lxml-4.9.3-cp310-cp310-win_amd64.whl", hash = "sha256:97047f0d25cd4bcae81f9ec9dc290ca3e15927c192df17331b53bebe0e3ff96d"}, + {file = "lxml-4.9.3-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:1f447ea5429b54f9582d4b955f5f1985f278ce5cf169f72eea8afd9502973dd5"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:57d6ba0ca2b0c462f339640d22882acc711de224d769edf29962b09f77129cbf"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:9767e79108424fb6c3edf8f81e6730666a50feb01a328f4a016464a5893f835a"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:71c52db65e4b56b8ddc5bb89fb2e66c558ed9d1a74a45ceb7dcb20c191c3df2f"}, + {file = "lxml-4.9.3-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:d73d8ecf8ecf10a3bd007f2192725a34bd62898e8da27eb9d32a58084f93962b"}, + {file = "lxml-4.9.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0a3d3487f07c1d7f150894c238299934a2a074ef590b583103a45002035be120"}, + {file = "lxml-4.9.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:9e28c51fa0ce5674be9f560c6761c1b441631901993f76700b1b30ca6c8378d6"}, + {file = "lxml-4.9.3-cp311-cp311-win32.whl", hash = "sha256:0bfd0767c5c1de2551a120673b72e5d4b628737cb05414f03c3277bf9bed3305"}, + {file = "lxml-4.9.3-cp311-cp311-win_amd64.whl", hash = "sha256:25f32acefac14ef7bd53e4218fe93b804ef6f6b92ffdb4322bb6d49d94cad2bc"}, + {file = "lxml-4.9.3-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:d3ff32724f98fbbbfa9f49d82852b159e9784d6094983d9a8b7f2ddaebb063d4"}, + {file = "lxml-4.9.3-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:48d6ed886b343d11493129e019da91d4039826794a3e3027321c56d9e71505be"}, + {file = "lxml-4.9.3-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:9a92d3faef50658dd2c5470af249985782bf754c4e18e15afb67d3ab06233f13"}, + {file = "lxml-4.9.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:b4e4bc18382088514ebde9328da057775055940a1f2e18f6ad2d78aa0f3ec5b9"}, + {file = "lxml-4.9.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:fc9b106a1bf918db68619fdcd6d5ad4f972fdd19c01d19bdb6bf63f3589a9ec5"}, + {file = "lxml-4.9.3-cp312-cp312-win_amd64.whl", hash = "sha256:d37017287a7adb6ab77e1c5bee9bcf9660f90ff445042b790402a654d2ad81d8"}, + {file = "lxml-4.9.3-cp35-cp35m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:56dc1f1ebccc656d1b3ed288f11e27172a01503fc016bcabdcbc0978b19352b7"}, + {file = "lxml-4.9.3-cp35-cp35m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:578695735c5a3f51569810dfebd05dd6f888147a34f0f98d4bb27e92b76e05c2"}, + {file = "lxml-4.9.3-cp35-cp35m-win32.whl", hash = "sha256:704f61ba8c1283c71b16135caf697557f5ecf3e74d9e453233e4771d68a1f42d"}, + {file = "lxml-4.9.3-cp35-cp35m-win_amd64.whl", hash = "sha256:c41bfca0bd3532d53d16fd34d20806d5c2b1ace22a2f2e4c0008570bf2c58833"}, + {file = "lxml-4.9.3-cp36-cp36m-macosx_11_0_x86_64.whl", hash = "sha256:64f479d719dc9f4c813ad9bb6b28f8390360660b73b2e4beb4cb0ae7104f1c12"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:dd708cf4ee4408cf46a48b108fb9427bfa00b9b85812a9262b5c668af2533ea5"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5c31c7462abdf8f2ac0577d9f05279727e698f97ecbb02f17939ea99ae8daa98"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:e3cd95e10c2610c360154afdc2f1480aea394f4a4f1ea0a5eacce49640c9b190"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_28_x86_64.whl", hash = "sha256:4930be26af26ac545c3dffb662521d4e6268352866956672231887d18f0eaab2"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4aec80cde9197340bc353d2768e2a75f5f60bacda2bab72ab1dc499589b3878c"}, + {file = "lxml-4.9.3-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:14e019fd83b831b2e61baed40cab76222139926b1fb5ed0e79225bc0cae14584"}, + {file = "lxml-4.9.3-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:0c0850c8b02c298d3c7006b23e98249515ac57430e16a166873fc47a5d549287"}, + {file = "lxml-4.9.3-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:aca086dc5f9ef98c512bac8efea4483eb84abbf926eaeedf7b91479feb092458"}, + {file = "lxml-4.9.3-cp36-cp36m-win32.whl", hash = "sha256:50baa9c1c47efcaef189f31e3d00d697c6d4afda5c3cde0302d063492ff9b477"}, + {file = "lxml-4.9.3-cp36-cp36m-win_amd64.whl", hash = "sha256:bef4e656f7d98aaa3486d2627e7d2df1157d7e88e7efd43a65aa5dd4714916cf"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:46f409a2d60f634fe550f7133ed30ad5321ae2e6630f13657fb9479506b00601"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:4c28a9144688aef80d6ea666c809b4b0e50010a2aca784c97f5e6bf143d9f129"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:141f1d1a9b663c679dc524af3ea1773e618907e96075262726c7612c02b149a4"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_28_x86_64.whl", hash = "sha256:53ace1c1fd5a74ef662f844a0413446c0629d151055340e9893da958a374f70d"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:17a753023436a18e27dd7769e798ce302963c236bc4114ceee5b25c18c52c693"}, + {file = "lxml-4.9.3-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:7d298a1bd60c067ea75d9f684f5f3992c9d6766fadbc0bcedd39750bf344c2f4"}, + {file = "lxml-4.9.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:081d32421db5df44c41b7f08a334a090a545c54ba977e47fd7cc2deece78809a"}, + {file = "lxml-4.9.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:23eed6d7b1a3336ad92d8e39d4bfe09073c31bfe502f20ca5116b2a334f8ec02"}, + {file = "lxml-4.9.3-cp37-cp37m-win32.whl", hash = "sha256:1509dd12b773c02acd154582088820893109f6ca27ef7291b003d0e81666109f"}, + {file = "lxml-4.9.3-cp37-cp37m-win_amd64.whl", hash = "sha256:120fa9349a24c7043854c53cae8cec227e1f79195a7493e09e0c12e29f918e52"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:4d2d1edbca80b510443f51afd8496be95529db04a509bc8faee49c7b0fb6d2cc"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.manylinux_2_24_aarch64.whl", hash = "sha256:8d7e43bd40f65f7d97ad8ef5c9b1778943d02f04febef12def25f7583d19baac"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:71d66ee82e7417828af6ecd7db817913cb0cf9d4e61aa0ac1fde0583d84358db"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:6fc3c450eaa0b56f815c7b62f2b7fba7266c4779adcf1cece9e6deb1de7305ce"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:65299ea57d82fb91c7f019300d24050c4ddeb7c5a190e076b5f48a2b43d19c42"}, + {file = "lxml-4.9.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:eadfbbbfb41b44034a4c757fd5d70baccd43296fb894dba0295606a7cf3124aa"}, + {file = "lxml-4.9.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:3e9bdd30efde2b9ccfa9cb5768ba04fe71b018a25ea093379c857c9dad262c40"}, + {file = "lxml-4.9.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:fcdd00edfd0a3001e0181eab3e63bd5c74ad3e67152c84f93f13769a40e073a7"}, + {file = "lxml-4.9.3-cp38-cp38-win32.whl", hash = "sha256:57aba1bbdf450b726d58b2aea5fe47c7875f5afb2c4a23784ed78f19a0462574"}, + {file = "lxml-4.9.3-cp38-cp38-win_amd64.whl", hash = "sha256:92af161ecbdb2883c4593d5ed4815ea71b31fafd7fd05789b23100d081ecac96"}, + {file = "lxml-4.9.3-cp39-cp39-macosx_11_0_x86_64.whl", hash = "sha256:9bb6ad405121241e99a86efff22d3ef469024ce22875a7ae045896ad23ba2340"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:8ed74706b26ad100433da4b9d807eae371efaa266ffc3e9191ea436087a9d6a7"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:fbf521479bcac1e25a663df882c46a641a9bff6b56dc8b0fafaebd2f66fb231b"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:303bf1edce6ced16bf67a18a1cf8339d0db79577eec5d9a6d4a80f0fb10aa2da"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:5515edd2a6d1a5a70bfcdee23b42ec33425e405c5b351478ab7dc9347228f96e"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:690dafd0b187ed38583a648076865d8c229661ed20e48f2335d68e2cf7dc829d"}, + {file = "lxml-4.9.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:b6420a005548ad52154c8ceab4a1290ff78d757f9e5cbc68f8c77089acd3c432"}, + {file = "lxml-4.9.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bb3bb49c7a6ad9d981d734ef7c7193bc349ac338776a0360cc671eaee89bcf69"}, + {file = "lxml-4.9.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d27be7405547d1f958b60837dc4c1007da90b8b23f54ba1f8b728c78fdb19d50"}, + {file = "lxml-4.9.3-cp39-cp39-win32.whl", hash = "sha256:8df133a2ea5e74eef5e8fc6f19b9e085f758768a16e9877a60aec455ed2609b2"}, + {file = "lxml-4.9.3-cp39-cp39-win_amd64.whl", hash = "sha256:4dd9a263e845a72eacb60d12401e37c616438ea2e5442885f65082c276dfb2b2"}, + {file = "lxml-4.9.3-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:6689a3d7fd13dc687e9102a27e98ef33730ac4fe37795d5036d18b4d527abd35"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:f6bdac493b949141b733c5345b6ba8f87a226029cbabc7e9e121a413e49441e0"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:05186a0f1346ae12553d66df1cfce6f251589fea3ad3da4f3ef4e34b2d58c6a3"}, + {file = "lxml-4.9.3-pp37-pypy37_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:c2006f5c8d28dee289f7020f721354362fa304acbaaf9745751ac4006650254b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-macosx_11_0_x86_64.whl", hash = "sha256:5c245b783db29c4e4fbbbfc9c5a78be496c9fea25517f90606aa1f6b2b3d5f7b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:4fb960a632a49f2f089d522f70496640fdf1218f1243889da3822e0a9f5f3ba7"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:50670615eaf97227d5dc60de2dc99fb134a7130d310d783314e7724bf163f75d"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:9719fe17307a9e814580af1f5c6e05ca593b12fb7e44fe62450a5384dbf61b4b"}, + {file = "lxml-4.9.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:3331bece23c9ee066e0fb3f96c61322b9e0f54d775fccefff4c38ca488de283a"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-macosx_11_0_x86_64.whl", hash = "sha256:ed667f49b11360951e201453fc3967344d0d0263aa415e1619e85ae7fd17b4e0"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_24_i686.whl", hash = "sha256:8b77946fd508cbf0fccd8e400a7f71d4ac0e1595812e66025bac475a8e811694"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:e4da8ca0c0c0aea88fd46be8e44bd49716772358d648cce45fe387f7b92374a7"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:fe4bda6bd4340caa6e5cf95e73f8fea5c4bfc55763dd42f1b50a94c1b4a2fbd4"}, + {file = "lxml-4.9.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:f3df3db1d336b9356dd3112eae5f5c2b8b377f3bc826848567f10bfddfee77e9"}, + {file = "lxml-4.9.3.tar.gz", hash = "sha256:48628bd53a426c9eb9bc066a923acaa0878d1e86129fd5359aee99285f4eed9c"}, +] + +[package.extras] +cssselect = ["cssselect (>=0.7)"] +html5 = ["html5lib"] +htmlsoup = ["BeautifulSoup4"] +source = ["Cython (>=0.29.35)"] + +[[package]] +name = "mako" +version = "1.3.0" +description = "A super-fast templating language that borrows the best ideas from the existing templating languages." +optional = false +python-versions = ">=3.8" +files = [ + {file = "Mako-1.3.0-py3-none-any.whl", hash = "sha256:57d4e997349f1a92035aa25c17ace371a4213f2ca42f99bee9a602500cfd54d9"}, + {file = "Mako-1.3.0.tar.gz", hash = "sha256:e3a9d388fd00e87043edbe8792f45880ac0114e9c4adc69f6e9bfb2c55e3b11b"}, +] + +[package.dependencies] +MarkupSafe = ">=0.9.2" + +[package.extras] +babel = ["Babel"] +lingua = ["lingua"] +testing = ["pytest"] + +[[package]] +name = "markupsafe" +version = "2.1.3" +description = "Safely add untrusted strings to HTML/XML markup." +optional = false +python-versions = ">=3.7" +files = [ + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:cd0f502fe016460680cd20aaa5a76d241d6f35a1c3350c474bac1273803893fa"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e09031c87a1e51556fdcb46e5bd4f59dfb743061cf93c4d6831bf894f125eb57"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:68e78619a61ecf91e76aa3e6e8e33fc4894a2bebe93410754bd28fce0a8a4f9f"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:65c1a9bcdadc6c28eecee2c119465aebff8f7a584dd719facdd9e825ec61ab52"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:525808b8019e36eb524b8c68acdd63a37e75714eac50e988180b169d64480a00"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:962f82a3086483f5e5f64dbad880d31038b698494799b097bc59c2edf392fce6"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:aa7bd130efab1c280bed0f45501b7c8795f9fdbeb02e965371bbef3523627779"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c9c804664ebe8f83a211cace637506669e7890fec1b4195b505c214e50dd4eb7"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win32.whl", hash = "sha256:10bbfe99883db80bdbaff2dcf681dfc6533a614f700da1287707e8a5d78a8431"}, + {file = "MarkupSafe-2.1.3-cp310-cp310-win_amd64.whl", hash = "sha256:1577735524cdad32f9f694208aa75e422adba74f1baee7551620e43a3141f559"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:ad9e82fb8f09ade1c3e1b996a6337afac2b8b9e365f926f5a61aacc71adc5b3c"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3c0fae6c3be832a0a0473ac912810b2877c8cb9d76ca48de1ed31e1c68386575"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b076b6226fb84157e3f7c971a47ff3a679d837cf338547532ab866c57930dbee"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bfce63a9e7834b12b87c64d6b155fdd9b3b96191b6bd334bf37db7ff1fe457f2"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:338ae27d6b8745585f87218a3f23f1512dbf52c26c28e322dbe54bcede54ccb9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e4dd52d80b8c83fdce44e12478ad2e85c64ea965e75d66dbeafb0a3e77308fcc"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:df0be2b576a7abbf737b1575f048c23fb1d769f267ec4358296f31c2479db8f9"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:5bbe06f8eeafd38e5d0a4894ffec89378b6c6a625ff57e3028921f8ff59318ac"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win32.whl", hash = "sha256:dd15ff04ffd7e05ffcb7fe79f1b98041b8ea30ae9234aed2a9168b5797c3effb"}, + {file = "MarkupSafe-2.1.3-cp311-cp311-win_amd64.whl", hash = "sha256:134da1eca9ec0ae528110ccc9e48041e0828d79f24121a1a146161103c76e686"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f698de3fd0c4e6972b92290a45bd9b1536bffe8c6759c62471efaa8acb4c37bc"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aa57bd9cf8ae831a362185ee444e15a93ecb2e344c8e52e4d721ea3ab6ef1823"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffcc3f7c66b5f5b7931a5aa68fc9cecc51e685ef90282f4a82f0f5e9b704ad11"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47d4f1c5f80fc62fdd7777d0d40a2e9dda0a05883ab11374334f6c4de38adffd"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1f67c7038d560d92149c060157d623c542173016c4babc0c1913cca0564b9939"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9aad3c1755095ce347e26488214ef77e0485a3c34a50c5a5e2471dff60b9dd9c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:14ff806850827afd6b07a5f32bd917fb7f45b046ba40c57abdb636674a8b559c"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8f9293864fe09b8149f0cc42ce56e3f0e54de883a9de90cd427f191c346eb2e1"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win32.whl", hash = "sha256:715d3562f79d540f251b99ebd6d8baa547118974341db04f5ad06d5ea3eb8007"}, + {file = "MarkupSafe-2.1.3-cp312-cp312-win_amd64.whl", hash = "sha256:1b8dd8c3fd14349433c79fa8abeb573a55fc0fdd769133baac1f5e07abf54aeb"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8e254ae696c88d98da6555f5ace2279cf7cd5b3f52be2b5cf97feafe883b58d2"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cb0932dc158471523c9637e807d9bfb93e06a95cbf010f1a38b98623b929ef2b"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9402b03f1a1b4dc4c19845e5c749e3ab82d5078d16a2a4c2cd2df62d57bb0707"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ca379055a47383d02a5400cb0d110cef0a776fc644cda797db0c5696cfd7e18e"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:b7ff0f54cb4ff66dd38bebd335a38e2c22c41a8ee45aa608efc890ac3e3931bc"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:c011a4149cfbcf9f03994ec2edffcb8b1dc2d2aede7ca243746df97a5d41ce48"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:56d9f2ecac662ca1611d183feb03a3fa4406469dafe241673d521dd5ae92a155"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win32.whl", hash = "sha256:8758846a7e80910096950b67071243da3e5a20ed2546e6392603c096778d48e0"}, + {file = "MarkupSafe-2.1.3-cp37-cp37m-win_amd64.whl", hash = "sha256:787003c0ddb00500e49a10f2844fac87aa6ce977b90b0feaaf9de23c22508b24"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:2ef12179d3a291be237280175b542c07a36e7f60718296278d8593d21ca937d4"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2c1b19b3aaacc6e57b7e25710ff571c24d6c3613a45e905b1fde04d691b98ee0"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8afafd99945ead6e075b973fefa56379c5b5c53fd8937dad92c662da5d8fd5ee"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8c41976a29d078bb235fea9b2ecd3da465df42a562910f9022f1a03107bd02be"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d080e0a5eb2529460b30190fcfcc4199bd7f827663f858a226a81bc27beaa97e"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:69c0f17e9f5a7afdf2cc9fb2d1ce6aabdb3bafb7f38017c0b77862bcec2bbad8"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:504b320cd4b7eff6f968eddf81127112db685e81f7e36e75f9f84f0df46041c3"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:42de32b22b6b804f42c5d98be4f7e5e977ecdd9ee9b660fda1a3edf03b11792d"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win32.whl", hash = "sha256:ceb01949af7121f9fc39f7d27f91be8546f3fb112c608bc4029aef0bab86a2a5"}, + {file = "MarkupSafe-2.1.3-cp38-cp38-win_amd64.whl", hash = "sha256:1b40069d487e7edb2676d3fbdb2b0829ffa2cd63a2ec26c4938b2d34391b4ecc"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:8023faf4e01efadfa183e863fefde0046de576c6f14659e8782065bcece22198"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6b2b56950d93e41f33b4223ead100ea0fe11f8e6ee5f641eb753ce4b77a7042b"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9dcdfd0eaf283af041973bff14a2e143b8bd64e069f4c383416ecd79a81aab58"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:05fb21170423db021895e1ea1e1f3ab3adb85d1c2333cbc2310f2a26bc77272e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:282c2cb35b5b673bbcadb33a585408104df04f14b2d9b01d4c345a3b92861c2c"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:ab4a0df41e7c16a1392727727e7998a467472d0ad65f3ad5e6e765015df08636"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:7ef3cb2ebbf91e330e3bb937efada0edd9003683db6b57bb108c4001f37a02ea"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:0a4e4a1aff6c7ac4cd55792abf96c915634c2b97e3cc1c7129578aa68ebd754e"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win32.whl", hash = "sha256:fec21693218efe39aa7f8599346e90c705afa52c5b31ae019b2e57e8f6542bb2"}, + {file = "MarkupSafe-2.1.3-cp39-cp39-win_amd64.whl", hash = "sha256:3fd4abcb888d15a94f32b75d8fd18ee162ca0c064f35b11134be77050296d6ba"}, + {file = "MarkupSafe-2.1.3.tar.gz", hash = "sha256:af598ed32d6ae86f1b747b82783958b1a4ab8f617b06fe68795c7f026abbdcad"}, +] + +[[package]] +name = "matplotlib-inline" +version = "0.1.6" +description = "Inline Matplotlib backend for Jupyter" +optional = false +python-versions = ">=3.5" +files = [ + {file = "matplotlib-inline-0.1.6.tar.gz", hash = "sha256:f887e5f10ba98e8d2b150ddcf4702c1e5f8b3a20005eb0f74bfdbd360ee6f304"}, + {file = "matplotlib_inline-0.1.6-py3-none-any.whl", hash = "sha256:f1f41aab5328aa5aaea9b16d083b128102f8712542f819fe7e6a420ff581b311"}, +] + +[package.dependencies] +traitlets = "*" + +[[package]] +name = "memory-profiler" +version = "0.58.0" +description = "A module for monitoring memory usage of a python program" +optional = false +python-versions = ">=3.4" +files = [ + {file = "memory_profiler-0.58.0.tar.gz", hash = "sha256:01385ac0fec944fcf7969814ec4406c6d8a9c66c079d09276723c5a7680f44e5"}, +] + +[package.dependencies] +psutil = "*" + +[[package]] +name = "msgpack" +version = "1.0.7" +description = "MessagePack serializer" +optional = false +python-versions = ">=3.8" +files = [ + {file = "msgpack-1.0.7-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:04ad6069c86e531682f9e1e71b71c1c3937d6014a7c3e9edd2aa81ad58842862"}, + {file = "msgpack-1.0.7-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:cca1b62fe70d761a282496b96a5e51c44c213e410a964bdffe0928e611368329"}, + {file = "msgpack-1.0.7-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:e50ebce52f41370707f1e21a59514e3375e3edd6e1832f5e5235237db933c98b"}, + {file = "msgpack-1.0.7-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4a7b4f35de6a304b5533c238bee86b670b75b03d31b7797929caa7a624b5dda6"}, + {file = "msgpack-1.0.7-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:28efb066cde83c479dfe5a48141a53bc7e5f13f785b92ddde336c716663039ee"}, + {file = "msgpack-1.0.7-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4cb14ce54d9b857be9591ac364cb08dc2d6a5c4318c1182cb1d02274029d590d"}, + {file = "msgpack-1.0.7-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:b573a43ef7c368ba4ea06050a957c2a7550f729c31f11dd616d2ac4aba99888d"}, + {file = "msgpack-1.0.7-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:ccf9a39706b604d884d2cb1e27fe973bc55f2890c52f38df742bc1d79ab9f5e1"}, + {file = "msgpack-1.0.7-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:cb70766519500281815dfd7a87d3a178acf7ce95390544b8c90587d76b227681"}, + {file = "msgpack-1.0.7-cp310-cp310-win32.whl", hash = "sha256:b610ff0f24e9f11c9ae653c67ff8cc03c075131401b3e5ef4b82570d1728f8a9"}, + {file = "msgpack-1.0.7-cp310-cp310-win_amd64.whl", hash = "sha256:a40821a89dc373d6427e2b44b572efc36a2778d3f543299e2f24eb1a5de65415"}, + {file = "msgpack-1.0.7-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:576eb384292b139821c41995523654ad82d1916da6a60cff129c715a6223ea84"}, + {file = "msgpack-1.0.7-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:730076207cb816138cf1af7f7237b208340a2c5e749707457d70705715c93b93"}, + {file = "msgpack-1.0.7-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:85765fdf4b27eb5086f05ac0491090fc76f4f2b28e09d9350c31aac25a5aaff8"}, + {file = "msgpack-1.0.7-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3476fae43db72bd11f29a5147ae2f3cb22e2f1a91d575ef130d2bf49afd21c46"}, + {file = "msgpack-1.0.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d4c80667de2e36970ebf74f42d1088cc9ee7ef5f4e8c35eee1b40eafd33ca5b"}, + {file = "msgpack-1.0.7-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5b0bf0effb196ed76b7ad883848143427a73c355ae8e569fa538365064188b8e"}, + {file = "msgpack-1.0.7-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:f9a7c509542db4eceed3dcf21ee5267ab565a83555c9b88a8109dcecc4709002"}, + {file = "msgpack-1.0.7-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:84b0daf226913133f899ea9b30618722d45feffa67e4fe867b0b5ae83a34060c"}, + {file = "msgpack-1.0.7-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:ec79ff6159dffcc30853b2ad612ed572af86c92b5168aa3fc01a67b0fa40665e"}, + {file = "msgpack-1.0.7-cp311-cp311-win32.whl", hash = "sha256:3e7bf4442b310ff154b7bb9d81eb2c016b7d597e364f97d72b1acc3817a0fdc1"}, + {file = "msgpack-1.0.7-cp311-cp311-win_amd64.whl", hash = "sha256:3f0c8c6dfa6605ab8ff0611995ee30d4f9fcff89966cf562733b4008a3d60d82"}, + {file = "msgpack-1.0.7-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:f0936e08e0003f66bfd97e74ee530427707297b0d0361247e9b4f59ab78ddc8b"}, + {file = "msgpack-1.0.7-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:98bbd754a422a0b123c66a4c341de0474cad4a5c10c164ceed6ea090f3563db4"}, + {file = "msgpack-1.0.7-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:b291f0ee7961a597cbbcc77709374087fa2a9afe7bdb6a40dbbd9b127e79afee"}, + {file = "msgpack-1.0.7-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ebbbba226f0a108a7366bf4b59bf0f30a12fd5e75100c630267d94d7f0ad20e5"}, + {file = "msgpack-1.0.7-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1e2d69948e4132813b8d1131f29f9101bc2c915f26089a6d632001a5c1349672"}, + {file = "msgpack-1.0.7-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bdf38ba2d393c7911ae989c3bbba510ebbcdf4ecbdbfec36272abe350c454075"}, + {file = "msgpack-1.0.7-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:993584fc821c58d5993521bfdcd31a4adf025c7d745bbd4d12ccfecf695af5ba"}, + {file = "msgpack-1.0.7-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:52700dc63a4676669b341ba33520f4d6e43d3ca58d422e22ba66d1736b0a6e4c"}, + {file = "msgpack-1.0.7-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:e45ae4927759289c30ccba8d9fdce62bb414977ba158286b5ddaf8df2cddb5c5"}, + {file = "msgpack-1.0.7-cp312-cp312-win32.whl", hash = "sha256:27dcd6f46a21c18fa5e5deed92a43d4554e3df8d8ca5a47bf0615d6a5f39dbc9"}, + {file = "msgpack-1.0.7-cp312-cp312-win_amd64.whl", hash = "sha256:7687e22a31e976a0e7fc99c2f4d11ca45eff652a81eb8c8085e9609298916dcf"}, + {file = "msgpack-1.0.7-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5b6ccc0c85916998d788b295765ea0e9cb9aac7e4a8ed71d12e7d8ac31c23c95"}, + {file = "msgpack-1.0.7-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:235a31ec7db685f5c82233bddf9858748b89b8119bf4538d514536c485c15fe0"}, + {file = "msgpack-1.0.7-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:cab3db8bab4b7e635c1c97270d7a4b2a90c070b33cbc00c99ef3f9be03d3e1f7"}, + {file = "msgpack-1.0.7-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0bfdd914e55e0d2c9e1526de210f6fe8ffe9705f2b1dfcc4aecc92a4cb4b533d"}, + {file = "msgpack-1.0.7-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:36e17c4592231a7dbd2ed09027823ab295d2791b3b1efb2aee874b10548b7524"}, + {file = "msgpack-1.0.7-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:38949d30b11ae5f95c3c91917ee7a6b239f5ec276f271f28638dec9156f82cfc"}, + {file = "msgpack-1.0.7-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:ff1d0899f104f3921d94579a5638847f783c9b04f2d5f229392ca77fba5b82fc"}, + {file = "msgpack-1.0.7-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:dc43f1ec66eb8440567186ae2f8c447d91e0372d793dfe8c222aec857b81a8cf"}, + {file = "msgpack-1.0.7-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:dd632777ff3beaaf629f1ab4396caf7ba0bdd075d948a69460d13d44357aca4c"}, + {file = "msgpack-1.0.7-cp38-cp38-win32.whl", hash = "sha256:4e71bc4416de195d6e9b4ee93ad3f2f6b2ce11d042b4d7a7ee00bbe0358bd0c2"}, + {file = "msgpack-1.0.7-cp38-cp38-win_amd64.whl", hash = "sha256:8f5b234f567cf76ee489502ceb7165c2a5cecec081db2b37e35332b537f8157c"}, + {file = "msgpack-1.0.7-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:bfef2bb6ef068827bbd021017a107194956918ab43ce4d6dc945ffa13efbc25f"}, + {file = "msgpack-1.0.7-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:484ae3240666ad34cfa31eea7b8c6cd2f1fdaae21d73ce2974211df099a95d81"}, + {file = "msgpack-1.0.7-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3967e4ad1aa9da62fd53e346ed17d7b2e922cba5ab93bdd46febcac39be636fc"}, + {file = "msgpack-1.0.7-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8dd178c4c80706546702c59529ffc005681bd6dc2ea234c450661b205445a34d"}, + {file = "msgpack-1.0.7-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f6ffbc252eb0d229aeb2f9ad051200668fc3a9aaa8994e49f0cb2ffe2b7867e7"}, + {file = "msgpack-1.0.7-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:822ea70dc4018c7e6223f13affd1c5c30c0f5c12ac1f96cd8e9949acddb48a61"}, + {file = "msgpack-1.0.7-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:384d779f0d6f1b110eae74cb0659d9aa6ff35aaf547b3955abf2ab4c901c4819"}, + {file = "msgpack-1.0.7-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:f64e376cd20d3f030190e8c32e1c64582eba56ac6dc7d5b0b49a9d44021b52fd"}, + {file = "msgpack-1.0.7-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5ed82f5a7af3697b1c4786053736f24a0efd0a1b8a130d4c7bfee4b9ded0f08f"}, + {file = "msgpack-1.0.7-cp39-cp39-win32.whl", hash = "sha256:f26a07a6e877c76a88e3cecac8531908d980d3d5067ff69213653649ec0f60ad"}, + {file = "msgpack-1.0.7-cp39-cp39-win_amd64.whl", hash = "sha256:1dc93e8e4653bdb5910aed79f11e165c85732067614f180f70534f056da97db3"}, + {file = "msgpack-1.0.7.tar.gz", hash = "sha256:572efc93db7a4d27e404501975ca6d2d9775705c2d922390d878fcf768d92c87"}, +] + +[[package]] +name = "multidict" +version = "6.0.4" +description = "multidict implementation" +optional = false +python-versions = ">=3.7" +files = [ + {file = "multidict-6.0.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:0b1a97283e0c85772d613878028fec909f003993e1007eafa715b24b377cb9b8"}, + {file = "multidict-6.0.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:eeb6dcc05e911516ae3d1f207d4b0520d07f54484c49dfc294d6e7d63b734171"}, + {file = "multidict-6.0.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d6d635d5209b82a3492508cf5b365f3446afb65ae7ebd755e70e18f287b0adf7"}, + {file = "multidict-6.0.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c048099e4c9e9d615545e2001d3d8a4380bd403e1a0578734e0d31703d1b0c0b"}, + {file = "multidict-6.0.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ea20853c6dbbb53ed34cb4d080382169b6f4554d394015f1bef35e881bf83547"}, + {file = "multidict-6.0.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:16d232d4e5396c2efbbf4f6d4df89bfa905eb0d4dc5b3549d872ab898451f569"}, + {file = "multidict-6.0.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:36c63aaa167f6c6b04ef2c85704e93af16c11d20de1d133e39de6a0e84582a93"}, + {file = "multidict-6.0.4-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:64bdf1086b6043bf519869678f5f2757f473dee970d7abf6da91ec00acb9cb98"}, + {file = "multidict-6.0.4-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:43644e38f42e3af682690876cff722d301ac585c5b9e1eacc013b7a3f7b696a0"}, + {file = "multidict-6.0.4-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:7582a1d1030e15422262de9f58711774e02fa80df0d1578995c76214f6954988"}, + {file = "multidict-6.0.4-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:ddff9c4e225a63a5afab9dd15590432c22e8057e1a9a13d28ed128ecf047bbdc"}, + {file = "multidict-6.0.4-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:ee2a1ece51b9b9e7752e742cfb661d2a29e7bcdba2d27e66e28a99f1890e4fa0"}, + {file = "multidict-6.0.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a2e4369eb3d47d2034032a26c7a80fcb21a2cb22e1173d761a162f11e562caa5"}, + {file = "multidict-6.0.4-cp310-cp310-win32.whl", hash = "sha256:574b7eae1ab267e5f8285f0fe881f17efe4b98c39a40858247720935b893bba8"}, + {file = "multidict-6.0.4-cp310-cp310-win_amd64.whl", hash = "sha256:4dcbb0906e38440fa3e325df2359ac6cb043df8e58c965bb45f4e406ecb162cc"}, + {file = "multidict-6.0.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:0dfad7a5a1e39c53ed00d2dd0c2e36aed4650936dc18fd9a1826a5ae1cad6f03"}, + {file = "multidict-6.0.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:64da238a09d6039e3bd39bb3aee9c21a5e34f28bfa5aa22518581f910ff94af3"}, + {file = "multidict-6.0.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ff959bee35038c4624250473988b24f846cbeb2c6639de3602c073f10410ceba"}, + {file = "multidict-6.0.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:01a3a55bd90018c9c080fbb0b9f4891db37d148a0a18722b42f94694f8b6d4c9"}, + {file = "multidict-6.0.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c5cb09abb18c1ea940fb99360ea0396f34d46566f157122c92dfa069d3e0e982"}, + {file = "multidict-6.0.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:666daae833559deb2d609afa4490b85830ab0dfca811a98b70a205621a6109fe"}, + {file = "multidict-6.0.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:11bdf3f5e1518b24530b8241529d2050014c884cf18b6fc69c0c2b30ca248710"}, + {file = "multidict-6.0.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7d18748f2d30f94f498e852c67d61261c643b349b9d2a581131725595c45ec6c"}, + {file = "multidict-6.0.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:458f37be2d9e4c95e2d8866a851663cbc76e865b78395090786f6cd9b3bbf4f4"}, + {file = "multidict-6.0.4-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:b1a2eeedcead3a41694130495593a559a668f382eee0727352b9a41e1c45759a"}, + {file = "multidict-6.0.4-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:7d6ae9d593ef8641544d6263c7fa6408cc90370c8cb2bbb65f8d43e5b0351d9c"}, + {file = "multidict-6.0.4-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:5979b5632c3e3534e42ca6ff856bb24b2e3071b37861c2c727ce220d80eee9ed"}, + {file = "multidict-6.0.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:dcfe792765fab89c365123c81046ad4103fcabbc4f56d1c1997e6715e8015461"}, + {file = "multidict-6.0.4-cp311-cp311-win32.whl", hash = "sha256:3601a3cece3819534b11d4efc1eb76047488fddd0c85a3948099d5da4d504636"}, + {file = "multidict-6.0.4-cp311-cp311-win_amd64.whl", hash = "sha256:81a4f0b34bd92df3da93315c6a59034df95866014ac08535fc819f043bfd51f0"}, + {file = "multidict-6.0.4-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:67040058f37a2a51ed8ea8f6b0e6ee5bd78ca67f169ce6122f3e2ec80dfe9b78"}, + {file = "multidict-6.0.4-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:853888594621e6604c978ce2a0444a1e6e70c8d253ab65ba11657659dcc9100f"}, + {file = "multidict-6.0.4-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:39ff62e7d0f26c248b15e364517a72932a611a9b75f35b45be078d81bdb86603"}, + {file = "multidict-6.0.4-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:af048912e045a2dc732847d33821a9d84ba553f5c5f028adbd364dd4765092ac"}, + {file = "multidict-6.0.4-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b1e8b901e607795ec06c9e42530788c45ac21ef3aaa11dbd0c69de543bfb79a9"}, + {file = "multidict-6.0.4-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:62501642008a8b9871ddfccbf83e4222cf8ac0d5aeedf73da36153ef2ec222d2"}, + {file = "multidict-6.0.4-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:99b76c052e9f1bc0721f7541e5e8c05db3941eb9ebe7b8553c625ef88d6eefde"}, + {file = "multidict-6.0.4-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:509eac6cf09c794aa27bcacfd4d62c885cce62bef7b2c3e8b2e49d365b5003fe"}, + {file = "multidict-6.0.4-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:21a12c4eb6ddc9952c415f24eef97e3e55ba3af61f67c7bc388dcdec1404a067"}, + {file = "multidict-6.0.4-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:5cad9430ab3e2e4fa4a2ef4450f548768400a2ac635841bc2a56a2052cdbeb87"}, + {file = "multidict-6.0.4-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:ab55edc2e84460694295f401215f4a58597f8f7c9466faec545093045476327d"}, + {file = "multidict-6.0.4-cp37-cp37m-win32.whl", hash = "sha256:5a4dcf02b908c3b8b17a45fb0f15b695bf117a67b76b7ad18b73cf8e92608775"}, + {file = "multidict-6.0.4-cp37-cp37m-win_amd64.whl", hash = "sha256:6ed5f161328b7df384d71b07317f4d8656434e34591f20552c7bcef27b0ab88e"}, + {file = "multidict-6.0.4-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5fc1b16f586f049820c5c5b17bb4ee7583092fa0d1c4e28b5239181ff9532e0c"}, + {file = "multidict-6.0.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1502e24330eb681bdaa3eb70d6358e818e8e8f908a22a1851dfd4e15bc2f8161"}, + {file = "multidict-6.0.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b692f419760c0e65d060959df05f2a531945af31fda0c8a3b3195d4efd06de11"}, + {file = "multidict-6.0.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:45e1ecb0379bfaab5eef059f50115b54571acfbe422a14f668fc8c27ba410e7e"}, + {file = "multidict-6.0.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ddd3915998d93fbcd2566ddf9cf62cdb35c9e093075f862935573d265cf8f65d"}, + {file = "multidict-6.0.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:59d43b61c59d82f2effb39a93c48b845efe23a3852d201ed2d24ba830d0b4cf2"}, + {file = "multidict-6.0.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cc8e1d0c705233c5dd0c5e6460fbad7827d5d36f310a0fadfd45cc3029762258"}, + {file = "multidict-6.0.4-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d6aa0418fcc838522256761b3415822626f866758ee0bc6632c9486b179d0b52"}, + {file = "multidict-6.0.4-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:6748717bb10339c4760c1e63da040f5f29f5ed6e59d76daee30305894069a660"}, + {file = "multidict-6.0.4-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:4d1a3d7ef5e96b1c9e92f973e43aa5e5b96c659c9bc3124acbbd81b0b9c8a951"}, + {file = "multidict-6.0.4-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4372381634485bec7e46718edc71528024fcdc6f835baefe517b34a33c731d60"}, + {file = "multidict-6.0.4-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:fc35cb4676846ef752816d5be2193a1e8367b4c1397b74a565a9d0389c433a1d"}, + {file = "multidict-6.0.4-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:4b9d9e4e2b37daddb5c23ea33a3417901fa7c7b3dee2d855f63ee67a0b21e5b1"}, + {file = "multidict-6.0.4-cp38-cp38-win32.whl", hash = "sha256:e41b7e2b59679edfa309e8db64fdf22399eec4b0b24694e1b2104fb789207779"}, + {file = "multidict-6.0.4-cp38-cp38-win_amd64.whl", hash = "sha256:d6c254ba6e45d8e72739281ebc46ea5eb5f101234f3ce171f0e9f5cc86991480"}, + {file = "multidict-6.0.4-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:16ab77bbeb596e14212e7bab8429f24c1579234a3a462105cda4a66904998664"}, + {file = "multidict-6.0.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bc779e9e6f7fda81b3f9aa58e3a6091d49ad528b11ed19f6621408806204ad35"}, + {file = "multidict-6.0.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4ceef517eca3e03c1cceb22030a3e39cb399ac86bff4e426d4fc6ae49052cc60"}, + {file = "multidict-6.0.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:281af09f488903fde97923c7744bb001a9b23b039a909460d0f14edc7bf59706"}, + {file = "multidict-6.0.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:52f2dffc8acaba9a2f27174c41c9e57f60b907bb9f096b36b1a1f3be71c6284d"}, + {file = "multidict-6.0.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b41156839806aecb3641f3208c0dafd3ac7775b9c4c422d82ee2a45c34ba81ca"}, + {file = "multidict-6.0.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d5e3fc56f88cc98ef8139255cf8cd63eb2c586531e43310ff859d6bb3a6b51f1"}, + {file = "multidict-6.0.4-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:8316a77808c501004802f9beebde51c9f857054a0c871bd6da8280e718444449"}, + {file = "multidict-6.0.4-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:f70b98cd94886b49d91170ef23ec5c0e8ebb6f242d734ed7ed677b24d50c82cf"}, + {file = "multidict-6.0.4-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:bf6774e60d67a9efe02b3616fee22441d86fab4c6d335f9d2051d19d90a40063"}, + {file = "multidict-6.0.4-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:e69924bfcdda39b722ef4d9aa762b2dd38e4632b3641b1d9a57ca9cd18f2f83a"}, + {file = "multidict-6.0.4-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:6b181d8c23da913d4ff585afd1155a0e1194c0b50c54fcfe286f70cdaf2b7176"}, + {file = "multidict-6.0.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:52509b5be062d9eafc8170e53026fbc54cf3b32759a23d07fd935fb04fc22d95"}, + {file = "multidict-6.0.4-cp39-cp39-win32.whl", hash = "sha256:27c523fbfbdfd19c6867af7346332b62b586eed663887392cff78d614f9ec313"}, + {file = "multidict-6.0.4-cp39-cp39-win_amd64.whl", hash = "sha256:33029f5734336aa0d4c0384525da0387ef89148dc7191aae00ca5fb23d7aafc2"}, + {file = "multidict-6.0.4.tar.gz", hash = "sha256:3666906492efb76453c0e7b97f2cf459b0682e7402c0489a95484965dbc1da49"}, +] + +[[package]] +name = "multipledispatch" +version = "0.6.0" +description = "Multiple dispatch" +optional = false +python-versions = "*" +files = [ + {file = "multipledispatch-0.6.0-py2-none-any.whl", hash = "sha256:407e6d8c5fa27075968ba07c4db3ef5f02bea4e871e959570eeb69ee39a6565b"}, + {file = "multipledispatch-0.6.0-py3-none-any.whl", hash = "sha256:a55c512128fb3f7c2efd2533f2550accb93c35f1045242ef74645fc92a2c3cba"}, + {file = "multipledispatch-0.6.0.tar.gz", hash = "sha256:a7ab1451fd0bf9b92cab3edbd7b205622fb767aeefb4fb536c2e3de9e0a38bea"}, +] + +[package.dependencies] +six = "*" + +[[package]] +name = "mypy" +version = "0.812" +description = "Optional static typing for Python" +optional = false +python-versions = ">=3.5" +files = [ + {file = "mypy-0.812-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:a26f8ec704e5a7423c8824d425086705e381b4f1dfdef6e3a1edab7ba174ec49"}, + {file = "mypy-0.812-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:28fb5479c494b1bab244620685e2eb3c3f988d71fd5d64cc753195e8ed53df7c"}, + {file = "mypy-0.812-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:9743c91088d396c1a5a3c9978354b61b0382b4e3c440ce83cf77994a43e8c521"}, + {file = "mypy-0.812-cp35-cp35m-win_amd64.whl", hash = "sha256:d7da2e1d5f558c37d6e8c1246f1aec1e7349e4913d8fb3cb289a35de573fe2eb"}, + {file = "mypy-0.812-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:4eec37370483331d13514c3f55f446fc5248d6373e7029a29ecb7b7494851e7a"}, + {file = "mypy-0.812-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d65cc1df038ef55a99e617431f0553cd77763869eebdf9042403e16089fe746c"}, + {file = "mypy-0.812-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:61a3d5b97955422964be6b3baf05ff2ce7f26f52c85dd88db11d5e03e146a3a6"}, + {file = "mypy-0.812-cp36-cp36m-win_amd64.whl", hash = "sha256:25adde9b862f8f9aac9d2d11971f226bd4c8fbaa89fb76bdadb267ef22d10064"}, + {file = "mypy-0.812-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:552a815579aa1e995f39fd05dde6cd378e191b063f031f2acfe73ce9fb7f9e56"}, + {file = "mypy-0.812-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:499c798053cdebcaa916eef8cd733e5584b5909f789de856b482cd7d069bdad8"}, + {file = "mypy-0.812-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:5873888fff1c7cf5b71efbe80e0e73153fe9212fafdf8e44adfe4c20ec9f82d7"}, + {file = "mypy-0.812-cp37-cp37m-win_amd64.whl", hash = "sha256:9f94aac67a2045ec719ffe6111df543bac7874cee01f41928f6969756e030564"}, + {file = "mypy-0.812-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d23e0ea196702d918b60c8288561e722bf437d82cb7ef2edcd98cfa38905d506"}, + {file = "mypy-0.812-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:674e822aa665b9fd75130c6c5f5ed9564a38c6cea6a6432ce47eafb68ee578c5"}, + {file = "mypy-0.812-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:abf7e0c3cf117c44d9285cc6128856106183938c68fd4944763003decdcfeb66"}, + {file = "mypy-0.812-cp38-cp38-win_amd64.whl", hash = "sha256:0d0a87c0e7e3a9becdfbe936c981d32e5ee0ccda3e0f07e1ef2c3d1a817cf73e"}, + {file = "mypy-0.812-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:7ce3175801d0ae5fdfa79b4f0cfed08807af4d075b402b7e294e6aa72af9aa2a"}, + {file = "mypy-0.812-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:b09669bcda124e83708f34a94606e01b614fa71931d356c1f1a5297ba11f110a"}, + {file = "mypy-0.812-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:33f159443db0829d16f0a8d83d94df3109bb6dd801975fe86bacb9bf71628e97"}, + {file = "mypy-0.812-cp39-cp39-win_amd64.whl", hash = "sha256:3f2aca7f68580dc2508289c729bd49ee929a436208d2b2b6aab15745a70a57df"}, + {file = "mypy-0.812-py3-none-any.whl", hash = "sha256:2f9b3407c58347a452fc0736861593e105139b905cca7d097e413453a1d650b4"}, + {file = "mypy-0.812.tar.gz", hash = "sha256:cd07039aa5df222037005b08fbbfd69b3ab0b0bd7a07d7906de75ae52c4e3119"}, +] + +[package.dependencies] +mypy-extensions = ">=0.4.3,<0.5.0" +typed-ast = ">=1.4.0,<1.5.0" +typing-extensions = ">=3.7.4" + +[package.extras] +dmypy = ["psutil (>=4.0)"] + +[[package]] +name = "mypy-extensions" +version = "0.4.4" +description = "Experimental type system extensions for programs checked with the mypy typechecker." +optional = false +python-versions = ">=2.7" +files = [ + {file = "mypy_extensions-0.4.4.tar.gz", hash = "sha256:c8b707883a96efe9b4bb3aaf0dcc07e7e217d7d8368eec4db4049ee9e142f4fd"}, +] + +[[package]] +name = "numpy" +version = "1.24.4" +description = "Fundamental package for array computing in Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "numpy-1.24.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c0bfb52d2169d58c1cdb8cc1f16989101639b34c7d3ce60ed70b19c63eba0b64"}, + {file = "numpy-1.24.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ed094d4f0c177b1b8e7aa9cba7d6ceed51c0e569a5318ac0ca9a090680a6a1b1"}, + {file = "numpy-1.24.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:79fc682a374c4a8ed08b331bef9c5f582585d1048fa6d80bc6c35bc384eee9b4"}, + {file = "numpy-1.24.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7ffe43c74893dbf38c2b0a1f5428760a1a9c98285553c89e12d70a96a7f3a4d6"}, + {file = "numpy-1.24.4-cp310-cp310-win32.whl", hash = "sha256:4c21decb6ea94057331e111a5bed9a79d335658c27ce2adb580fb4d54f2ad9bc"}, + {file = "numpy-1.24.4-cp310-cp310-win_amd64.whl", hash = "sha256:b4bea75e47d9586d31e892a7401f76e909712a0fd510f58f5337bea9572c571e"}, + {file = "numpy-1.24.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:f136bab9c2cfd8da131132c2cf6cc27331dd6fae65f95f69dcd4ae3c3639c810"}, + {file = "numpy-1.24.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e2926dac25b313635e4d6cf4dc4e51c8c0ebfed60b801c799ffc4c32bf3d1254"}, + {file = "numpy-1.24.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:222e40d0e2548690405b0b3c7b21d1169117391c2e82c378467ef9ab4c8f0da7"}, + {file = "numpy-1.24.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7215847ce88a85ce39baf9e89070cb860c98fdddacbaa6c0da3ffb31b3350bd5"}, + {file = "numpy-1.24.4-cp311-cp311-win32.whl", hash = "sha256:4979217d7de511a8d57f4b4b5b2b965f707768440c17cb70fbf254c4b225238d"}, + {file = "numpy-1.24.4-cp311-cp311-win_amd64.whl", hash = "sha256:b7b1fc9864d7d39e28f41d089bfd6353cb5f27ecd9905348c24187a768c79694"}, + {file = "numpy-1.24.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1452241c290f3e2a312c137a9999cdbf63f78864d63c79039bda65ee86943f61"}, + {file = "numpy-1.24.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:04640dab83f7c6c85abf9cd729c5b65f1ebd0ccf9de90b270cd61935eef0197f"}, + {file = "numpy-1.24.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5425b114831d1e77e4b5d812b69d11d962e104095a5b9c3b641a218abcc050e"}, + {file = "numpy-1.24.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd80e219fd4c71fc3699fc1dadac5dcf4fd882bfc6f7ec53d30fa197b8ee22dc"}, + {file = "numpy-1.24.4-cp38-cp38-win32.whl", hash = "sha256:4602244f345453db537be5314d3983dbf5834a9701b7723ec28923e2889e0bb2"}, + {file = "numpy-1.24.4-cp38-cp38-win_amd64.whl", hash = "sha256:692f2e0f55794943c5bfff12b3f56f99af76f902fc47487bdfe97856de51a706"}, + {file = "numpy-1.24.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:2541312fbf09977f3b3ad449c4e5f4bb55d0dbf79226d7724211acc905049400"}, + {file = "numpy-1.24.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9667575fb6d13c95f1b36aca12c5ee3356bf001b714fc354eb5465ce1609e62f"}, + {file = "numpy-1.24.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f3a86ed21e4f87050382c7bc96571755193c4c1392490744ac73d660e8f564a9"}, + {file = "numpy-1.24.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d11efb4dbecbdf22508d55e48d9c8384db795e1b7b51ea735289ff96613ff74d"}, + {file = "numpy-1.24.4-cp39-cp39-win32.whl", hash = "sha256:6620c0acd41dbcb368610bb2f4d83145674040025e5536954782467100aa8835"}, + {file = "numpy-1.24.4-cp39-cp39-win_amd64.whl", hash = "sha256:befe2bf740fd8373cf56149a5c23a0f601e82869598d41f8e188a0e9869926f8"}, + {file = "numpy-1.24.4-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:31f13e25b4e304632a4619d0e0777662c2ffea99fcae2029556b17d8ff958aef"}, + {file = "numpy-1.24.4-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:95f7ac6540e95bc440ad77f56e520da5bf877f87dca58bd095288dce8940532a"}, + {file = "numpy-1.24.4-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:e98f220aa76ca2a977fe435f5b04d7b3470c0a2e6312907b37ba6068f26787f2"}, + {file = "numpy-1.24.4.tar.gz", hash = "sha256:80f5e3a4e498641401868df4208b74581206afbee7cf7b8329daae82676d9463"}, +] + +[[package]] +name = "objprint" +version = "0.1.4" +description = "A library that can print Python objects in human readable format" +optional = false +python-versions = ">=3.6" +files = [ + {file = "objprint-0.1.4-py3-none-any.whl", hash = "sha256:af46c3b002098db11d9a7340bc728848ec97f3bb278a460169ff9482719a6e9d"}, + {file = "objprint-0.1.4.tar.gz", hash = "sha256:df237e330ff78b9661a9b84d2ee3aa6fcda56d6f185793e3942e78cea7601ff4"}, +] + +[[package]] +name = "openpyxl" +version = "3.1.2" +description = "A Python library to read/write Excel 2010 xlsx/xlsm files" +optional = false +python-versions = ">=3.6" +files = [ + {file = "openpyxl-3.1.2-py2.py3-none-any.whl", hash = "sha256:f91456ead12ab3c6c2e9491cf33ba6d08357d802192379bb482f1033ade496f5"}, + {file = "openpyxl-3.1.2.tar.gz", hash = "sha256:a6f5977418eff3b2d5500d54d9db50c8277a368436f4e4f8ddb1be3422870184"}, +] + +[package.dependencies] +et-xmlfile = "*" + +[[package]] +name = "packaging" +version = "23.2" +description = "Core utilities for Python packages" +optional = false +python-versions = ">=3.7" +files = [ + {file = "packaging-23.2-py3-none-any.whl", hash = "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7"}, + {file = "packaging-23.2.tar.gz", hash = "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5"}, +] + +[[package]] +name = "pandas" +version = "2.0.3" +description = "Powerful data structures for data analysis, time series, and statistics" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pandas-2.0.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e4c7c9f27a4185304c7caf96dc7d91bc60bc162221152de697c98eb0b2648dd8"}, + {file = "pandas-2.0.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:f167beed68918d62bffb6ec64f2e1d8a7d297a038f86d4aed056b9493fca407f"}, + {file = "pandas-2.0.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce0c6f76a0f1ba361551f3e6dceaff06bde7514a374aa43e33b588ec10420183"}, + {file = "pandas-2.0.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba619e410a21d8c387a1ea6e8a0e49bb42216474436245718d7f2e88a2f8d7c0"}, + {file = "pandas-2.0.3-cp310-cp310-win32.whl", hash = "sha256:3ef285093b4fe5058eefd756100a367f27029913760773c8bf1d2d8bebe5d210"}, + {file = "pandas-2.0.3-cp310-cp310-win_amd64.whl", hash = "sha256:9ee1a69328d5c36c98d8e74db06f4ad518a1840e8ccb94a4ba86920986bb617e"}, + {file = "pandas-2.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b084b91d8d66ab19f5bb3256cbd5ea661848338301940e17f4492b2ce0801fe8"}, + {file = "pandas-2.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:37673e3bdf1551b95bf5d4ce372b37770f9529743d2498032439371fc7b7eb26"}, + {file = "pandas-2.0.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b9cb1e14fdb546396b7e1b923ffaeeac24e4cedd14266c3497216dd4448e4f2d"}, + {file = "pandas-2.0.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d9cd88488cceb7635aebb84809d087468eb33551097d600c6dad13602029c2df"}, + {file = "pandas-2.0.3-cp311-cp311-win32.whl", hash = "sha256:694888a81198786f0e164ee3a581df7d505024fbb1f15202fc7db88a71d84ebd"}, + {file = "pandas-2.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:6a21ab5c89dcbd57f78d0ae16630b090eec626360085a4148693def5452d8a6b"}, + {file = "pandas-2.0.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:9e4da0d45e7f34c069fe4d522359df7d23badf83abc1d1cef398895822d11061"}, + {file = "pandas-2.0.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:32fca2ee1b0d93dd71d979726b12b61faa06aeb93cf77468776287f41ff8fdc5"}, + {file = "pandas-2.0.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:258d3624b3ae734490e4d63c430256e716f488c4fcb7c8e9bde2d3aa46c29089"}, + {file = "pandas-2.0.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9eae3dc34fa1aa7772dd3fc60270d13ced7346fcbcfee017d3132ec625e23bb0"}, + {file = "pandas-2.0.3-cp38-cp38-win32.whl", hash = "sha256:f3421a7afb1a43f7e38e82e844e2bca9a6d793d66c1a7f9f0ff39a795bbc5e02"}, + {file = "pandas-2.0.3-cp38-cp38-win_amd64.whl", hash = "sha256:69d7f3884c95da3a31ef82b7618af5710dba95bb885ffab339aad925c3e8ce78"}, + {file = "pandas-2.0.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5247fb1ba347c1261cbbf0fcfba4a3121fbb4029d95d9ef4dc45406620b25c8b"}, + {file = "pandas-2.0.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:81af086f4543c9d8bb128328b5d32e9986e0c84d3ee673a2ac6fb57fd14f755e"}, + {file = "pandas-2.0.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1994c789bf12a7c5098277fb43836ce090f1073858c10f9220998ac74f37c69b"}, + {file = "pandas-2.0.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5ec591c48e29226bcbb316e0c1e9423622bc7a4eaf1ef7c3c9fa1a3981f89641"}, + {file = "pandas-2.0.3-cp39-cp39-win32.whl", hash = "sha256:04dbdbaf2e4d46ca8da896e1805bc04eb85caa9a82e259e8eed00254d5e0c682"}, + {file = "pandas-2.0.3-cp39-cp39-win_amd64.whl", hash = "sha256:1168574b036cd8b93abc746171c9b4f1b83467438a5e45909fed645cf8692dbc"}, + {file = "pandas-2.0.3.tar.gz", hash = "sha256:c02f372a88e0d17f36d3093a644c73cfc1788e876a7c4bcb4020a77512e2043c"}, +] + +[package.dependencies] +numpy = [ + {version = ">=1.20.3", markers = "python_version < \"3.10\""}, + {version = ">=1.21.0", markers = "python_version >= \"3.10\""}, + {version = ">=1.23.2", markers = "python_version >= \"3.11\""}, +] +python-dateutil = ">=2.8.2" +pytz = ">=2020.1" +tzdata = ">=2022.1" + +[package.extras] +all = ["PyQt5 (>=5.15.1)", "SQLAlchemy (>=1.4.16)", "beautifulsoup4 (>=4.9.3)", "bottleneck (>=1.3.2)", "brotlipy (>=0.7.0)", "fastparquet (>=0.6.3)", "fsspec (>=2021.07.0)", "gcsfs (>=2021.07.0)", "html5lib (>=1.1)", "hypothesis (>=6.34.2)", "jinja2 (>=3.0.0)", "lxml (>=4.6.3)", "matplotlib (>=3.6.1)", "numba (>=0.53.1)", "numexpr (>=2.7.3)", "odfpy (>=1.4.1)", "openpyxl (>=3.0.7)", "pandas-gbq (>=0.15.0)", "psycopg2 (>=2.8.6)", "pyarrow (>=7.0.0)", "pymysql (>=1.0.2)", "pyreadstat (>=1.1.2)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)", "python-snappy (>=0.6.0)", "pyxlsb (>=1.0.8)", "qtpy (>=2.2.0)", "s3fs (>=2021.08.0)", "scipy (>=1.7.1)", "tables (>=3.6.1)", "tabulate (>=0.8.9)", "xarray (>=0.21.0)", "xlrd (>=2.0.1)", "xlsxwriter (>=1.4.3)", "zstandard (>=0.15.2)"] +aws = ["s3fs (>=2021.08.0)"] +clipboard = ["PyQt5 (>=5.15.1)", "qtpy (>=2.2.0)"] +compression = ["brotlipy (>=0.7.0)", "python-snappy (>=0.6.0)", "zstandard (>=0.15.2)"] +computation = ["scipy (>=1.7.1)", "xarray (>=0.21.0)"] +excel = ["odfpy (>=1.4.1)", "openpyxl (>=3.0.7)", "pyxlsb (>=1.0.8)", "xlrd (>=2.0.1)", "xlsxwriter (>=1.4.3)"] +feather = ["pyarrow (>=7.0.0)"] +fss = ["fsspec (>=2021.07.0)"] +gcp = ["gcsfs (>=2021.07.0)", "pandas-gbq (>=0.15.0)"] +hdf5 = ["tables (>=3.6.1)"] +html = ["beautifulsoup4 (>=4.9.3)", "html5lib (>=1.1)", "lxml (>=4.6.3)"] +mysql = ["SQLAlchemy (>=1.4.16)", "pymysql (>=1.0.2)"] +output-formatting = ["jinja2 (>=3.0.0)", "tabulate (>=0.8.9)"] +parquet = ["pyarrow (>=7.0.0)"] +performance = ["bottleneck (>=1.3.2)", "numba (>=0.53.1)", "numexpr (>=2.7.1)"] +plot = ["matplotlib (>=3.6.1)"] +postgresql = ["SQLAlchemy (>=1.4.16)", "psycopg2 (>=2.8.6)"] +spss = ["pyreadstat (>=1.1.2)"] +sql-other = ["SQLAlchemy (>=1.4.16)"] +test = ["hypothesis (>=6.34.2)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)"] +xml = ["lxml (>=4.6.3)"] + +[[package]] +name = "parso" +version = "0.8.3" +description = "A Python Parser" +optional = false +python-versions = ">=3.6" +files = [ + {file = "parso-0.8.3-py2.py3-none-any.whl", hash = "sha256:c001d4636cd3aecdaf33cbb40aebb59b094be2a74c556778ef5576c175e19e75"}, + {file = "parso-0.8.3.tar.gz", hash = "sha256:8c07be290bb59f03588915921e29e8a50002acaf2cdc5fa0e0114f91709fafa0"}, +] + +[package.extras] +qa = ["flake8 (==3.8.3)", "mypy (==0.782)"] +testing = ["docopt", "pytest (<6.0.0)"] + +[[package]] +name = "partd" +version = "1.4.1" +description = "Appendable key-value storage" +optional = false +python-versions = ">=3.7" +files = [ + {file = "partd-1.4.1-py3-none-any.whl", hash = "sha256:27e766663d36c161e2827aa3e28541c992f0b9527d3cca047e13fb3acdb989e6"}, + {file = "partd-1.4.1.tar.gz", hash = "sha256:56c25dd49e6fea5727e731203c466c6e092f308d8f0024e199d02f6aa2167f67"}, +] + +[package.dependencies] +locket = "*" +toolz = "*" + +[package.extras] +complete = ["blosc", "numpy (>=1.9.0)", "pandas (>=0.19.0)", "pyzmq"] + +[[package]] +name = "petl" +version = "1.7.14" +description = "A Python package for extracting, transforming and loading tables of data." +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "petl-1.7.14.tar.gz", hash = "sha256:d4802e3c4804bf85f2267a0102fcad35c61e6a37c90d9e1a1674331f35a90a7f"}, +] + +[package.extras] +avro = ["fastavro (>=0.24.0)"] +bcolz = ["bcolz (>=1.2.1)"] +db = ["SQLAlchemy (>=1.3.6,<2.0)"] +hdf5 = ["cython (>=0.29.13)", "numexpr (>=2.6.9)", "numpy (>=1.16.4)", "tables (>=3.5.2)"] +http = ["aiohttp (>=3.6.2)", "requests"] +interval = ["intervaltree (>=3.0.2)"] +numpy = ["numpy (>=1.16.4)"] +pandas = ["pandas (>=0.24.2)"] +remote = ["fsspec (>=0.7.4)"] +smb = ["smbprotocol (>=1.0.1)"] +whoosh = ["whoosh"] +xls = ["xlrd (>=2.0.1)", "xlwt (>=1.3.0)"] +xlsx = ["openpyxl (>=2.6.2)"] +xpath = ["lxml (>=4.4.0)"] + +[[package]] +name = "pexpect" +version = "4.9.0" +description = "Pexpect allows easy control of interactive console applications." +optional = false +python-versions = "*" +files = [ + {file = "pexpect-4.9.0-py2.py3-none-any.whl", hash = "sha256:7236d1e080e4936be2dc3e326cec0af72acf9212a7e1d060210e70a47e253523"}, + {file = "pexpect-4.9.0.tar.gz", hash = "sha256:ee7d41123f3c9911050ea2c2dac107568dc43b2d3b0c7557a33212c398ead30f"}, +] + +[package.dependencies] +ptyprocess = ">=0.5" + +[[package]] +name = "phonenumbers" +version = "8.13.26" +description = "Python version of Google's common library for parsing, formatting, storing and validating international phone numbers." +optional = false +python-versions = "*" +files = [ + {file = "phonenumbers-8.13.26-py2.py3-none-any.whl", hash = "sha256:b2308c9c5750b8f10dd30d94547afd66bce60ac5e93aff227f95740557f32752"}, + {file = "phonenumbers-8.13.26.tar.gz", hash = "sha256:937d70aeceb317f5831dfec28de855a60260ef4a9d551964bec8e7a7d0cf81cd"}, +] + +[[package]] +name = "pickleshare" +version = "0.7.5" +description = "Tiny 'shelve'-like database with concurrency support" +optional = false +python-versions = "*" +files = [ + {file = "pickleshare-0.7.5-py2.py3-none-any.whl", hash = "sha256:9649af414d74d4df115d5d718f82acb59c9d418196b7b4290ed47a12ce62df56"}, + {file = "pickleshare-0.7.5.tar.gz", hash = "sha256:87683d47965c1da65cdacaf31c8441d12b8044cdec9aca500cd78fc2c683afca"}, +] + +[[package]] +name = "pkgutil-resolve-name" +version = "1.3.10" +description = "Resolve a name to an object." +optional = false +python-versions = ">=3.6" +files = [ + {file = "pkgutil_resolve_name-1.3.10-py3-none-any.whl", hash = "sha256:ca27cc078d25c5ad71a9de0a7a330146c4e014c2462d9af19c6b828280649c5e"}, + {file = "pkgutil_resolve_name-1.3.10.tar.gz", hash = "sha256:357d6c9e6a755653cfd78893817c0853af365dd51ec97f3d358a819373bbd174"}, +] + +[[package]] +name = "pluggy" +version = "1.3.0" +description = "plugin and hook calling mechanisms for python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pluggy-1.3.0-py3-none-any.whl", hash = "sha256:d89c696a773f8bd377d18e5ecda92b7a3793cbe66c87060a6fb58c7b6e1061f7"}, + {file = "pluggy-1.3.0.tar.gz", hash = "sha256:cf61ae8f126ac6f7c451172cf30e3e43d3ca77615509771b3a984a0730651e12"}, +] + +[package.extras] +dev = ["pre-commit", "tox"] +testing = ["pytest", "pytest-benchmark"] + +[[package]] +name = "pp-ez" +version = "0.2.0" +description = "A short alias for the pprintpp or pprint module" +optional = false +python-versions = "*" +files = [ + {file = "pp-ez-0.2.0.tar.gz", hash = "sha256:87b9e5a0fdd7ced5b46d1b61706113cac9ec1670a76b10d308334db44c20cc29"}, + {file = "pp_ez-0.2.0-py2.py3-none-any.whl", hash = "sha256:554dc28eff8b2df864fcd3326bd5fa0ec3f2b85bc9590fd96771a7c49e45c121"}, +] + +[[package]] +name = "pprintpp" +version = "0.4.0" +description = "A drop-in replacement for pprint that's actually pretty" +optional = false +python-versions = "*" +files = [ + {file = "pprintpp-0.4.0-py2.py3-none-any.whl", hash = "sha256:b6b4dcdd0c0c0d75e4d7b2f21a9e933e5b2ce62b26e1a54537f9651ae5a5c01d"}, + {file = "pprintpp-0.4.0.tar.gz", hash = "sha256:ea826108e2c7f49dc6d66c752973c3fc9749142a798d6b254e1e301cfdbc6403"}, +] + +[[package]] +name = "prompt-toolkit" +version = "3.0.42" +description = "Library for building powerful interactive command lines in Python" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "prompt_toolkit-3.0.42-py3-none-any.whl", hash = "sha256:3b50b5fc50660dc8e39dfe464b170959ad82ff185ffa53bfd3be02222e7156a1"}, + {file = "prompt_toolkit-3.0.42.tar.gz", hash = "sha256:bfbf7d6ea9744e4ec94c9a69539e8106c77a2a607d728ded87c9182a4aec39be"}, +] + +[package.dependencies] +wcwidth = "*" + +[[package]] +name = "psutil" +version = "5.9.6" +description = "Cross-platform lib for process and system monitoring in Python." +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" +files = [ + {file = "psutil-5.9.6-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:fb8a697f11b0f5994550555fcfe3e69799e5b060c8ecf9e2f75c69302cc35c0d"}, + {file = "psutil-5.9.6-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:91ecd2d9c00db9817a4b4192107cf6954addb5d9d67a969a4f436dbc9200f88c"}, + {file = "psutil-5.9.6-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:10e8c17b4f898d64b121149afb136c53ea8b68c7531155147867b7b1ac9e7e28"}, + {file = "psutil-5.9.6-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:18cd22c5db486f33998f37e2bb054cc62fd06646995285e02a51b1e08da97017"}, + {file = "psutil-5.9.6-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:ca2780f5e038379e520281e4c032dddd086906ddff9ef0d1b9dcf00710e5071c"}, + {file = "psutil-5.9.6-cp27-none-win32.whl", hash = "sha256:70cb3beb98bc3fd5ac9ac617a327af7e7f826373ee64c80efd4eb2856e5051e9"}, + {file = "psutil-5.9.6-cp27-none-win_amd64.whl", hash = "sha256:51dc3d54607c73148f63732c727856f5febec1c7c336f8f41fcbd6315cce76ac"}, + {file = "psutil-5.9.6-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:c69596f9fc2f8acd574a12d5f8b7b1ba3765a641ea5d60fb4736bf3c08a8214a"}, + {file = "psutil-5.9.6-cp36-abi3-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:92e0cc43c524834af53e9d3369245e6cc3b130e78e26100d1f63cdb0abeb3d3c"}, + {file = "psutil-5.9.6-cp36-abi3-manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:748c9dd2583ed86347ed65d0035f45fa8c851e8d90354c122ab72319b5f366f4"}, + {file = "psutil-5.9.6-cp36-cp36m-win32.whl", hash = "sha256:3ebf2158c16cc69db777e3c7decb3c0f43a7af94a60d72e87b2823aebac3d602"}, + {file = "psutil-5.9.6-cp36-cp36m-win_amd64.whl", hash = "sha256:ff18b8d1a784b810df0b0fff3bcb50ab941c3b8e2c8de5726f9c71c601c611aa"}, + {file = "psutil-5.9.6-cp37-abi3-win32.whl", hash = "sha256:a6f01f03bf1843280f4ad16f4bde26b817847b4c1a0db59bf6419807bc5ce05c"}, + {file = "psutil-5.9.6-cp37-abi3-win_amd64.whl", hash = "sha256:6e5fb8dc711a514da83098bc5234264e551ad980cec5f85dabf4d38ed6f15e9a"}, + {file = "psutil-5.9.6-cp38-abi3-macosx_11_0_arm64.whl", hash = "sha256:daecbcbd29b289aac14ece28eca6a3e60aa361754cf6da3dfb20d4d32b6c7f57"}, + {file = "psutil-5.9.6.tar.gz", hash = "sha256:e4b92ddcd7dd4cdd3f900180ea1e104932c7bce234fb88976e2a3b296441225a"}, +] + +[package.extras] +test = ["enum34", "ipaddress", "mock", "pywin32", "wmi"] + +[[package]] +name = "psycopg2-binary" +version = "2.9.9" +description = "psycopg2 - Python-PostgreSQL Database Adapter" +optional = false +python-versions = ">=3.7" +files = [ + {file = "psycopg2-binary-2.9.9.tar.gz", hash = "sha256:7f01846810177d829c7692f1f5ada8096762d9172af1b1a28d4ab5b77c923c1c"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c2470da5418b76232f02a2fcd2229537bb2d5a7096674ce61859c3229f2eb202"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c6af2a6d4b7ee9615cbb162b0738f6e1fd1f5c3eda7e5da17861eacf4c717ea7"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:75723c3c0fbbf34350b46a3199eb50638ab22a0228f93fb472ef4d9becc2382b"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:83791a65b51ad6ee6cf0845634859d69a038ea9b03d7b26e703f94c7e93dbcf9"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0ef4854e82c09e84cc63084a9e4ccd6d9b154f1dbdd283efb92ecd0b5e2b8c84"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ed1184ab8f113e8d660ce49a56390ca181f2981066acc27cf637d5c1e10ce46e"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d2997c458c690ec2bc6b0b7ecbafd02b029b7b4283078d3b32a852a7ce3ddd98"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:b58b4710c7f4161b5e9dcbe73bb7c62d65670a87df7bcce9e1faaad43e715245"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:0c009475ee389757e6e34611d75f6e4f05f0cf5ebb76c6037508318e1a1e0d7e"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:8dbf6d1bc73f1d04ec1734bae3b4fb0ee3cb2a493d35ede9badbeb901fb40f6f"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-win32.whl", hash = "sha256:3f78fd71c4f43a13d342be74ebbc0666fe1f555b8837eb113cb7416856c79682"}, + {file = "psycopg2_binary-2.9.9-cp310-cp310-win_amd64.whl", hash = "sha256:876801744b0dee379e4e3c38b76fc89f88834bb15bf92ee07d94acd06ec890a0"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ee825e70b1a209475622f7f7b776785bd68f34af6e7a46e2e42f27b659b5bc26"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:1ea665f8ce695bcc37a90ee52de7a7980be5161375d42a0b6c6abedbf0d81f0f"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:143072318f793f53819048fdfe30c321890af0c3ec7cb1dfc9cc87aa88241de2"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c332c8d69fb64979ebf76613c66b985414927a40f8defa16cf1bc028b7b0a7b0"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f7fc5a5acafb7d6ccca13bfa8c90f8c51f13d8fb87d95656d3950f0158d3ce53"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:977646e05232579d2e7b9c59e21dbe5261f403a88417f6a6512e70d3f8a046be"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:b6356793b84728d9d50ead16ab43c187673831e9d4019013f1402c41b1db9b27"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:bc7bb56d04601d443f24094e9e31ae6deec9ccb23581f75343feebaf30423359"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:77853062a2c45be16fd6b8d6de2a99278ee1d985a7bd8b103e97e41c034006d2"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:78151aa3ec21dccd5cdef6c74c3e73386dcdfaf19bced944169697d7ac7482fc"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-win32.whl", hash = "sha256:dc4926288b2a3e9fd7b50dc6a1909a13bbdadfc67d93f3374d984e56f885579d"}, + {file = "psycopg2_binary-2.9.9-cp311-cp311-win_amd64.whl", hash = "sha256:b76bedd166805480ab069612119ea636f5ab8f8771e640ae103e05a4aae3e417"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:8532fd6e6e2dc57bcb3bc90b079c60de896d2128c5d9d6f24a63875a95a088cf"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:b0605eaed3eb239e87df0d5e3c6489daae3f7388d455d0c0b4df899519c6a38d"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f8544b092a29a6ddd72f3556a9fcf249ec412e10ad28be6a0c0d948924f2212"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2d423c8d8a3c82d08fe8af900ad5b613ce3632a1249fd6a223941d0735fce493"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2e5afae772c00980525f6d6ecf7cbca55676296b580c0e6abb407f15f3706996"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6e6f98446430fdf41bd36d4faa6cb409f5140c1c2cf58ce0bbdaf16af7d3f119"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:c77e3d1862452565875eb31bdb45ac62502feabbd53429fdc39a1cc341d681ba"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:cb16c65dcb648d0a43a2521f2f0a2300f40639f6f8c1ecbc662141e4e3e1ee07"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:911dda9c487075abd54e644ccdf5e5c16773470a6a5d3826fda76699410066fb"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:57fede879f08d23c85140a360c6a77709113efd1c993923c59fde17aa27599fe"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-win32.whl", hash = "sha256:64cf30263844fa208851ebb13b0732ce674d8ec6a0c86a4e160495d299ba3c93"}, + {file = "psycopg2_binary-2.9.9-cp312-cp312-win_amd64.whl", hash = "sha256:81ff62668af011f9a48787564ab7eded4e9fb17a4a6a74af5ffa6a457400d2ab"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:2293b001e319ab0d869d660a704942c9e2cce19745262a8aba2115ef41a0a42a"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:03ef7df18daf2c4c07e2695e8cfd5ee7f748a1d54d802330985a78d2a5a6dca9"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a602ea5aff39bb9fac6308e9c9d82b9a35c2bf288e184a816002c9fae930b77"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8359bf4791968c5a78c56103702000105501adb557f3cf772b2c207284273984"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:275ff571376626195ab95a746e6a04c7df8ea34638b99fc11160de91f2fef503"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:f9b5571d33660d5009a8b3c25dc1db560206e2d2f89d3df1cb32d72c0d117d52"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:420f9bbf47a02616e8554e825208cb947969451978dceb77f95ad09c37791dae"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:4154ad09dac630a0f13f37b583eae260c6aa885d67dfbccb5b02c33f31a6d420"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:a148c5d507bb9b4f2030a2025c545fccb0e1ef317393eaba42e7eabd28eb6041"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-win32.whl", hash = "sha256:68fc1f1ba168724771e38bee37d940d2865cb0f562380a1fb1ffb428b75cb692"}, + {file = "psycopg2_binary-2.9.9-cp37-cp37m-win_amd64.whl", hash = "sha256:281309265596e388ef483250db3640e5f414168c5a67e9c665cafce9492eda2f"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:60989127da422b74a04345096c10d416c2b41bd7bf2a380eb541059e4e999980"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:246b123cc54bb5361588acc54218c8c9fb73068bf227a4a531d8ed56fa3ca7d6"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:34eccd14566f8fe14b2b95bb13b11572f7c7d5c36da61caf414d23b91fcc5d94"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:18d0ef97766055fec15b5de2c06dd8e7654705ce3e5e5eed3b6651a1d2a9a152"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d3f82c171b4ccd83bbaf35aa05e44e690113bd4f3b7b6cc54d2219b132f3ae55"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ead20f7913a9c1e894aebe47cccf9dc834e1618b7aa96155d2091a626e59c972"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:ca49a8119c6cbd77375ae303b0cfd8c11f011abbbd64601167ecca18a87e7cdd"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:323ba25b92454adb36fa425dc5cf6f8f19f78948cbad2e7bc6cdf7b0d7982e59"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:1236ed0952fbd919c100bc839eaa4a39ebc397ed1c08a97fc45fee2a595aa1b3"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:729177eaf0aefca0994ce4cffe96ad3c75e377c7b6f4efa59ebf003b6d398716"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-win32.whl", hash = "sha256:804d99b24ad523a1fe18cc707bf741670332f7c7412e9d49cb5eab67e886b9b5"}, + {file = "psycopg2_binary-2.9.9-cp38-cp38-win_amd64.whl", hash = "sha256:a6cdcc3ede532f4a4b96000b6362099591ab4a3e913d70bcbac2b56c872446f7"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:72dffbd8b4194858d0941062a9766f8297e8868e1dd07a7b36212aaa90f49472"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:30dcc86377618a4c8f3b72418df92e77be4254d8f89f14b8e8f57d6d43603c0f"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:31a34c508c003a4347d389a9e6fcc2307cc2150eb516462a7a17512130de109e"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:15208be1c50b99203fe88d15695f22a5bed95ab3f84354c494bcb1d08557df67"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1873aade94b74715be2246321c8650cabf5a0d098a95bab81145ffffa4c13876"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a58c98a7e9c021f357348867f537017057c2ed7f77337fd914d0bedb35dace7"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4686818798f9194d03c9129a4d9a702d9e113a89cb03bffe08c6cf799e053291"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:ebdc36bea43063116f0486869652cb2ed7032dbc59fbcb4445c4862b5c1ecf7f"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:ca08decd2697fdea0aea364b370b1249d47336aec935f87b8bbfd7da5b2ee9c1"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ac05fb791acf5e1a3e39402641827780fe44d27e72567a000412c648a85ba860"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-win32.whl", hash = "sha256:9dba73be7305b399924709b91682299794887cbbd88e38226ed9f6712eabee90"}, + {file = "psycopg2_binary-2.9.9-cp39-cp39-win_amd64.whl", hash = "sha256:f7ae5d65ccfbebdfa761585228eb4d0df3a8b15cfb53bd953e713e09fbb12957"}, +] + +[[package]] +name = "ptyprocess" +version = "0.7.0" +description = "Run a subprocess in a pseudo terminal" +optional = false +python-versions = "*" +files = [ + {file = "ptyprocess-0.7.0-py2.py3-none-any.whl", hash = "sha256:4b41f3967fce3af57cc7e94b888626c18bf37a083e3651ca8feeb66d492fef35"}, + {file = "ptyprocess-0.7.0.tar.gz", hash = "sha256:5c5d0a3b48ceee0b48485e0c26037c0acd7d29765ca3fbb5cb3831d347423220"}, +] + +[[package]] +name = "pure-eval" +version = "0.2.2" +description = "Safely evaluate AST nodes without side effects" +optional = false +python-versions = "*" +files = [ + {file = "pure_eval-0.2.2-py3-none-any.whl", hash = "sha256:01eaab343580944bc56080ebe0a674b39ec44a945e6d09ba7db3cb8cec289350"}, + {file = "pure_eval-0.2.2.tar.gz", hash = "sha256:2b45320af6dfaa1750f543d714b6d1c520a1688dec6fd24d339063ce0aaa9ac3"}, +] + +[package.extras] +tests = ["pytest"] + +[[package]] +name = "pycparser" +version = "2.21" +description = "C parser in Python" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "pycparser-2.21-py2.py3-none-any.whl", hash = "sha256:8ee45429555515e1f6b185e78100aea234072576aa43ab53aefcae078162fca9"}, + {file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206"}, +] + +[[package]] +name = "pygments" +version = "2.17.2" +description = "Pygments is a syntax highlighting package written in Python." +optional = false +python-versions = ">=3.7" +files = [ + {file = "pygments-2.17.2-py3-none-any.whl", hash = "sha256:b27c2826c47d0f3219f29554824c30c5e8945175d888647acd804ddd04af846c"}, + {file = "pygments-2.17.2.tar.gz", hash = "sha256:da46cec9fd2de5be3a8a784f434e4c4ab670b4ff54d605c4c2717e9d49c4c367"}, +] + +[package.extras] +plugins = ["importlib-metadata"] +windows-terminal = ["colorama (>=0.4.6)"] + +[[package]] +name = "pymongo" +version = "4.6.1" +description = "Python driver for MongoDB " +optional = false +python-versions = ">=3.7" +files = [ + {file = "pymongo-4.6.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:4344c30025210b9fa80ec257b0e0aab5aa1d5cca91daa70d82ab97b482cc038e"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux1_i686.whl", hash = "sha256:1c5654bb8bb2bdb10e7a0bc3c193dd8b49a960b9eebc4381ff5a2043f4c3c441"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux2014_aarch64.whl", hash = "sha256:eaf2f65190c506def2581219572b9c70b8250615dc918b3b7c218361a51ec42e"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux2014_i686.whl", hash = "sha256:262356ea5fcb13d35fb2ab6009d3927bafb9504ef02339338634fffd8a9f1ae4"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux2014_ppc64le.whl", hash = "sha256:2dd2f6960ee3c9360bed7fb3c678be0ca2d00f877068556785ec2eb6b73d2414"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux2014_s390x.whl", hash = "sha256:ff925f1cca42e933376d09ddc254598f8c5fcd36efc5cac0118bb36c36217c41"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux2014_x86_64.whl", hash = "sha256:3cadf7f4c8e94d8a77874b54a63c80af01f4d48c4b669c8b6867f86a07ba994f"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:55dac73316e7e8c2616ba2e6f62b750918e9e0ae0b2053699d66ca27a7790105"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:154b361dcb358ad377d5d40df41ee35f1cc14c8691b50511547c12404f89b5cb"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2940aa20e9cc328e8ddeacea8b9a6f5ddafe0b087fedad928912e787c65b4909"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:010bc9aa90fd06e5cc52c8fac2c2fd4ef1b5f990d9638548dde178005770a5e8"}, + {file = "pymongo-4.6.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e470fa4bace5f50076c32f4b3cc182b31303b4fefb9b87f990144515d572820b"}, + {file = "pymongo-4.6.1-cp310-cp310-win32.whl", hash = "sha256:da08ea09eefa6b960c2dd9a68ec47949235485c623621eb1d6c02b46765322ac"}, + {file = "pymongo-4.6.1-cp310-cp310-win_amd64.whl", hash = "sha256:13d613c866f9f07d51180f9a7da54ef491d130f169e999c27e7633abe8619ec9"}, + {file = "pymongo-4.6.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:6a0ae7a48a6ef82ceb98a366948874834b86c84e288dbd55600c1abfc3ac1d88"}, + {file = "pymongo-4.6.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5bd94c503271e79917b27c6e77f7c5474da6930b3fb9e70a12e68c2dff386b9a"}, + {file = "pymongo-4.6.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2d4ccac3053b84a09251da8f5350bb684cbbf8c8c01eda6b5418417d0a8ab198"}, + {file = "pymongo-4.6.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:349093675a2d3759e4fb42b596afffa2b2518c890492563d7905fac503b20daa"}, + {file = "pymongo-4.6.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:88beb444fb438385e53dc9110852910ec2a22f0eab7dd489e827038fdc19ed8d"}, + {file = "pymongo-4.6.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d8e62d06e90f60ea2a3d463ae51401475568b995bafaffd81767d208d84d7bb1"}, + {file = "pymongo-4.6.1-cp311-cp311-win32.whl", hash = "sha256:5556e306713e2522e460287615d26c0af0fe5ed9d4f431dad35c6624c5d277e9"}, + {file = "pymongo-4.6.1-cp311-cp311-win_amd64.whl", hash = "sha256:b10d8cda9fc2fcdcfa4a000aa10413a2bf8b575852cd07cb8a595ed09689ca98"}, + {file = "pymongo-4.6.1-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:b435b13bb8e36be11b75f7384a34eefe487fe87a6267172964628e2b14ecf0a7"}, + {file = "pymongo-4.6.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e438417ce1dc5b758742e12661d800482200b042d03512a8f31f6aaa9137ad40"}, + {file = "pymongo-4.6.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8b47ebd89e69fbf33d1c2df79759d7162fc80c7652dacfec136dae1c9b3afac7"}, + {file = "pymongo-4.6.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bbed8cccebe1169d45cedf00461b2842652d476d2897fd1c42cf41b635d88746"}, + {file = "pymongo-4.6.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c30a9e06041fbd7a7590693ec5e407aa8737ad91912a1e70176aff92e5c99d20"}, + {file = "pymongo-4.6.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b8729dbf25eb32ad0dc0b9bd5e6a0d0b7e5c2dc8ec06ad171088e1896b522a74"}, + {file = "pymongo-4.6.1-cp312-cp312-win32.whl", hash = "sha256:3177f783ae7e08aaf7b2802e0df4e4b13903520e8380915e6337cdc7a6ff01d8"}, + {file = "pymongo-4.6.1-cp312-cp312-win_amd64.whl", hash = "sha256:00c199e1c593e2c8b033136d7a08f0c376452bac8a896c923fcd6f419e07bdd2"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:13552ca505366df74e3e2f0a4f27c363928f3dff0eef9f281eb81af7f29bc3c5"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:77e0df59b1a4994ad30c6d746992ae887f9756a43fc25dec2db515d94cf0222d"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:3a7f02a58a0c2912734105e05dedbee4f7507e6f1bd132ebad520be0b11d46fd"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux2014_i686.whl", hash = "sha256:026a24a36394dc8930cbcb1d19d5eb35205ef3c838a7e619e04bd170713972e7"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux2014_ppc64le.whl", hash = "sha256:3b287e814a01deddb59b88549c1e0c87cefacd798d4afc0c8bd6042d1c3d48aa"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:9a710c184ba845afb05a6f876edac8f27783ba70e52d5eaf939f121fc13b2f59"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux2014_x86_64.whl", hash = "sha256:30b2c9caf3e55c2e323565d1f3b7e7881ab87db16997dc0cbca7c52885ed2347"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ff62ba8ff70f01ab4fe0ae36b2cb0b5d1f42e73dfc81ddf0758cd9f77331ad25"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:547dc5d7f834b1deefda51aedb11a7af9c51c45e689e44e14aa85d44147c7657"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1de3c6faf948f3edd4e738abdb4b76572b4f4fdfc1fed4dad02427e70c5a6219"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a2831e05ce0a4df10c4ac5399ef50b9a621f90894c2a4d2945dc5658765514ed"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:144a31391a39a390efce0c5ebcaf4bf112114af4384c90163f402cec5ede476b"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:33bb16a07d3cc4e0aea37b242097cd5f7a156312012455c2fa8ca396953b11c4"}, + {file = "pymongo-4.6.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:b7b1a83ce514700276a46af3d9e481ec381f05b64939effc9065afe18456a6b9"}, + {file = "pymongo-4.6.1-cp37-cp37m-win32.whl", hash = "sha256:3071ec998cc3d7b4944377e5f1217c2c44b811fae16f9a495c7a1ce9b42fb038"}, + {file = "pymongo-4.6.1-cp37-cp37m-win_amd64.whl", hash = "sha256:2346450a075625c4d6166b40a013b605a38b6b6168ce2232b192a37fb200d588"}, + {file = "pymongo-4.6.1-cp38-cp38-macosx_11_0_universal2.whl", hash = "sha256:061598cbc6abe2f382ab64c9caa83faa2f4c51256f732cdd890bcc6e63bfb67e"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:d483793a384c550c2d12cb794ede294d303b42beff75f3b3081f57196660edaf"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:f9756f1d25454ba6a3c2f1ef8b7ddec23e5cdeae3dc3c3377243ae37a383db00"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:1ed23b0e2dac6f84f44c8494fbceefe6eb5c35db5c1099f56ab78fc0d94ab3af"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux2014_i686.whl", hash = "sha256:3d18a9b9b858ee140c15c5bfcb3e66e47e2a70a03272c2e72adda2482f76a6ad"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux2014_ppc64le.whl", hash = "sha256:c258dbacfff1224f13576147df16ce3c02024a0d792fd0323ac01bed5d3c545d"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:f7acc03a4f1154ba2643edeb13658d08598fe6e490c3dd96a241b94f09801626"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux2014_x86_64.whl", hash = "sha256:76013fef1c9cd1cd00d55efde516c154aa169f2bf059b197c263a255ba8a9ddf"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3f0e6a6c807fa887a0c51cc24fe7ea51bb9e496fe88f00d7930063372c3664c3"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dd1fa413f8b9ba30140de198e4f408ffbba6396864c7554e0867aa7363eb58b2"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d219b4508f71d762368caec1fc180960569766049bbc4d38174f05e8ef2fe5b"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:27b81ecf18031998ad7db53b960d1347f8f29e8b7cb5ea7b4394726468e4295e"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:56816e43c92c2fa8c11dc2a686f0ca248bea7902f4a067fa6cbc77853b0f041e"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:ef801027629c5b511cf2ba13b9be29bfee36ae834b2d95d9877818479cdc99ea"}, + {file = "pymongo-4.6.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:d4c2be9760b112b1caf649b4977b81b69893d75aa86caf4f0f398447be871f3c"}, + {file = "pymongo-4.6.1-cp38-cp38-win32.whl", hash = "sha256:39d77d8bbb392fa443831e6d4ae534237b1f4eee6aa186f0cdb4e334ba89536e"}, + {file = "pymongo-4.6.1-cp38-cp38-win_amd64.whl", hash = "sha256:4497d49d785482cc1a44a0ddf8830b036a468c088e72a05217f5b60a9e025012"}, + {file = "pymongo-4.6.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:69247f7a2835fc0984bbf0892e6022e9a36aec70e187fcfe6cae6a373eb8c4de"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:7bb0e9049e81def6829d09558ad12d16d0454c26cabe6efc3658e544460688d9"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:6a1810c2cbde714decf40f811d1edc0dae45506eb37298fd9d4247b8801509fe"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:e2aced6fb2f5261b47d267cb40060b73b6527e64afe54f6497844c9affed5fd0"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux2014_i686.whl", hash = "sha256:d0355cff58a4ed6d5e5f6b9c3693f52de0784aa0c17119394e2a8e376ce489d4"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux2014_ppc64le.whl", hash = "sha256:3c74f4725485f0a7a3862cfd374cc1b740cebe4c133e0c1425984bcdcce0f4bb"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:9c79d597fb3a7c93d7c26924db7497eba06d58f88f58e586aa69b2ad89fee0f8"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux2014_x86_64.whl", hash = "sha256:8ec75f35f62571a43e31e7bd11749d974c1b5cd5ea4a8388725d579263c0fdf6"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5e641f931c5cd95b376fd3c59db52770e17bec2bf86ef16cc83b3906c054845"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9aafd036f6f2e5ad109aec92f8dbfcbe76cff16bad683eb6dd18013739c0b3ae"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f2b856518bfcfa316c8dae3d7b412aecacf2e8ba30b149f5eb3b63128d703b9"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5ec31adc2e988fd7db3ab509954791bbc5a452a03c85e45b804b4bfc31fa221d"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9167e735379ec43d8eafa3fd675bfbb12e2c0464f98960586e9447d2cf2c7a83"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1461199b07903fc1424709efafe379205bf5f738144b1a50a08b0396357b5abf"}, + {file = "pymongo-4.6.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:3094c7d2f820eecabadae76bfec02669567bbdd1730eabce10a5764778564f7b"}, + {file = "pymongo-4.6.1-cp39-cp39-win32.whl", hash = "sha256:c91ea3915425bd4111cb1b74511cdc56d1d16a683a48bf2a5a96b6a6c0f297f7"}, + {file = "pymongo-4.6.1-cp39-cp39-win_amd64.whl", hash = "sha256:ef102a67ede70e1721fe27f75073b5314911dbb9bc27cde0a1c402a11531e7bd"}, + {file = "pymongo-4.6.1.tar.gz", hash = "sha256:31dab1f3e1d0cdd57e8df01b645f52d43cc1b653ed3afd535d2891f4fc4f9712"}, +] + +[package.dependencies] +dnspython = ">=1.16.0,<3.0.0" + +[package.extras] +aws = ["pymongo-auth-aws (<2.0.0)"] +encryption = ["certifi", "pymongo[aws]", "pymongocrypt (>=1.6.0,<2.0.0)"] +gssapi = ["pykerberos", "winkerberos (>=0.5.0)"] +ocsp = ["certifi", "cryptography (>=2.5)", "pyopenssl (>=17.2.0)", "requests (<3.0.0)", "service-identity (>=18.1.0)"] +snappy = ["python-snappy"] +test = ["pytest (>=7)"] +zstd = ["zstandard"] + +[[package]] +name = "pyparsing" +version = "3.1.1" +description = "pyparsing module - Classes and methods to define and execute parsing grammars" +optional = false +python-versions = ">=3.6.8" +files = [ + {file = "pyparsing-3.1.1-py3-none-any.whl", hash = "sha256:32c7c0b711493c72ff18a981d24f28aaf9c1fb7ed5e9667c9e84e3db623bdbfb"}, + {file = "pyparsing-3.1.1.tar.gz", hash = "sha256:ede28a1a32462f5a9705e07aea48001a08f7cf81a021585011deba701581a0db"}, +] + +[package.extras] +diagrams = ["jinja2", "railroad-diagrams"] + +[[package]] +name = "pyproj" +version = "3.5.0" +description = "Python interface to PROJ (cartographic projections and coordinate transformations library)" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pyproj-3.5.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6475ce653880938468a1a1b7321267243909e34b972ba9e53d5982c41d555918"}, + {file = "pyproj-3.5.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:61e4ad57d89b03a7b173793b31bca8ee110112cde1937ef0f42a70b9120c827d"}, + {file = "pyproj-3.5.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7bdd2021bb6f7f346bfe1d2a358aa109da017d22c4704af2d994e7c7ee0a7a53"}, + {file = "pyproj-3.5.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5674923351e76222e2c10c58b5e1ac119d7a46b270d822c463035971b06f724b"}, + {file = "pyproj-3.5.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cd5e2b6aa255023c4acd0b977590f1f7cc801ba21b4d806fcf6dfac3474ebb83"}, + {file = "pyproj-3.5.0-cp310-cp310-win32.whl", hash = "sha256:6f316a66031a14e9c5a88c91f8b77aa97f5454895674541ed6ab630b682be35d"}, + {file = "pyproj-3.5.0-cp310-cp310-win_amd64.whl", hash = "sha256:f7c2f4d9681e810cf40239caaca00079930a6d9ee6591139b88d592d36051d82"}, + {file = "pyproj-3.5.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:7572983134e310e0ca809c63f1722557a040fe9443df5f247bf11ba887eb1229"}, + {file = "pyproj-3.5.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:eccb417b91d0be27805dfc97550bfb8b7db94e9fe1db5ebedb98f5b88d601323"}, + {file = "pyproj-3.5.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:621d78a9d8bf4d06e08bef2471021fbcb1a65aa629ad4a20c22e521ce729cc20"}, + {file = "pyproj-3.5.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d9a024370e917c899bff9171f03ea6079deecdc7482a146a2c565f3b9df134ea"}, + {file = "pyproj-3.5.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1b7c2113c4d11184a238077ec85e31eda1dcc58ffeb9a4429830e0a7036e787d"}, + {file = "pyproj-3.5.0-cp311-cp311-win32.whl", hash = "sha256:a730f5b4c98c8a0f312437873e6e34dbd4cc6dc23d5afd91a6691c62724b1f68"}, + {file = "pyproj-3.5.0-cp311-cp311-win_amd64.whl", hash = "sha256:e97573de0ab3bbbcb4c7748bc41f4ceb6da10b45d35b1a294b5820701e7c25f0"}, + {file = "pyproj-3.5.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2b708fd43453b985642b737d4a6e7f1d6a0ab1677ffa4e14cc258537b49224b0"}, + {file = "pyproj-3.5.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b60d93a200639e8367c6542a964fd0aa2dbd152f256c1831dc18cd5aa470fb8a"}, + {file = "pyproj-3.5.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:38862fe07316ae12b79d82d298e390973a4f00b684f3c2d037238e20e00610ba"}, + {file = "pyproj-3.5.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:71b65f2a38cd9e16883dbb0f8ae82bdf8f6b79b1b02975c78483ab8428dbbf2f"}, + {file = "pyproj-3.5.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b752b7d9c4b08181c7e8c0d9c7f277cbefff42227f34d3310696a87c863d9dd3"}, + {file = "pyproj-3.5.0-cp38-cp38-win32.whl", hash = "sha256:b937215bfbaf404ec8f03ca741fc3f9f2c4c2c5590a02ccddddd820ae3c71331"}, + {file = "pyproj-3.5.0-cp38-cp38-win_amd64.whl", hash = "sha256:97ed199033c2c770e7eea2ef80ff5e6413426ec2d7ec985b869792f04ab95d05"}, + {file = "pyproj-3.5.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:052c49fce8b5d55943a35c36ccecb87350c68b48ba95bc02a789770c374ef819"}, + {file = "pyproj-3.5.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1507138ea28bf2134d31797675380791cc1a7156a3aeda484e65a78a4aba9b62"}, + {file = "pyproj-3.5.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c02742ef3d846401861a878a61ef7ad911ea7539d6cc4619ddb52dbdf7b45aee"}, + {file = "pyproj-3.5.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:385b0341861d3ebc8cad98337a738821dcb548d465576527399f4955ca24b6ed"}, + {file = "pyproj-3.5.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8fe6bb1b68a35d07378d38be77b5b2f8dd2bea5910c957bfcc7bee55988d3910"}, + {file = "pyproj-3.5.0-cp39-cp39-win32.whl", hash = "sha256:5c4b85ac10d733c42d73a2e6261c8d6745bf52433a31848dd1b6561c9a382da3"}, + {file = "pyproj-3.5.0-cp39-cp39-win_amd64.whl", hash = "sha256:1798ff7d65d9057ebb2d017ffe8403268b8452f24d0428b2140018c25c7fa1bc"}, + {file = "pyproj-3.5.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:d711517a8487ef3245b08dc82f781a906df9abb3b6cb0ce0486f0eeb823ca570"}, + {file = "pyproj-3.5.0-pp38-pypy38_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:788a5dadb532644a64efe0f5f01bf508c821eb7e984f13a677d56002f1e8a67a"}, + {file = "pyproj-3.5.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:73f7960a97225812f9b1d7aeda5fb83812f38de9441e3476fcc8abb3e2b2f4de"}, + {file = "pyproj-3.5.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:fde5ece4d2436b5a57c8f5f97b49b5de06a856d03959f836c957d3e609f2de7e"}, + {file = "pyproj-3.5.0-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e08db25b61cf024648d55973cc3d1c3f1d0818fabf594d5f5a8e2318103d2aa0"}, + {file = "pyproj-3.5.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6a87b419a2a352413fbf759ecb66da9da50bd19861c8f26db6a25439125b27b9"}, + {file = "pyproj-3.5.0.tar.gz", hash = "sha256:9859d1591c1863414d875ae0759e72c2cffc01ab989dc64137fbac572cc81bf6"}, +] + +[package.dependencies] +certifi = "*" + +[[package]] +name = "pytest" +version = "7.4.3" +description = "pytest: simple powerful testing with Python" +optional = false +python-versions = ">=3.7" +files = [ + {file = "pytest-7.4.3-py3-none-any.whl", hash = "sha256:0d009c083ea859a71b76adf7c1d502e4bc170b80a8ef002da5806527b9591fac"}, + {file = "pytest-7.4.3.tar.gz", hash = "sha256:d989d136982de4e3b29dabcc838ad581c64e8ed52c11fbe86ddebd9da0818cd5"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "sys_platform == \"win32\""} +exceptiongroup = {version = ">=1.0.0rc8", markers = "python_version < \"3.11\""} +iniconfig = "*" +packaging = "*" +pluggy = ">=0.12,<2.0" +tomli = {version = ">=1.0.0", markers = "python_version < \"3.11\""} + +[package.extras] +testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "nose", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"] + +[[package]] +name = "pytest-asyncio" +version = "0.19.0" +description = "Pytest support for asyncio" +optional = false +python-versions = ">=3.7" +files = [ + {file = "pytest-asyncio-0.19.0.tar.gz", hash = "sha256:ac4ebf3b6207259750bc32f4c1d8fcd7e79739edbc67ad0c58dd150b1d072fed"}, + {file = "pytest_asyncio-0.19.0-py3-none-any.whl", hash = "sha256:7a97e37cfe1ed296e2e84941384bdd37c376453912d397ed39293e0916f521fa"}, +] + +[package.dependencies] +pytest = ">=6.1.0" + +[package.extras] +testing = ["coverage (>=6.2)", "flaky (>=3.5.0)", "hypothesis (>=5.7.1)", "mypy (>=0.931)", "pytest-trio (>=0.7.0)"] + +[[package]] +name = "pytest-cov" +version = "4.1.0" +description = "Pytest plugin for measuring coverage." +optional = false +python-versions = ">=3.7" +files = [ + {file = "pytest-cov-4.1.0.tar.gz", hash = "sha256:3904b13dfbfec47f003b8e77fd5b589cd11904a21ddf1ab38a64f204d6a10ef6"}, + {file = "pytest_cov-4.1.0-py3-none-any.whl", hash = "sha256:6ba70b9e97e69fcc3fb45bfeab2d0a138fb65c4d0d6a41ef33983ad114be8c3a"}, +] + +[package.dependencies] +coverage = {version = ">=5.2.1", extras = ["toml"]} +pytest = ">=4.6" + +[package.extras] +testing = ["fields", "hunter", "process-tests", "pytest-xdist", "six", "virtualenv"] + +[[package]] +name = "pytest-mock" +version = "3.12.0" +description = "Thin-wrapper around the mock package for easier use with pytest" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pytest-mock-3.12.0.tar.gz", hash = "sha256:31a40f038c22cad32287bb43932054451ff5583ff094bca6f675df2f8bc1a6e9"}, + {file = "pytest_mock-3.12.0-py3-none-any.whl", hash = "sha256:0972719a7263072da3a21c7f4773069bcc7486027d7e8e1f81d98a47e701bc4f"}, +] + +[package.dependencies] +pytest = ">=5.0" + +[package.extras] +dev = ["pre-commit", "pytest-asyncio", "tox"] + +[[package]] +name = "python-dateutil" +version = "2.8.2" +description = "Extensions to the standard Python datetime module" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" +files = [ + {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"}, + {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"}, +] + +[package.dependencies] +six = ">=1.5" + +[[package]] +name = "python-dotenv" +version = "0.21.1" +description = "Read key-value pairs from a .env file and set them as environment variables" +optional = false +python-versions = ">=3.7" +files = [ + {file = "python-dotenv-0.21.1.tar.gz", hash = "sha256:1c93de8f636cde3ce377292818d0e440b6e45a82f215c3744979151fa8151c49"}, + {file = "python_dotenv-0.21.1-py3-none-any.whl", hash = "sha256:41e12e0318bebc859fcc4d97d4db8d20ad21721a6aa5047dd59f090391cb549a"}, +] + +[package.extras] +cli = ["click (>=5.0)"] + +[[package]] +name = "python-multipart" +version = "0.0.5" +description = "A streaming multipart parser for Python" +optional = false +python-versions = "*" +files = [ + {file = "python-multipart-0.0.5.tar.gz", hash = "sha256:f7bb5f611fc600d15fa47b3974c8aa16e93724513b49b5f95c81e6624c83fa43"}, +] + +[package.dependencies] +six = ">=1.4.0" + +[[package]] +name = "python-slugify" +version = "8.0.1" +description = "A Python slugify application that also handles Unicode" +optional = false +python-versions = ">=3.7" +files = [ + {file = "python-slugify-8.0.1.tar.gz", hash = "sha256:ce0d46ddb668b3be82f4ed5e503dbc33dd815d83e2eb6824211310d3fb172a27"}, + {file = "python_slugify-8.0.1-py2.py3-none-any.whl", hash = "sha256:70ca6ea68fe63ecc8fa4fcf00ae651fc8a5d02d93dcd12ae6d4fc7ca46c4d395"}, +] + +[package.dependencies] +text-unidecode = ">=1.3" + +[package.extras] +unidecode = ["Unidecode (>=1.1.1)"] + +[[package]] +name = "pytz" +version = "2022.7.1" +description = "World timezone definitions, modern and historical" +optional = false +python-versions = "*" +files = [ + {file = "pytz-2022.7.1-py2.py3-none-any.whl", hash = "sha256:78f4f37d8198e0627c5f1143240bb0206b8691d8d7ac6d78fee88b78733f8c4a"}, + {file = "pytz-2022.7.1.tar.gz", hash = "sha256:01a0681c4b9684a28304615eba55d1ab31ae00bf68ec157ec3708a8182dbbcd0"}, +] + +[[package]] +name = "pyyaml" +version = "6.0.1" +description = "YAML parser and emitter for Python" +optional = false +python-versions = ">=3.6" +files = [ + {file = "PyYAML-6.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d858aa552c999bc8a8d57426ed01e40bef403cd8ccdd0fc5f6f04a00414cac2a"}, + {file = "PyYAML-6.0.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:fd66fc5d0da6d9815ba2cebeb4205f95818ff4b79c3ebe268e75d961704af52f"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"}, + {file = "PyYAML-6.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:326c013efe8048858a6d312ddd31d56e468118ad4cdeda36c719bf5bb6192290"}, + {file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"}, + {file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f003ed9ad21d6a4713f0a9b5a7a0a79e08dd0f221aff4525a2be4c346ee60aab"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"}, + {file = "PyYAML-6.0.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e7d73685e87afe9f3b36c799222440d6cf362062f78be1013661b00c5c6f678b"}, + {file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"}, + {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"}, + {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"}, + {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"}, + {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"}, + {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"}, + {file = "PyYAML-6.0.1-cp312-cp312-win_amd64.whl", hash = "sha256:0d3304d8c0adc42be59c5f8a4d9e3d7379e6955ad754aa9d6ab7a398b59dd1df"}, + {file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:afd7e57eddb1a54f0f1a974bc4391af8bcce0b444685d936840f125cf046d5bd"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win32.whl", hash = "sha256:fca0e3a251908a499833aa292323f32437106001d436eca0e6e7833256674585"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:f22ac1c3cac4dbc50079e965eba2c1058622631e526bd9afd45fedd49ba781fa"}, + {file = "PyYAML-6.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b1275ad35a5d18c62a7220633c913e1b42d44b46ee12554e5fd39c70a243d6a3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:18aeb1bf9a78867dc38b259769503436b7c72f7a1f1f4c93ff9a17de54319b27"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:596106435fa6ad000c2991a98fa58eeb8656ef2325d7e158344fb33864ed87e3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:baa90d3f661d43131ca170712d903e6295d1f7a0f595074f151c0aed377c9b9c"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win32.whl", hash = "sha256:9046c58c4395dff28dd494285c82ba00b546adfc7ef001486fbf0324bc174fba"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:4fb147e7a67ef577a588a0e2c17b6db51dda102c71de36f8549b6816a96e1867"}, + {file = "PyYAML-6.0.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1d4c7e777c441b20e32f52bd377e0c409713e8bb1386e1099c2415f26e479595"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"}, + {file = "PyYAML-6.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:49a183be227561de579b4a36efbb21b3eab9651dd81b1858589f796549873dd6"}, + {file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"}, + {file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c8098ddcc2a85b61647b2590f825f3db38891662cfc2fc776415143f599bb859"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"}, + {file = "PyYAML-6.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:04ac92ad1925b2cff1db0cfebffb6ffc43457495c9b3c39d3fcae417d7125dc5"}, + {file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"}, + {file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"}, + {file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"}, +] + +[[package]] +name = "rdflib" +version = "6.3.2" +description = "RDFLib is a Python library for working with RDF, a simple yet powerful language for representing information." +optional = false +python-versions = ">=3.7,<4.0" +files = [ + {file = "rdflib-6.3.2-py3-none-any.whl", hash = "sha256:36b4e74a32aa1e4fa7b8719876fb192f19ecd45ff932ea5ebbd2e417a0247e63"}, + {file = "rdflib-6.3.2.tar.gz", hash = "sha256:72af591ff704f4caacea7ecc0c5a9056b8553e0489dd4f35a9bc52dbd41522e0"}, +] + +[package.dependencies] +isodate = ">=0.6.0,<0.7.0" +pyparsing = ">=2.1.0,<4" + +[package.extras] +berkeleydb = ["berkeleydb (>=18.1.0,<19.0.0)"] +html = ["html5lib (>=1.0,<2.0)"] +lxml = ["lxml (>=4.3.0,<5.0.0)"] +networkx = ["networkx (>=2.0.0,<3.0.0)"] + +[[package]] +name = "referencing" +version = "0.32.0" +description = "JSON Referencing + Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "referencing-0.32.0-py3-none-any.whl", hash = "sha256:bdcd3efb936f82ff86f993093f6da7435c7de69a3b3a5a06678a6050184bee99"}, + {file = "referencing-0.32.0.tar.gz", hash = "sha256:689e64fe121843dcfd57b71933318ef1f91188ffb45367332700a86ac8fd6161"}, +] + +[package.dependencies] +attrs = ">=22.2.0" +rpds-py = ">=0.7.0" + +[[package]] +name = "requests" +version = "2.31.0" +description = "Python HTTP for Humans." +optional = false +python-versions = ">=3.7" +files = [ + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, +] + +[package.dependencies] +certifi = ">=2017.4.17" +charset-normalizer = ">=2,<4" +idna = ">=2.5,<4" +urllib3 = ">=1.21.1,<3" + +[package.extras] +socks = ["PySocks (>=1.5.6,!=1.5.7)"] +use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] + +[[package]] +name = "responses" +version = "0.21.0" +description = "A utility library for mocking out the `requests` Python library." +optional = false +python-versions = ">=3.7" +files = [ + {file = "responses-0.21.0-py3-none-any.whl", hash = "sha256:2dcc863ba63963c0c3d9ee3fa9507cbe36b7d7b0fccb4f0bdfd9e96c539b1487"}, + {file = "responses-0.21.0.tar.gz", hash = "sha256:b82502eb5f09a0289d8e209e7bad71ef3978334f56d09b444253d5ad67bf5253"}, +] + +[package.dependencies] +requests = ">=2.0,<3.0" +urllib3 = ">=1.25.10" + +[package.extras] +tests = ["coverage (>=6.0.0)", "flake8", "mypy", "pytest (>=7.0.0)", "pytest-asyncio", "pytest-cov", "pytest-localserver", "types-mock", "types-requests"] + +[[package]] +name = "rfc3986" +version = "1.5.0" +description = "Validating URI References per RFC 3986" +optional = false +python-versions = "*" +files = [ + {file = "rfc3986-1.5.0-py2.py3-none-any.whl", hash = "sha256:a86d6e1f5b1dc238b218b012df0aa79409667bb209e58da56d0b94704e712a97"}, + {file = "rfc3986-1.5.0.tar.gz", hash = "sha256:270aaf10d87d0d4e095063c65bf3ddbc6ee3d0b226328ce21e036f946e421835"}, +] + +[package.dependencies] +idna = {version = "*", optional = true, markers = "extra == \"idna2008\""} + +[package.extras] +idna2008 = ["idna"] + +[[package]] +name = "rich" +version = "12.6.0" +description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" +optional = false +python-versions = ">=3.6.3,<4.0.0" +files = [ + {file = "rich-12.6.0-py3-none-any.whl", hash = "sha256:a4eb26484f2c82589bd9a17c73d32a010b1e29d89f1604cd9bf3a2097b81bb5e"}, + {file = "rich-12.6.0.tar.gz", hash = "sha256:ba3a3775974105c221d31141f2c116f4fd65c5ceb0698657a11e9f295ec93fd0"}, +] + +[package.dependencies] +commonmark = ">=0.9.0,<0.10.0" +pygments = ">=2.6.0,<3.0.0" +typing-extensions = {version = ">=4.0.0,<5.0", markers = "python_version < \"3.9\""} + +[package.extras] +jupyter = ["ipywidgets (>=7.5.1,<8.0.0)"] + +[[package]] +name = "rpds-py" +version = "0.13.2" +description = "Python bindings to Rust's persistent data structures (rpds)" +optional = false +python-versions = ">=3.8" +files = [ + {file = "rpds_py-0.13.2-cp310-cp310-macosx_10_7_x86_64.whl", hash = "sha256:1ceebd0ae4f3e9b2b6b553b51971921853ae4eebf3f54086be0565d59291e53d"}, + {file = "rpds_py-0.13.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:46e1ed994a0920f350a4547a38471217eb86f57377e9314fbaaa329b71b7dfe3"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ee353bb51f648924926ed05e0122b6a0b1ae709396a80eb583449d5d477fcdf7"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:530190eb0cd778363bbb7596612ded0bb9fef662daa98e9d92a0419ab27ae914"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:29d311e44dd16d2434d5506d57ef4d7036544fc3c25c14b6992ef41f541b10fb"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2e72f750048b32d39e87fc85c225c50b2a6715034848dbb196bf3348aa761fa1"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db09b98c7540df69d4b47218da3fbd7cb466db0fb932e971c321f1c76f155266"}, + {file = "rpds_py-0.13.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2ac26f50736324beb0282c819668328d53fc38543fa61eeea2c32ea8ea6eab8d"}, + {file = "rpds_py-0.13.2-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:12ecf89bd54734c3c2c79898ae2021dca42750c7bcfb67f8fb3315453738ac8f"}, + {file = "rpds_py-0.13.2-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:3a44c8440183b43167fd1a0819e8356692bf5db1ad14ce140dbd40a1485f2dea"}, + {file = "rpds_py-0.13.2-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:bcef4f2d3dc603150421de85c916da19471f24d838c3c62a4f04c1eb511642c1"}, + {file = "rpds_py-0.13.2-cp310-none-win32.whl", hash = "sha256:ee6faebb265e28920a6f23a7d4c362414b3f4bb30607141d718b991669e49ddc"}, + {file = "rpds_py-0.13.2-cp310-none-win_amd64.whl", hash = "sha256:ac96d67b37f28e4b6ecf507c3405f52a40658c0a806dffde624a8fcb0314d5fd"}, + {file = "rpds_py-0.13.2-cp311-cp311-macosx_10_7_x86_64.whl", hash = "sha256:b5f6328e8e2ae8238fc767703ab7b95785521c42bb2b8790984e3477d7fa71ad"}, + {file = "rpds_py-0.13.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:729408136ef8d45a28ee9a7411917c9e3459cf266c7e23c2f7d4bb8ef9e0da42"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:65cfed9c807c27dee76407e8bb29e6f4e391e436774bcc769a037ff25ad8646e"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:aefbdc934115d2f9278f153952003ac52cd2650e7313750390b334518c589568"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d48db29bd47814671afdd76c7652aefacc25cf96aad6daefa82d738ee87461e2"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3c55d7f2d817183d43220738270efd3ce4e7a7b7cbdaefa6d551ed3d6ed89190"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6aadae3042f8e6db3376d9e91f194c606c9a45273c170621d46128f35aef7cd0"}, + {file = "rpds_py-0.13.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:5feae2f9aa7270e2c071f488fab256d768e88e01b958f123a690f1cc3061a09c"}, + {file = "rpds_py-0.13.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:51967a67ea0d7b9b5cd86036878e2d82c0b6183616961c26d825b8c994d4f2c8"}, + {file = "rpds_py-0.13.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:4d0c10d803549427f427085ed7aebc39832f6e818a011dcd8785e9c6a1ba9b3e"}, + {file = "rpds_py-0.13.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:603d5868f7419081d616dab7ac3cfa285296735e7350f7b1e4f548f6f953ee7d"}, + {file = "rpds_py-0.13.2-cp311-none-win32.whl", hash = "sha256:b8996ffb60c69f677245f5abdbcc623e9442bcc91ed81b6cd6187129ad1fa3e7"}, + {file = "rpds_py-0.13.2-cp311-none-win_amd64.whl", hash = "sha256:5379e49d7e80dca9811b36894493d1c1ecb4c57de05c36f5d0dd09982af20211"}, + {file = "rpds_py-0.13.2-cp312-cp312-macosx_10_7_x86_64.whl", hash = "sha256:8a776a29b77fe0cc28fedfd87277b0d0f7aa930174b7e504d764e0b43a05f381"}, + {file = "rpds_py-0.13.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2a1472956c5bcc49fb0252b965239bffe801acc9394f8b7c1014ae9258e4572b"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f252dfb4852a527987a9156cbcae3022a30f86c9d26f4f17b8c967d7580d65d2"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f0d320e70b6b2300ff6029e234e79fe44e9dbbfc7b98597ba28e054bd6606a57"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ade2ccb937060c299ab0dfb2dea3d2ddf7e098ed63ee3d651ebfc2c8d1e8632a"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b9d121be0217787a7d59a5c6195b0842d3f701007333426e5154bf72346aa658"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8fa6bd071ec6d90f6e7baa66ae25820d57a8ab1b0a3c6d3edf1834d4b26fafa2"}, + {file = "rpds_py-0.13.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c918621ee0a3d1fe61c313f2489464f2ae3d13633e60f520a8002a5e910982ee"}, + {file = "rpds_py-0.13.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:25b28b3d33ec0a78e944aaaed7e5e2a94ac811bcd68b557ca48a0c30f87497d2"}, + {file = "rpds_py-0.13.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:31e220a040b89a01505128c2f8a59ee74732f666439a03e65ccbf3824cdddae7"}, + {file = "rpds_py-0.13.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:15253fff410873ebf3cfba1cc686a37711efcd9b8cb30ea21bb14a973e393f60"}, + {file = "rpds_py-0.13.2-cp312-none-win32.whl", hash = "sha256:b981a370f8f41c4024c170b42fbe9e691ae2dbc19d1d99151a69e2c84a0d194d"}, + {file = "rpds_py-0.13.2-cp312-none-win_amd64.whl", hash = "sha256:4c4e314d36d4f31236a545696a480aa04ea170a0b021e9a59ab1ed94d4c3ef27"}, + {file = "rpds_py-0.13.2-cp38-cp38-macosx_10_7_x86_64.whl", hash = "sha256:80e5acb81cb49fd9f2d5c08f8b74ffff14ee73b10ca88297ab4619e946bcb1e1"}, + {file = "rpds_py-0.13.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:efe093acc43e869348f6f2224df7f452eab63a2c60a6c6cd6b50fd35c4e075ba"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8c2a61c0e4811012b0ba9f6cdcb4437865df5d29eab5d6018ba13cee1c3064a0"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:751758d9dd04d548ec679224cc00e3591f5ebf1ff159ed0d4aba6a0746352452"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6ba8858933f0c1a979781272a5f65646fca8c18c93c99c6ddb5513ad96fa54b1"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bfdfbe6a36bc3059fff845d64c42f2644cf875c65f5005db54f90cdfdf1df815"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aa0379c1935c44053c98826bc99ac95f3a5355675a297ac9ce0dfad0ce2d50ca"}, + {file = "rpds_py-0.13.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:d5593855b5b2b73dd8413c3fdfa5d95b99d657658f947ba2c4318591e745d083"}, + {file = "rpds_py-0.13.2-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:2a7bef6977043673750a88da064fd513f89505111014b4e00fbdd13329cd4e9a"}, + {file = "rpds_py-0.13.2-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:3ab96754d23372009638a402a1ed12a27711598dd49d8316a22597141962fe66"}, + {file = "rpds_py-0.13.2-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:e06cfea0ece444571d24c18ed465bc93afb8c8d8d74422eb7026662f3d3f779b"}, + {file = "rpds_py-0.13.2-cp38-none-win32.whl", hash = "sha256:5493569f861fb7b05af6d048d00d773c6162415ae521b7010197c98810a14cab"}, + {file = "rpds_py-0.13.2-cp38-none-win_amd64.whl", hash = "sha256:b07501b720cf060c5856f7b5626e75b8e353b5f98b9b354a21eb4bfa47e421b1"}, + {file = "rpds_py-0.13.2-cp39-cp39-macosx_10_7_x86_64.whl", hash = "sha256:881df98f0a8404d32b6de0fd33e91c1b90ed1516a80d4d6dc69d414b8850474c"}, + {file = "rpds_py-0.13.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:d79c159adea0f1f4617f54aa156568ac69968f9ef4d1e5fefffc0a180830308e"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:38d4f822ee2f338febcc85aaa2547eb5ba31ba6ff68d10b8ec988929d23bb6b4"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5d75d6d220d55cdced2f32cc22f599475dbe881229aeddba6c79c2e9df35a2b3"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5d97e9ae94fb96df1ee3cb09ca376c34e8a122f36927230f4c8a97f469994bff"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:67a429520e97621a763cf9b3ba27574779c4e96e49a27ff8a1aa99ee70beb28a"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:188435794405c7f0573311747c85a96b63c954a5f2111b1df8018979eca0f2f0"}, + {file = "rpds_py-0.13.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:38f9bf2ad754b4a45b8210a6c732fe876b8a14e14d5992a8c4b7c1ef78740f53"}, + {file = "rpds_py-0.13.2-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:a6ba2cb7d676e9415b9e9ac7e2aae401dc1b1e666943d1f7bc66223d3d73467b"}, + {file = "rpds_py-0.13.2-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:eaffbd8814bb1b5dc3ea156a4c5928081ba50419f9175f4fc95269e040eff8f0"}, + {file = "rpds_py-0.13.2-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:5a4c1058cdae6237d97af272b326e5f78ee7ee3bbffa6b24b09db4d828810468"}, + {file = "rpds_py-0.13.2-cp39-none-win32.whl", hash = "sha256:b5267feb19070bef34b8dea27e2b504ebd9d31748e3ecacb3a4101da6fcb255c"}, + {file = "rpds_py-0.13.2-cp39-none-win_amd64.whl", hash = "sha256:ddf23960cb42b69bce13045d5bc66f18c7d53774c66c13f24cf1b9c144ba3141"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-macosx_10_7_x86_64.whl", hash = "sha256:97163a1ab265a1073a6372eca9f4eeb9f8c6327457a0b22ddfc4a17dcd613e74"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:25ea41635d22b2eb6326f58e608550e55d01df51b8a580ea7e75396bafbb28e9"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:76d59d4d451ba77f08cb4cd9268dec07be5bc65f73666302dbb5061989b17198"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:e7c564c58cf8f248fe859a4f0fe501b050663f3d7fbc342172f259124fb59933"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:61dbc1e01dc0c5875da2f7ae36d6e918dc1b8d2ce04e871793976594aad8a57a"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fdb82eb60d31b0c033a8e8ee9f3fc7dfbaa042211131c29da29aea8531b4f18f"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d204957169f0b3511fb95395a9da7d4490fb361763a9f8b32b345a7fe119cb45"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:c45008ca79bad237cbc03c72bc5205e8c6f66403773929b1b50f7d84ef9e4d07"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:79bf58c08f0756adba691d480b5a20e4ad23f33e1ae121584cf3a21717c36dfa"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-musllinux_1_2_i686.whl", hash = "sha256:e86593bf8637659e6a6ed58854b6c87ec4e9e45ee8a4adfd936831cef55c2d21"}, + {file = "rpds_py-0.13.2-pp310-pypy310_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:d329896c40d9e1e5c7715c98529e4a188a1f2df51212fd65102b32465612b5dc"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-macosx_10_7_x86_64.whl", hash = "sha256:4a5375c5fff13f209527cd886dc75394f040c7d1ecad0a2cb0627f13ebe78a12"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-macosx_11_0_arm64.whl", hash = "sha256:06d218e4464d31301e943b65b2c6919318ea6f69703a351961e1baaf60347276"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c1f41d32a2ddc5a94df4b829b395916a4b7f103350fa76ba6de625fcb9e773ac"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:6bc568b05e02cd612be53900c88aaa55012e744930ba2eeb56279db4c6676eb3"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9d94d78418203904730585efa71002286ac4c8ac0689d0eb61e3c465f9e608ff"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:bed0252c85e21cf73d2d033643c945b460d6a02fc4a7d644e3b2d6f5f2956c64"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:244e173bb6d8f3b2f0c4d7370a1aa341f35da3e57ffd1798e5b2917b91731fd3"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:7f55cd9cf1564b7b03f238e4c017ca4794c05b01a783e9291065cb2858d86ce4"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:f03a1b3a4c03e3e0161642ac5367f08479ab29972ea0ffcd4fa18f729cd2be0a"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-musllinux_1_2_i686.whl", hash = "sha256:f5f4424cb87a20b016bfdc157ff48757b89d2cc426256961643d443c6c277007"}, + {file = "rpds_py-0.13.2-pp38-pypy38_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:c82bbf7e03748417c3a88c1b0b291288ce3e4887a795a3addaa7a1cfd9e7153e"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-macosx_10_7_x86_64.whl", hash = "sha256:c0095b8aa3e432e32d372e9a7737e65b58d5ed23b9620fea7cb81f17672f1fa1"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:4c2d26aa03d877c9730bf005621c92da263523a1e99247590abbbe252ccb7824"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:96f2975fb14f39c5fe75203f33dd3010fe37d1c4e33177feef1107b5ced750e3"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:4dcc5ee1d0275cb78d443fdebd0241e58772a354a6d518b1d7af1580bbd2c4e8"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:61d42d2b08430854485135504f672c14d4fc644dd243a9c17e7c4e0faf5ed07e"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d3a61e928feddc458a55110f42f626a2a20bea942ccedb6fb4cee70b4830ed41"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7de12b69d95072394998c622cfd7e8cea8f560db5fca6a62a148f902a1029f8b"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:87a90f5545fd61f6964e65eebde4dc3fa8660bb7d87adb01d4cf17e0a2b484ad"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-musllinux_1_2_aarch64.whl", hash = "sha256:9c95a1a290f9acf7a8f2ebbdd183e99215d491beea52d61aa2a7a7d2c618ddc6"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-musllinux_1_2_i686.whl", hash = "sha256:35f53c76a712e323c779ca39b9a81b13f219a8e3bc15f106ed1e1462d56fcfe9"}, + {file = "rpds_py-0.13.2-pp39-pypy39_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:96fb0899bb2ab353f42e5374c8f0789f54e0a94ef2f02b9ac7149c56622eaf31"}, + {file = "rpds_py-0.13.2.tar.gz", hash = "sha256:f8eae66a1304de7368932b42d801c67969fd090ddb1a7a24f27b435ed4bed68f"}, +] + +[[package]] +name = "ruamel-yaml" +version = "0.17.40" +description = "ruamel.yaml is a YAML parser/emitter that supports roundtrip preservation of comments, seq/map flow style, and map key order" +optional = false +python-versions = ">=3" +files = [ + {file = "ruamel.yaml-0.17.40-py3-none-any.whl", hash = "sha256:b16b6c3816dff0a93dca12acf5e70afd089fa5acb80604afd1ffa8b465b7722c"}, + {file = "ruamel.yaml-0.17.40.tar.gz", hash = "sha256:6024b986f06765d482b5b07e086cc4b4cd05dd22ddcbc758fa23d54873cf313d"}, +] + +[package.dependencies] +"ruamel.yaml.clib" = {version = ">=0.2.7", markers = "platform_python_implementation == \"CPython\" and python_version < \"3.13\""} + +[package.extras] +docs = ["mercurial (>5.7)", "ryd"] +jinja2 = ["ruamel.yaml.jinja2 (>=0.2)"] + +[[package]] +name = "ruamel-yaml-clib" +version = "0.2.8" +description = "C version of reader, parser and emitter for ruamel.yaml derived from libyaml" +optional = false +python-versions = ">=3.6" +files = [ + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:b42169467c42b692c19cf539c38d4602069d8c1505e97b86387fcf7afb766e1d"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-macosx_13_0_arm64.whl", hash = "sha256:07238db9cbdf8fc1e9de2489a4f68474e70dffcb32232db7c08fa61ca0c7c462"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:fff3573c2db359f091e1589c3d7c5fc2f86f5bdb6f24252c2d8e539d4e45f412"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-manylinux_2_24_aarch64.whl", hash = "sha256:aa2267c6a303eb483de8d02db2871afb5c5fc15618d894300b88958f729ad74f"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:840f0c7f194986a63d2c2465ca63af8ccbbc90ab1c6001b1978f05119b5e7334"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:024cfe1fc7c7f4e1aff4a81e718109e13409767e4f871443cbff3dba3578203d"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-win32.whl", hash = "sha256:c69212f63169ec1cfc9bb44723bf2917cbbd8f6191a00ef3410f5a7fe300722d"}, + {file = "ruamel.yaml.clib-0.2.8-cp310-cp310-win_amd64.whl", hash = "sha256:cabddb8d8ead485e255fe80429f833172b4cadf99274db39abc080e068cbcc31"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:bef08cd86169d9eafb3ccb0a39edb11d8e25f3dae2b28f5c52fd997521133069"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-macosx_13_0_arm64.whl", hash = "sha256:b16420e621d26fdfa949a8b4b47ade8810c56002f5389970db4ddda51dbff248"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_24_x86_64.whl", hash = "sha256:25c515e350e5b739842fc3228d662413ef28f295791af5e5110b543cf0b57d9b"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-manylinux_2_24_aarch64.whl", hash = "sha256:1707814f0d9791df063f8c19bb51b0d1278b8e9a2353abbb676c2f685dee6afe"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:46d378daaac94f454b3a0e3d8d78cafd78a026b1d71443f4966c696b48a6d899"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:09b055c05697b38ecacb7ac50bdab2240bfca1a0c4872b0fd309bb07dc9aa3a9"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-win32.whl", hash = "sha256:53a300ed9cea38cf5a2a9b069058137c2ca1ce658a874b79baceb8f892f915a7"}, + {file = "ruamel.yaml.clib-0.2.8-cp311-cp311-win_amd64.whl", hash = "sha256:c2a72e9109ea74e511e29032f3b670835f8a59bbdc9ce692c5b4ed91ccf1eedb"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:ebc06178e8821efc9692ea7544aa5644217358490145629914d8020042c24aa1"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-macosx_13_0_arm64.whl", hash = "sha256:edaef1c1200c4b4cb914583150dcaa3bc30e592e907c01117c08b13a07255ec2"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d176b57452ab5b7028ac47e7b3cf644bcfdc8cacfecf7e71759f7f51a59e5c92"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-manylinux_2_24_aarch64.whl", hash = "sha256:1dc67314e7e1086c9fdf2680b7b6c2be1c0d8e3a8279f2e993ca2a7545fecf62"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:3213ece08ea033eb159ac52ae052a4899b56ecc124bb80020d9bbceeb50258e9"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:aab7fd643f71d7946f2ee58cc88c9b7bfc97debd71dcc93e03e2d174628e7e2d"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-win32.whl", hash = "sha256:5c365d91c88390c8d0a8545df0b5857172824b1c604e867161e6b3d59a827eaa"}, + {file = "ruamel.yaml.clib-0.2.8-cp312-cp312-win_amd64.whl", hash = "sha256:1758ce7d8e1a29d23de54a16ae867abd370f01b5a69e1a3ba75223eaa3ca1a1b"}, + {file = "ruamel.yaml.clib-0.2.8-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:a5aa27bad2bb83670b71683aae140a1f52b0857a2deff56ad3f6c13a017a26ed"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c58ecd827313af6864893e7af0a3bb85fd529f862b6adbefe14643947cfe2942"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-macosx_12_0_arm64.whl", hash = "sha256:f481f16baec5290e45aebdc2a5168ebc6d35189ae6fea7a58787613a25f6e875"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-manylinux_2_24_aarch64.whl", hash = "sha256:77159f5d5b5c14f7c34073862a6b7d34944075d9f93e681638f6d753606c6ce6"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:7f67a1ee819dc4562d444bbafb135832b0b909f81cc90f7aa00260968c9ca1b3"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:4ecbf9c3e19f9562c7fdd462e8d18dd902a47ca046a2e64dba80699f0b6c09b7"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:87ea5ff66d8064301a154b3933ae406b0863402a799b16e4a1d24d9fbbcbe0d3"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-win32.whl", hash = "sha256:75e1ed13e1f9de23c5607fe6bd1aeaae21e523b32d83bb33918245361e9cc51b"}, + {file = "ruamel.yaml.clib-0.2.8-cp37-cp37m-win_amd64.whl", hash = "sha256:3f215c5daf6a9d7bbed4a0a4f760f3113b10e82ff4c5c44bec20a68c8014f675"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1b617618914cb00bf5c34d4357c37aa15183fa229b24767259657746c9077615"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-macosx_12_0_arm64.whl", hash = "sha256:a6a9ffd280b71ad062eae53ac1659ad86a17f59a0fdc7699fd9be40525153337"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-manylinux_2_24_aarch64.whl", hash = "sha256:305889baa4043a09e5b76f8e2a51d4ffba44259f6b4c72dec8ca56207d9c6fe1"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:700e4ebb569e59e16a976857c8798aee258dceac7c7d6b50cab63e080058df91"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:e2b4c44b60eadec492926a7270abb100ef9f72798e18743939bdbf037aab8c28"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:e79e5db08739731b0ce4850bed599235d601701d5694c36570a99a0c5ca41a9d"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-win32.whl", hash = "sha256:955eae71ac26c1ab35924203fda6220f84dce57d6d7884f189743e2abe3a9fbe"}, + {file = "ruamel.yaml.clib-0.2.8-cp38-cp38-win_amd64.whl", hash = "sha256:56f4252222c067b4ce51ae12cbac231bce32aee1d33fbfc9d17e5b8d6966c312"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:03d1162b6d1df1caa3a4bd27aa51ce17c9afc2046c31b0ad60a0a96ec22f8001"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:bba64af9fa9cebe325a62fa398760f5c7206b215201b0ec825005f1b18b9bccf"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-manylinux_2_24_aarch64.whl", hash = "sha256:a1a45e0bb052edf6a1d3a93baef85319733a888363938e1fc9924cb00c8df24c"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:da09ad1c359a728e112d60116f626cc9f29730ff3e0e7db72b9a2dbc2e4beed5"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:184565012b60405d93838167f425713180b949e9d8dd0bbc7b49f074407c5a8b"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:a75879bacf2c987c003368cf14bed0ffe99e8e85acfa6c0bfffc21a090f16880"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-win32.whl", hash = "sha256:84b554931e932c46f94ab306913ad7e11bba988104c5cff26d90d03f68258cd5"}, + {file = "ruamel.yaml.clib-0.2.8-cp39-cp39-win_amd64.whl", hash = "sha256:25ac8c08322002b06fa1d49d1646181f0b2c72f5cbc15a85e80b4c30a544bb15"}, + {file = "ruamel.yaml.clib-0.2.8.tar.gz", hash = "sha256:beb2e0404003de9a4cab9753a8805a8fe9320ee6673136ed7f04255fe60bb512"}, +] + +[[package]] +name = "setuptools" +version = "65.7.0" +description = "Easily download, build, install, upgrade, and uninstall Python packages" +optional = false +python-versions = ">=3.7" +files = [ + {file = "setuptools-65.7.0-py3-none-any.whl", hash = "sha256:8ab4f1dbf2b4a65f7eec5ad0c620e84c34111a68d3349833494b9088212214dd"}, + {file = "setuptools-65.7.0.tar.gz", hash = "sha256:4d3c92fac8f1118bb77a22181355e29c239cabfe2b9effdaa665c66b711136d7"}, +] + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-hoverxref (<2)", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (==0.8.3)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8 (<5)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pip-run (>=8.8)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.3)", "pytest-flake8", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] + +[[package]] +name = "setuptools-scm" +version = "7.1.0" +description = "the blessed package to manage your versions by scm tags" +optional = false +python-versions = ">=3.7" +files = [ + {file = "setuptools_scm-7.1.0-py3-none-any.whl", hash = "sha256:73988b6d848709e2af142aa48c986ea29592bbcfca5375678064708205253d8e"}, + {file = "setuptools_scm-7.1.0.tar.gz", hash = "sha256:6c508345a771aad7d56ebff0e70628bf2b0ec7573762be9960214730de278f27"}, +] + +[package.dependencies] +packaging = ">=20.0" +setuptools = "*" +tomli = {version = ">=1.0.0", markers = "python_version < \"3.11\""} +typing-extensions = "*" + +[package.extras] +test = ["pytest (>=6.2)", "virtualenv (>20)"] +toml = ["setuptools (>=42)"] + +[[package]] +name = "shapely" +version = "1.8.5.post1" +description = "Geometric objects, predicates, and operations" +optional = false +python-versions = ">=3.6" +files = [ + {file = "Shapely-1.8.5.post1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:d048f93e42ba578b82758c15d8ae037d08e69d91d9872bca5a1895b118f4e2b0"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:99ab0ddc05e44acabdbe657c599fdb9b2d82e86c5493bdae216c0c4018a82dee"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:99a2f0da0109e81e0c101a2b4cd8412f73f5f299e7b5b2deaf64cd2a100ac118"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6fe855e7d45685926b6ba00aaeb5eba5862611f7465775dacd527e081a8ced6d"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:ec14ceca36f67cb48b34d02d7f65a9acae15cd72b48e303531893ba4a960f3ea"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d8a2b2a65fa7f97115c1cd989fe9d6f39281ca2a8a014f1d4904c1a6e34d7f25"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-win32.whl", hash = "sha256:21776184516a16bf82a0c3d6d6a312b3cd15a4cabafc61ee01cf2714a82e8396"}, + {file = "Shapely-1.8.5.post1-cp310-cp310-win_amd64.whl", hash = "sha256:a354199219c8d836f280b88f2c5102c81bb044ccea45bd361dc38a79f3873714"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:783bad5f48e2708a0e2f695a34ed382e4162c795cb2f0368b39528ac1d6db7ed"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:a23ef3882d6aa203dd3623a3d55d698f59bfbd9f8a3bfed52c2da05a7f0f8640"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ab38f7b5196ace05725e407cb8cab9ff66edb8e6f7bb36a398e8f73f52a7aaa2"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:8d086591f744be483b34628b391d741e46f2645fe37594319e0a673cc2c26bcf"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4728666fff8cccc65a07448cae72c75a8773fea061c3f4f139c44adc429b18c3"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-win32.whl", hash = "sha256:84010db15eb364a52b74ea8804ef92a6a930dfc1981d17a369444b6ddec66efd"}, + {file = "Shapely-1.8.5.post1-cp311-cp311-win_amd64.whl", hash = "sha256:48dcfffb9e225c0481120f4bdf622131c8c95f342b00b158cdbe220edbbe20b6"}, + {file = "Shapely-1.8.5.post1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:2fd15397638df291c427a53d641d3e6fd60458128029c8c4f487190473a69a91"}, + {file = "Shapely-1.8.5.post1-cp36-cp36m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a74631e511153366c6dbe3229fa93f877e3c87ea8369cd00f1d38c76b0ed9ace"}, + {file = "Shapely-1.8.5.post1-cp36-cp36m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:66bdac74fbd1d3458fa787191a90fa0ae610f09e2a5ec398c36f968cc0ed743f"}, + {file = "Shapely-1.8.5.post1-cp36-cp36m-win32.whl", hash = "sha256:6d388c0c1bd878ed1af4583695690aa52234b02ed35f93a1c8486ff52a555838"}, + {file = "Shapely-1.8.5.post1-cp36-cp36m-win_amd64.whl", hash = "sha256:be9423d5a3577ac2e92c7e758bd8a2b205f5e51a012177a590bc46fc51eb4834"}, + {file = "Shapely-1.8.5.post1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5d7f85c2d35d39ff53c9216bc76b7641c52326f7e09aaad1789a3611a0f812f2"}, + {file = "Shapely-1.8.5.post1-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:adcf8a11b98af9375e32bff91de184f33a68dc48b9cb9becad4f132fa25cfa3c"}, + {file = "Shapely-1.8.5.post1-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:753ed0e21ab108bd4282405b9b659f2e985e8502b1a72b978eaa51d3496dee19"}, + {file = "Shapely-1.8.5.post1-cp37-cp37m-win32.whl", hash = "sha256:65b21243d8f6bcd421210daf1fabb9de84de2c04353c5b026173b88d17c1a581"}, + {file = "Shapely-1.8.5.post1-cp37-cp37m-win_amd64.whl", hash = "sha256:370b574c78dc5af3a198a6da5d9b3d7c04654bd2ef7e80e80a3a0992dfb2d9cd"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:532a55ee2a6c52d23d6f7d1567c8f0473635f3b270262c44e1b0c88096827e22"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:3480657460e939f45a7d359ef0e172a081f249312557fe9aa78c4fd3a362d993"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b65f5d530ba91e49ffc7c589255e878d2506a8b96ffce69d3b7c4500a9a9eaf8"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:147066da0be41b147a61f8eb805dea3b13709dbc873a431ccd7306e24d712bc0"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c2822111ddc5bcfb116e6c663e403579d0fe3f147d2a97426011a191c43a7458"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4b47bb6f9369e8bf3e6dbd33e6a25a47ee02b2874792a529fe04a49bf8bc0df6"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-win32.whl", hash = "sha256:2e0a8c2e55f1be1312b51c92b06462ea89e6bb703fab4b114e7a846d941cfc40"}, + {file = "Shapely-1.8.5.post1-cp38-cp38-win_amd64.whl", hash = "sha256:0d885cb0cf670c1c834df3f371de8726efdf711f18e2a75da5cfa82843a7ab65"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:0b4ee3132ee90f07d63db3aea316c4c065ed7a26231458dda0874414a09d6ba3"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:02dd5d7dc6e46515d88874134dc8fcdc65826bca93c3eecee59d1910c42c1b17"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c6a9a4a31cd6e86d0fbe8473ceed83d4fe760b19d949fb557ef668defafea0f6"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:38f0fbbcb8ca20c16451c966c1f527cc43968e121c8a048af19ed3e339a921cd"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:78fb9d929b8ee15cfd424b6c10879ce1907f24e05fb83310fc47d2cd27088e40"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:89164e7a9776a19e29f01369a98529321994e2e4d852b92b7e01d4d9804c55bf"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-win32.whl", hash = "sha256:8e59817b0fe63d34baedaabba8c393c0090f061917d18fc0bcc2f621937a8f73"}, + {file = "Shapely-1.8.5.post1-cp39-cp39-win_amd64.whl", hash = "sha256:e9c30b311de2513555ab02464ebb76115d242842b29c412f5a9aa0cac57be9f6"}, + {file = "Shapely-1.8.5.post1.tar.gz", hash = "sha256:ef3be705c3eac282a28058e6c6e5503419b250f482320df2172abcbea642c831"}, +] + +[package.extras] +all = ["numpy", "pytest", "pytest-cov"] +test = ["pytest", "pytest-cov"] +vectorized = ["numpy"] + +[[package]] +name = "shellingham" +version = "1.5.4" +description = "Tool to Detect Surrounding Shell" +optional = false +python-versions = ">=3.7" +files = [ + {file = "shellingham-1.5.4-py2.py3-none-any.whl", hash = "sha256:7ecfff8f2fd72616f7481040475a65b2bf8af90a56c89140852d1120324e8686"}, + {file = "shellingham-1.5.4.tar.gz", hash = "sha256:8dbca0739d487e5bd35ab3ca4b36e11c4078f3a234bfce294b0a0291363404de"}, +] + +[[package]] +name = "simpleeval" +version = "0.9.13" +description = "A simple, safe single expression evaluator library." +optional = false +python-versions = "*" +files = [ + {file = "simpleeval-0.9.13-py2.py3-none-any.whl", hash = "sha256:22a2701a5006e4188d125d34accf2405c2c37c93f6b346f2484b6422415ae54a"}, + {file = "simpleeval-0.9.13.tar.gz", hash = "sha256:4a30f9cc01825fe4c719c785e3762623e350c4840d5e6855c2a8496baaa65fac"}, +] + +[[package]] +name = "six" +version = "1.16.0" +description = "Python 2 and 3 compatibility utilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"}, + {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, +] + +[[package]] +name = "sniffio" +version = "1.3.0" +description = "Sniff out which async library your code is running under" +optional = false +python-versions = ">=3.7" +files = [ + {file = "sniffio-1.3.0-py3-none-any.whl", hash = "sha256:eecefdce1e5bbfb7ad2eeaabf7c1eeb404d7757c379bd1f7e5cce9d8bf425384"}, + {file = "sniffio-1.3.0.tar.gz", hash = "sha256:e60305c5e5d314f5389259b7f22aaa33d8f7dee49763119234af3755c55b9101"}, +] + +[[package]] +name = "snoop" +version = "0.4.3" +description = "Powerful debugging tools for Python" +optional = false +python-versions = "*" +files = [ + {file = "snoop-0.4.3-py2.py3-none-any.whl", hash = "sha256:b7418581889ff78b29d9dc5ad4625c4c475c74755fb5cba82c693c6e32afadc0"}, + {file = "snoop-0.4.3.tar.gz", hash = "sha256:2e0930bb19ff0dbdaa6f5933f88e89ed5984210ea9f9de0e1d8231fa5c1c1f25"}, +] + +[package.dependencies] +asttokens = "*" +cheap-repr = ">=0.4.0" +executing = "*" +pygments = "*" +six = "*" + +[package.extras] +tests = ["Django", "birdseye", "littleutils", "numpy (>=1.16.5)", "pandas (>=0.24.2)", "pprintpp", "prettyprinter", "pytest", "pytest-order", "pytest-order (<=0.11.0)"] + +[[package]] +name = "snowballstemmer" +version = "2.2.0" +description = "This package provides 29 stemmers for 28 languages generated from Snowball algorithms." +optional = false +python-versions = "*" +files = [ + {file = "snowballstemmer-2.2.0-py2.py3-none-any.whl", hash = "sha256:c8e1716e83cc398ae16824e5572ae04e0d9fc2c6b985fb0f900f5f0c96ecba1a"}, + {file = "snowballstemmer-2.2.0.tar.gz", hash = "sha256:09b16deb8547d3412ad7b590689584cd0fe25ec8db3be37788be3810cbf19cb1"}, +] + +[[package]] +name = "sphinx" +version = "5.3.0" +description = "Python documentation generator" +optional = false +python-versions = ">=3.6" +files = [ + {file = "Sphinx-5.3.0.tar.gz", hash = "sha256:51026de0a9ff9fc13c05d74913ad66047e104f56a129ff73e174eb5c3ee794b5"}, + {file = "sphinx-5.3.0-py3-none-any.whl", hash = "sha256:060ca5c9f7ba57a08a1219e547b269fadf125ae25b06b9fa7f66768efb652d6d"}, +] + +[package.dependencies] +alabaster = ">=0.7,<0.8" +babel = ">=2.9" +colorama = {version = ">=0.4.5", markers = "sys_platform == \"win32\""} +docutils = ">=0.14,<0.20" +imagesize = ">=1.3" +importlib-metadata = {version = ">=4.8", markers = "python_version < \"3.10\""} +Jinja2 = ">=3.0" +packaging = ">=21.0" +Pygments = ">=2.12" +requests = ">=2.5.0" +snowballstemmer = ">=2.0" +sphinxcontrib-applehelp = "*" +sphinxcontrib-devhelp = "*" +sphinxcontrib-htmlhelp = ">=2.0.0" +sphinxcontrib-jsmath = "*" +sphinxcontrib-qthelp = "*" +sphinxcontrib-serializinghtml = ">=1.1.5" + +[package.extras] +docs = ["sphinxcontrib-websupport"] +lint = ["docutils-stubs", "flake8 (>=3.5.0)", "flake8-bugbear", "flake8-comprehensions", "flake8-simplify", "isort", "mypy (>=0.981)", "sphinx-lint", "types-requests", "types-typed-ast"] +test = ["cython", "html5lib", "pytest (>=4.6)", "typed_ast"] + +[[package]] +name = "sphinx-autobuild" +version = "2021.3.14" +description = "Rebuild Sphinx documentation on changes, with live-reload in the browser." +optional = false +python-versions = ">=3.6" +files = [ + {file = "sphinx-autobuild-2021.3.14.tar.gz", hash = "sha256:de1ca3b66e271d2b5b5140c35034c89e47f263f2cd5db302c9217065f7443f05"}, + {file = "sphinx_autobuild-2021.3.14-py3-none-any.whl", hash = "sha256:8fe8cbfdb75db04475232f05187c776f46f6e9e04cacf1e49ce81bdac649ccac"}, +] + +[package.dependencies] +colorama = "*" +livereload = "*" +sphinx = "*" + +[package.extras] +test = ["pytest", "pytest-cov"] + +[[package]] +name = "sphinx-rtd-theme" +version = "1.3.0" +description = "Read the Docs theme for Sphinx" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" +files = [ + {file = "sphinx_rtd_theme-1.3.0-py2.py3-none-any.whl", hash = "sha256:46ddef89cc2416a81ecfbeaceab1881948c014b1b6e4450b815311a89fb977b0"}, + {file = "sphinx_rtd_theme-1.3.0.tar.gz", hash = "sha256:590b030c7abb9cf038ec053b95e5380b5c70d61591eb0b552063fbe7c41f0931"}, +] + +[package.dependencies] +docutils = "<0.19" +sphinx = ">=1.6,<8" +sphinxcontrib-jquery = ">=4,<5" + +[package.extras] +dev = ["bump2version", "sphinxcontrib-httpdomain", "transifex-client", "wheel"] + +[[package]] +name = "sphinxcontrib-applehelp" +version = "1.0.4" +description = "sphinxcontrib-applehelp is a Sphinx extension which outputs Apple help books" +optional = false +python-versions = ">=3.8" +files = [ + {file = "sphinxcontrib-applehelp-1.0.4.tar.gz", hash = "sha256:828f867945bbe39817c210a1abfd1bc4895c8b73fcaade56d45357a348a07d7e"}, + {file = "sphinxcontrib_applehelp-1.0.4-py3-none-any.whl", hash = "sha256:29d341f67fb0f6f586b23ad80e072c8e6ad0b48417db2bde114a4c9746feb228"}, +] + +[package.extras] +lint = ["docutils-stubs", "flake8", "mypy"] +test = ["pytest"] + +[[package]] +name = "sphinxcontrib-devhelp" +version = "1.0.2" +description = "sphinxcontrib-devhelp is a sphinx extension which outputs Devhelp document." +optional = false +python-versions = ">=3.5" +files = [ + {file = "sphinxcontrib-devhelp-1.0.2.tar.gz", hash = "sha256:ff7f1afa7b9642e7060379360a67e9c41e8f3121f2ce9164266f61b9f4b338e4"}, + {file = "sphinxcontrib_devhelp-1.0.2-py2.py3-none-any.whl", hash = "sha256:8165223f9a335cc1af7ffe1ed31d2871f325254c0423bc0c4c7cd1c1e4734a2e"}, +] + +[package.extras] +lint = ["docutils-stubs", "flake8", "mypy"] +test = ["pytest"] + +[[package]] +name = "sphinxcontrib-htmlhelp" +version = "2.0.1" +description = "sphinxcontrib-htmlhelp is a sphinx extension which renders HTML help files" +optional = false +python-versions = ">=3.8" +files = [ + {file = "sphinxcontrib-htmlhelp-2.0.1.tar.gz", hash = "sha256:0cbdd302815330058422b98a113195c9249825d681e18f11e8b1f78a2f11efff"}, + {file = "sphinxcontrib_htmlhelp-2.0.1-py3-none-any.whl", hash = "sha256:c38cb46dccf316c79de6e5515e1770414b797162b23cd3d06e67020e1d2a6903"}, +] + +[package.extras] +lint = ["docutils-stubs", "flake8", "mypy"] +test = ["html5lib", "pytest"] + +[[package]] +name = "sphinxcontrib-httpdomain" +version = "1.8.1" +description = "Sphinx domain for documenting HTTP APIs" +optional = false +python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" +files = [ + {file = "sphinxcontrib-httpdomain-1.8.1.tar.gz", hash = "sha256:6c2dfe6ca282d75f66df333869bb0ce7331c01b475db6809ff9d107b7cdfe04b"}, + {file = "sphinxcontrib_httpdomain-1.8.1-py2.py3-none-any.whl", hash = "sha256:21eefe1270e4d9de8d717cc89ee92cc4871b8736774393bafc5e38a6bb77b1d5"}, +] + +[package.dependencies] +six = "*" +Sphinx = ">=1.6" + +[[package]] +name = "sphinxcontrib-jquery" +version = "4.1" +description = "Extension to include jQuery on newer Sphinx releases" +optional = false +python-versions = ">=2.7" +files = [ + {file = "sphinxcontrib-jquery-4.1.tar.gz", hash = "sha256:1620739f04e36a2c779f1a131a2dfd49b2fd07351bf1968ced074365933abc7a"}, + {file = "sphinxcontrib_jquery-4.1-py2.py3-none-any.whl", hash = "sha256:f936030d7d0147dd026a4f2b5a57343d233f1fc7b363f68b3d4f1cb0993878ae"}, +] + +[package.dependencies] +Sphinx = ">=1.8" + +[[package]] +name = "sphinxcontrib-jsmath" +version = "1.0.1" +description = "A sphinx extension which renders display math in HTML via JavaScript" +optional = false +python-versions = ">=3.5" +files = [ + {file = "sphinxcontrib-jsmath-1.0.1.tar.gz", hash = "sha256:a9925e4a4587247ed2191a22df5f6970656cb8ca2bd6284309578f2153e0c4b8"}, + {file = "sphinxcontrib_jsmath-1.0.1-py2.py3-none-any.whl", hash = "sha256:2ec2eaebfb78f3f2078e73666b1415417a116cc848b72e5172e596c871103178"}, +] + +[package.extras] +test = ["flake8", "mypy", "pytest"] + +[[package]] +name = "sphinxcontrib-qthelp" +version = "1.0.3" +description = "sphinxcontrib-qthelp is a sphinx extension which outputs QtHelp document." +optional = false +python-versions = ">=3.5" +files = [ + {file = "sphinxcontrib-qthelp-1.0.3.tar.gz", hash = "sha256:4c33767ee058b70dba89a6fc5c1892c0d57a54be67ddd3e7875a18d14cba5a72"}, + {file = "sphinxcontrib_qthelp-1.0.3-py2.py3-none-any.whl", hash = "sha256:bd9fc24bcb748a8d51fd4ecaade681350aa63009a347a8c14e637895444dfab6"}, +] + +[package.extras] +lint = ["docutils-stubs", "flake8", "mypy"] +test = ["pytest"] + +[[package]] +name = "sphinxcontrib-serializinghtml" +version = "1.1.5" +description = "sphinxcontrib-serializinghtml is a sphinx extension which outputs \"serialized\" HTML files (json and pickle)." +optional = false +python-versions = ">=3.5" +files = [ + {file = "sphinxcontrib-serializinghtml-1.1.5.tar.gz", hash = "sha256:aa5f6de5dfdf809ef505c4895e51ef5c9eac17d0f287933eb49ec495280b6952"}, + {file = "sphinxcontrib_serializinghtml-1.1.5-py2.py3-none-any.whl", hash = "sha256:352a9a00ae864471d3a7ead8d7d79f5fc0b57e8b3f95e9867eb9eb28999b92fd"}, +] + +[package.extras] +lint = ["docutils-stubs", "flake8", "mypy"] +test = ["pytest"] + +[[package]] +name = "sqlalchemy" +version = "1.4.50" +description = "Database Abstraction Library" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" +files = [ + {file = "SQLAlchemy-1.4.50-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d00665725063692c42badfd521d0c4392e83c6c826795d38eb88fb108e5660e5"}, + {file = "SQLAlchemy-1.4.50-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:85292ff52ddf85a39367057c3d7968a12ee1fb84565331a36a8fead346f08796"}, + {file = "SQLAlchemy-1.4.50-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d0fed0f791d78e7767c2db28d34068649dfeea027b83ed18c45a423f741425cb"}, + {file = "SQLAlchemy-1.4.50-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:db4db3c08ffbb18582f856545f058a7a5e4ab6f17f75795ca90b3c38ee0a8ba4"}, + {file = "SQLAlchemy-1.4.50-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:14b0cacdc8a4759a1e1bd47dc3ee3f5db997129eb091330beda1da5a0e9e5bd7"}, + {file = "SQLAlchemy-1.4.50-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1fb9cb60e0f33040e4f4681e6658a7eb03b5cb4643284172f91410d8c493dace"}, + {file = "SQLAlchemy-1.4.50-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c4cb501d585aa74a0f86d0ea6263b9c5e1d1463f8f9071392477fd401bd3c7cc"}, + {file = "SQLAlchemy-1.4.50-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8a7a66297e46f85a04d68981917c75723e377d2e0599d15fbe7a56abed5e2d75"}, + {file = "SQLAlchemy-1.4.50-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c1db0221cb26d66294f4ca18c533e427211673ab86c1fbaca8d6d9ff78654293"}, + {file = "SQLAlchemy-1.4.50-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0b7dbe6369677a2bea68fe9812c6e4bbca06ebfa4b5cde257b2b0bf208709131"}, + {file = "SQLAlchemy-1.4.50-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:a9bddb60566dc45c57fd0a5e14dd2d9e5f106d2241e0a2dc0c1da144f9444516"}, + {file = "SQLAlchemy-1.4.50-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:82dd4131d88395df7c318eeeef367ec768c2a6fe5bd69423f7720c4edb79473c"}, + {file = "SQLAlchemy-1.4.50-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:273505fcad22e58cc67329cefab2e436006fc68e3c5423056ee0513e6523268a"}, + {file = "SQLAlchemy-1.4.50-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a3257a6e09626d32b28a0c5b4f1a97bced585e319cfa90b417f9ab0f6145c33c"}, + {file = "SQLAlchemy-1.4.50-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d69738d582e3a24125f0c246ed8d712b03bd21e148268421e4a4d09c34f521a5"}, + {file = "SQLAlchemy-1.4.50-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:34e1c5d9cd3e6bf3d1ce56971c62a40c06bfc02861728f368dcfec8aeedb2814"}, + {file = "SQLAlchemy-1.4.50-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f1fcee5a2c859eecb4ed179edac5ffbc7c84ab09a5420219078ccc6edda45436"}, + {file = "SQLAlchemy-1.4.50-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fbaf6643a604aa17e7a7afd74f665f9db882df5c297bdd86c38368f2c471f37d"}, + {file = "SQLAlchemy-1.4.50-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:2e70e0673d7d12fa6cd363453a0d22dac0d9978500aa6b46aa96e22690a55eab"}, + {file = "SQLAlchemy-1.4.50-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8b881ac07d15fb3e4f68c5a67aa5cdaf9eb8f09eb5545aaf4b0a5f5f4659be18"}, + {file = "SQLAlchemy-1.4.50-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3f6997da81114daef9203d30aabfa6b218a577fc2bd797c795c9c88c9eb78d49"}, + {file = "SQLAlchemy-1.4.50-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bdb77e1789e7596b77fd48d99ec1d2108c3349abd20227eea0d48d3f8cf398d9"}, + {file = "SQLAlchemy-1.4.50-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:128a948bd40780667114b0297e2cc6d657b71effa942e0a368d8cc24293febb3"}, + {file = "SQLAlchemy-1.4.50-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f2d526aeea1bd6a442abc7c9b4b00386fd70253b80d54a0930c0a216230a35be"}, + {file = "SQLAlchemy-1.4.50.tar.gz", hash = "sha256:3b97ddf509fc21e10b09403b5219b06c5b558b27fc2453150274fa4e70707dbf"}, +] + +[package.dependencies] +greenlet = {version = "!=0.4.17", markers = "python_version >= \"3\" and (platform_machine == \"win32\" or platform_machine == \"WIN32\" or platform_machine == \"AMD64\" or platform_machine == \"amd64\" or platform_machine == \"x86_64\" or platform_machine == \"ppc64le\" or platform_machine == \"aarch64\")"} + +[package.extras] +aiomysql = ["aiomysql (>=0.2.0)", "greenlet (!=0.4.17)"] +aiosqlite = ["aiosqlite", "greenlet (!=0.4.17)", "typing-extensions (!=3.10.0.1)"] +asyncio = ["greenlet (!=0.4.17)"] +asyncmy = ["asyncmy (>=0.2.3,!=0.2.4)", "greenlet (!=0.4.17)"] +mariadb-connector = ["mariadb (>=1.0.1,!=1.1.2)"] +mssql = ["pyodbc"] +mssql-pymssql = ["pymssql"] +mssql-pyodbc = ["pyodbc"] +mypy = ["mypy (>=0.910)", "sqlalchemy2-stubs"] +mysql = ["mysqlclient (>=1.4.0)", "mysqlclient (>=1.4.0,<2)"] +mysql-connector = ["mysql-connector-python"] +oracle = ["cx-oracle (>=7)", "cx-oracle (>=7,<8)"] +postgresql = ["psycopg2 (>=2.7)"] +postgresql-asyncpg = ["asyncpg", "greenlet (!=0.4.17)"] +postgresql-pg8000 = ["pg8000 (>=1.16.6,!=1.29.0)"] +postgresql-psycopg2binary = ["psycopg2-binary"] +postgresql-psycopg2cffi = ["psycopg2cffi"] +pymysql = ["pymysql", "pymysql (<1)"] +sqlcipher = ["sqlcipher3-binary"] + +[[package]] +name = "sqlalchemy-stubs" +version = "0.4" +description = "SQLAlchemy stubs and mypy plugin" +optional = false +python-versions = "*" +files = [ + {file = "sqlalchemy-stubs-0.4.tar.gz", hash = "sha256:c665d6dd4482ef642f01027fa06c3d5e91befabb219dc71fc2a09e7d7695f7ae"}, + {file = "sqlalchemy_stubs-0.4-py3-none-any.whl", hash = "sha256:5eec7aa110adf9b957b631799a72fef396b23ff99fe296df726645d01e312aa5"}, +] + +[package.dependencies] +mypy = ">=0.790" +typing-extensions = ">=3.7.4" + +[[package]] +name = "sqlalchemy-utils" +version = "0.38.3" +description = "Various utility functions for SQLAlchemy." +optional = false +python-versions = "~=3.6" +files = [ + {file = "SQLAlchemy-Utils-0.38.3.tar.gz", hash = "sha256:9f9afba607a40455cf703adfa9846584bf26168a0c5a60a70063b70d65051f4d"}, + {file = "SQLAlchemy_Utils-0.38.3-py3-none-any.whl", hash = "sha256:5c13b5d08adfaa85f3d4e8ec09a75136216fad41346980d02974a70a77988bf9"}, +] + +[package.dependencies] +SQLAlchemy = ">=1.3" + +[package.extras] +arrow = ["arrow (>=0.3.4)"] +babel = ["Babel (>=1.3)"] +color = ["colour (>=0.0.4)"] +encrypted = ["cryptography (>=0.6)"] +intervals = ["intervals (>=0.7.1)"] +password = ["passlib (>=1.6,<2.0)"] +pendulum = ["pendulum (>=2.0.5)"] +phone = ["phonenumbers (>=5.9.2)"] +test = ["Jinja2 (>=2.3)", "Pygments (>=1.2)", "backports.zoneinfo", "docutils (>=0.10)", "flake8 (>=2.4.0)", "flexmock (>=0.9.7)", "isort (>=4.2.2)", "pg8000 (>=1.12.4)", "psycopg2 (>=2.5.1)", "psycopg2cffi (>=2.8.1)", "pymysql", "pyodbc", "pytest (>=2.7.1)", "python-dateutil (>=2.6)", "pytz (>=2014.2)"] +test-all = ["Babel (>=1.3)", "Jinja2 (>=2.3)", "Pygments (>=1.2)", "arrow (>=0.3.4)", "backports.zoneinfo", "colour (>=0.0.4)", "cryptography (>=0.6)", "docutils (>=0.10)", "flake8 (>=2.4.0)", "flexmock (>=0.9.7)", "furl (>=0.4.1)", "intervals (>=0.7.1)", "isort (>=4.2.2)", "passlib (>=1.6,<2.0)", "pendulum (>=2.0.5)", "pg8000 (>=1.12.4)", "phonenumbers (>=5.9.2)", "psycopg2 (>=2.5.1)", "psycopg2cffi (>=2.8.1)", "pymysql", "pyodbc", "pytest (>=2.7.1)", "python-dateutil", "python-dateutil (>=2.6)", "pytz (>=2014.2)"] +timezone = ["python-dateutil"] +url = ["furl (>=0.4.1)"] + +[[package]] +name = "sqlean-py" +version = "0.21.8.5" +description = "sqlite3 with extensions" +optional = false +python-versions = "*" +files = [ + {file = "sqlean.py-0.21.8.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:bf5f546be1de81d3b27f2ee2ddc18ba8d228d6faadef261cdb15c792ba49e7b6"}, + {file = "sqlean.py-0.21.8.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c87c5e06f12e511945e44dbcf750d1278c4b8895beb286d47fa6c8ec473e432f"}, + {file = "sqlean.py-0.21.8.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5c12a2f519328ccebeedc4ecdcfe722c95b0adcb4ecb5d46235c9ab36e32afba"}, + {file = "sqlean.py-0.21.8.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:99d051243fa38c3a4c17c32817780cb6319afc8ab86a34e21953d96e7b3823d4"}, + {file = "sqlean.py-0.21.8.5-cp310-cp310-win_amd64.whl", hash = "sha256:e12be5422c7c0aaab6795452fa1b9bfb809718874007a6d3f06527142cefde1b"}, + {file = "sqlean.py-0.21.8.5-cp310-cp310-win_arm64.whl", hash = "sha256:4b432865c2ca29574b89b42eeeae01a3fc8527359f4c66766a7659b59fcf6789"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ca49a20fdcca110518a6900ab462ae2072b2e5e78fa4453bb542262a46bd510e"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:0985e78f8f1c9c6e6dd0df0f6899b9a51fbf2b19fff1852885f8751b949d3308"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:acf9178519354e11db5634d8f591f934d15c73d4502c73122546ec2512505084"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:692b0664d4f43d622917ca698d47511211f06d1497e041e613df5bceab511273"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-win_amd64.whl", hash = "sha256:71f836aee785aa583c7b66b14678ab3f2deaeb3d537a20bac2669f1ecd956904"}, + {file = "sqlean.py-0.21.8.5-cp311-cp311-win_arm64.whl", hash = "sha256:1368f77ce4f665f858f09cf1f77b48ff1de747ec52e7b5939c7a8caec0d3a1ed"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:c81aec740da911d54e89bd9b6918da5926d98ae9f87cdc972a9a2745c49fd73e"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:655c51f43b92ef035510a96bcd9365a7afbad7aabc099ba03aa08b7b260966d6"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5e101bb54f267e07bc9a0a276c8ad4c939309b0d4ab2f62f56b6b6ff7d0124a"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:33c1d614af0ffbbd781c56ba975dfb03cdb4f560400a3948608aca87c6b49126"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-win_amd64.whl", hash = "sha256:b632486718e3866867a409af367043e6b6c4922a21ffde70734ffe6cb6d446ab"}, + {file = "sqlean.py-0.21.8.5-cp312-cp312-win_arm64.whl", hash = "sha256:350dd22de76a2ab44f3ba89582e99aa089ae90e1644b51f1aeaf897634b6d3c7"}, + {file = "sqlean.py-0.21.8.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:9b6633514a18d3dffe576a70d02ba8949b63f378953cdf61cd34d9918c067f2f"}, + {file = "sqlean.py-0.21.8.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:05d1c21fdb3d4059401ba4980b2f0599a23aea0a39c79d07b803bb20e33522fe"}, + {file = "sqlean.py-0.21.8.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0190a4dbaaab40b2bed1786edb8df8e477757671eed63921f2fe90b3f4f42a14"}, + {file = "sqlean.py-0.21.8.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c254dfbfa69d13ebf6c257ac8f1f075e4e9a81f459110ff37de3b420c453f8af"}, + {file = "sqlean.py-0.21.8.5-cp38-cp38-win_amd64.whl", hash = "sha256:256ae2c8215def61651495d56cb6408d7657620319c5af6555d62159ad0bea06"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:96cb39fbeb1d89121b08e2d49edcad0c7f4d3dc4c6d370459162b8fb6b25755b"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:6fd03218bc2b9af800b716a1208bd66b1eddc604ece4654cb5d1488c3bb2e69f"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:77bd0e4f105074237b72190d63d6bc0576a8f5fc085f8b1959335454dfa3003b"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b49891876176e3b3092acf1fd4ae4d6147367341f5cf58501fd2d8cfbc5427f0"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-win_amd64.whl", hash = "sha256:8a1b729e57628aa7cf5f7d738b792788bac2f566459dcd16f9aea1ffc7f296b5"}, + {file = "sqlean.py-0.21.8.5-cp39-cp39-win_arm64.whl", hash = "sha256:75f049c68067e7fe8a64538559cc084d29c0ba363ffdc6ec794fc35ed1564a02"}, + {file = "sqlean.py-0.21.8.5.tar.gz", hash = "sha256:033a641f8b8146087a5879d8c9f373ae376bf463c00e8de728daff0c29be3bb7"}, +] + +[[package]] +name = "sqlparse" +version = "0.4.4" +description = "A non-validating SQL parser." +optional = false +python-versions = ">=3.5" +files = [ + {file = "sqlparse-0.4.4-py3-none-any.whl", hash = "sha256:5430a4fe2ac7d0f93e66f1efc6e1338a41884b7ddf2a350cedd20ccc4d9d28f3"}, + {file = "sqlparse-0.4.4.tar.gz", hash = "sha256:d446183e84b8349fa3061f0fe7f06ca94ba65b426946ffebe6e3e8295332420c"}, +] + +[package.extras] +dev = ["build", "flake8"] +doc = ["sphinx"] +test = ["pytest", "pytest-cov"] + +[[package]] +name = "stack-data" +version = "0.6.3" +description = "Extract data from python stack frames and tracebacks for informative displays" +optional = false +python-versions = "*" +files = [ + {file = "stack_data-0.6.3-py3-none-any.whl", hash = "sha256:d5558e0c25a4cb0853cddad3d77da9891a08cb85dd9f9f91b9f8cd66e511e695"}, + {file = "stack_data-0.6.3.tar.gz", hash = "sha256:836a778de4fec4dcd1dcd89ed8abff8a221f58308462e1c4aa2a3cf30148f0b9"}, +] + +[package.dependencies] +asttokens = ">=2.1.0" +executing = ">=1.2.0" +pure-eval = "*" + +[package.extras] +tests = ["cython", "littleutils", "pygments", "pytest", "typeguard"] + +[[package]] +name = "starlette" +version = "0.22.0" +description = "The little ASGI library that shines." +optional = false +python-versions = ">=3.7" +files = [ + {file = "starlette-0.22.0-py3-none-any.whl", hash = "sha256:b5eda991ad5f0ee5d8ce4c4540202a573bb6691ecd0c712262d0bc85cf8f2c50"}, + {file = "starlette-0.22.0.tar.gz", hash = "sha256:b092cbc365bea34dd6840b42861bdabb2f507f8671e642e8272d2442e08ea4ff"}, +] + +[package.dependencies] +anyio = ">=3.4.0,<5" +typing-extensions = {version = ">=3.10.0", markers = "python_version < \"3.10\""} + +[package.extras] +full = ["httpx (>=0.22.0)", "itsdangerous", "jinja2", "python-multipart", "pyyaml"] + +[[package]] +name = "stringcase" +version = "1.2.0" +description = "String case converter." +optional = false +python-versions = "*" +files = [ + {file = "stringcase-1.2.0.tar.gz", hash = "sha256:48a06980661908efe8d9d34eab2b6c13aefa2163b3ced26972902e3bdfd87008"}, +] + +[[package]] +name = "tabulate" +version = "0.9.0" +description = "Pretty-print tabular data" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tabulate-0.9.0-py3-none-any.whl", hash = "sha256:024ca478df22e9340661486f85298cff5f6dcdba14f3813e8830015b9ed1948f"}, + {file = "tabulate-0.9.0.tar.gz", hash = "sha256:0095b12bf5966de529c0feb1fa08671671b3368eec77d7ef7ab114be2c068b3c"}, +] + +[package.extras] +widechars = ["wcwidth"] + +[[package]] +name = "text-unidecode" +version = "1.3" +description = "The most basic Text::Unidecode port" +optional = false +python-versions = "*" +files = [ + {file = "text-unidecode-1.3.tar.gz", hash = "sha256:bad6603bb14d279193107714b288be206cac565dfa49aa5b105294dd5c4aab93"}, + {file = "text_unidecode-1.3-py2.py3-none-any.whl", hash = "sha256:1311f10e8b895935241623731c2ba64f4c455287888b18189350b67134a822e8"}, +] + +[[package]] +name = "tomli" +version = "2.0.1" +description = "A lil' TOML parser" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tomli-2.0.1-py3-none-any.whl", hash = "sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc"}, + {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"}, +] + +[[package]] +name = "toolz" +version = "0.12.0" +description = "List processing tools and functional utilities" +optional = false +python-versions = ">=3.5" +files = [ + {file = "toolz-0.12.0-py3-none-any.whl", hash = "sha256:2059bd4148deb1884bb0eb770a3cde70e7f954cfbbdc2285f1f2de01fd21eb6f"}, + {file = "toolz-0.12.0.tar.gz", hash = "sha256:88c570861c440ee3f2f6037c4654613228ff40c93a6c25e0eba70d17282c6194"}, +] + +[[package]] +name = "toposort" +version = "1.10" +description = "Implements a topological sort algorithm." +optional = false +python-versions = "*" +files = [ + {file = "toposort-1.10-py3-none-any.whl", hash = "sha256:cbdbc0d0bee4d2695ab2ceec97fe0679e9c10eab4b2a87a9372b929e70563a87"}, + {file = "toposort-1.10.tar.gz", hash = "sha256:bfbb479c53d0a696ea7402601f4e693c97b0367837c8898bc6471adfca37a6bd"}, +] + +[[package]] +name = "tornado" +version = "6.4" +description = "Tornado is a Python web framework and asynchronous networking library, originally developed at FriendFeed." +optional = false +python-versions = ">= 3.8" +files = [ + {file = "tornado-6.4-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:02ccefc7d8211e5a7f9e8bc3f9e5b0ad6262ba2fbb683a6443ecc804e5224ce0"}, + {file = "tornado-6.4-cp38-abi3-macosx_10_9_x86_64.whl", hash = "sha256:27787de946a9cffd63ce5814c33f734c627a87072ec7eed71f7fc4417bb16263"}, + {file = "tornado-6.4-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f7894c581ecdcf91666a0912f18ce5e757213999e183ebfc2c3fdbf4d5bd764e"}, + {file = "tornado-6.4-cp38-abi3-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e43bc2e5370a6a8e413e1e1cd0c91bedc5bd62a74a532371042a18ef19e10579"}, + {file = "tornado-6.4-cp38-abi3-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f0251554cdd50b4b44362f73ad5ba7126fc5b2c2895cc62b14a1c2d7ea32f212"}, + {file = "tornado-6.4-cp38-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:fd03192e287fbd0899dd8f81c6fb9cbbc69194d2074b38f384cb6fa72b80e9c2"}, + {file = "tornado-6.4-cp38-abi3-musllinux_1_1_i686.whl", hash = "sha256:88b84956273fbd73420e6d4b8d5ccbe913c65d31351b4c004ae362eba06e1f78"}, + {file = "tornado-6.4-cp38-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:71ddfc23a0e03ef2df1c1397d859868d158c8276a0603b96cf86892bff58149f"}, + {file = "tornado-6.4-cp38-abi3-win32.whl", hash = "sha256:6f8a6c77900f5ae93d8b4ae1196472d0ccc2775cc1dfdc9e7727889145c45052"}, + {file = "tornado-6.4-cp38-abi3-win_amd64.whl", hash = "sha256:10aeaa8006333433da48dec9fe417877f8bcc21f48dda8d661ae79da357b2a63"}, + {file = "tornado-6.4.tar.gz", hash = "sha256:72291fa6e6bc84e626589f1c29d90a5a6d593ef5ae68052ee2ef000dfd273dee"}, +] + +[[package]] +name = "tqdm" +version = "4.66.1" +description = "Fast, Extensible Progress Meter" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tqdm-4.66.1-py3-none-any.whl", hash = "sha256:d302b3c5b53d47bce91fea46679d9c3c6508cf6332229aa1e7d8653723793386"}, + {file = "tqdm-4.66.1.tar.gz", hash = "sha256:d88e651f9db8d8551a62556d3cff9e3034274ca5d66e93197cf2490e2dcb69c7"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[package.extras] +dev = ["pytest (>=6)", "pytest-cov", "pytest-timeout", "pytest-xdist"] +notebook = ["ipywidgets (>=6)"] +slack = ["slack-sdk"] +telegram = ["requests"] + +[[package]] +name = "traitlets" +version = "5.14.0" +description = "Traitlets Python configuration system" +optional = false +python-versions = ">=3.8" +files = [ + {file = "traitlets-5.14.0-py3-none-any.whl", hash = "sha256:f14949d23829023013c47df20b4a76ccd1a85effb786dc060f34de7948361b33"}, + {file = "traitlets-5.14.0.tar.gz", hash = "sha256:fcdaa8ac49c04dfa0ed3ee3384ef6dfdb5d6f3741502be247279407679296772"}, +] + +[package.extras] +docs = ["myst-parser", "pydata-sphinx-theme", "sphinx"] +test = ["argcomplete (>=3.0.3)", "mypy (>=1.7.0)", "pre-commit", "pytest (>=7.0,<7.5)", "pytest-mock", "pytest-mypy-testing"] + +[[package]] +name = "typed-ast" +version = "1.4.3" +description = "a fork of Python 2 and 3 ast modules with type comment support" +optional = false +python-versions = "*" +files = [ + {file = "typed_ast-1.4.3-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:2068531575a125b87a41802130fa7e29f26c09a2833fea68d9a40cf33902eba6"}, + {file = "typed_ast-1.4.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:c907f561b1e83e93fad565bac5ba9c22d96a54e7ea0267c708bffe863cbe4075"}, + {file = "typed_ast-1.4.3-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:1b3ead4a96c9101bef08f9f7d1217c096f31667617b58de957f690c92378b528"}, + {file = "typed_ast-1.4.3-cp35-cp35m-win32.whl", hash = "sha256:dde816ca9dac1d9c01dd504ea5967821606f02e510438120091b84e852367428"}, + {file = "typed_ast-1.4.3-cp35-cp35m-win_amd64.whl", hash = "sha256:777a26c84bea6cd934422ac2e3b78863a37017618b6e5c08f92ef69853e765d3"}, + {file = "typed_ast-1.4.3-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f8afcf15cc511ada719a88e013cec87c11aff7b91f019295eb4530f96fe5ef2f"}, + {file = "typed_ast-1.4.3-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:52b1eb8c83f178ab787f3a4283f68258525f8d70f778a2f6dd54d3b5e5fb4341"}, + {file = "typed_ast-1.4.3-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:01ae5f73431d21eead5015997ab41afa53aa1fbe252f9da060be5dad2c730ace"}, + {file = "typed_ast-1.4.3-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:c190f0899e9f9f8b6b7863debfb739abcb21a5c054f911ca3596d12b8a4c4c7f"}, + {file = "typed_ast-1.4.3-cp36-cp36m-win32.whl", hash = "sha256:398e44cd480f4d2b7ee8d98385ca104e35c81525dd98c519acff1b79bdaac363"}, + {file = "typed_ast-1.4.3-cp36-cp36m-win_amd64.whl", hash = "sha256:bff6ad71c81b3bba8fa35f0f1921fb24ff4476235a6e94a26ada2e54370e6da7"}, + {file = "typed_ast-1.4.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0fb71b8c643187d7492c1f8352f2c15b4c4af3f6338f21681d3681b3dc31a266"}, + {file = "typed_ast-1.4.3-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:760ad187b1041a154f0e4d0f6aae3e40fdb51d6de16e5c99aedadd9246450e9e"}, + {file = "typed_ast-1.4.3-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:5feca99c17af94057417d744607b82dd0a664fd5e4ca98061480fd8b14b18d04"}, + {file = "typed_ast-1.4.3-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:95431a26309a21874005845c21118c83991c63ea800dd44843e42a916aec5899"}, + {file = "typed_ast-1.4.3-cp37-cp37m-win32.whl", hash = "sha256:aee0c1256be6c07bd3e1263ff920c325b59849dc95392a05f258bb9b259cf39c"}, + {file = "typed_ast-1.4.3-cp37-cp37m-win_amd64.whl", hash = "sha256:9ad2c92ec681e02baf81fdfa056fe0d818645efa9af1f1cd5fd6f1bd2bdfd805"}, + {file = "typed_ast-1.4.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:b36b4f3920103a25e1d5d024d155c504080959582b928e91cb608a65c3a49e1a"}, + {file = "typed_ast-1.4.3-cp38-cp38-manylinux1_i686.whl", hash = "sha256:067a74454df670dcaa4e59349a2e5c81e567d8d65458d480a5b3dfecec08c5ff"}, + {file = "typed_ast-1.4.3-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7538e495704e2ccda9b234b82423a4038f324f3a10c43bc088a1636180f11a41"}, + {file = "typed_ast-1.4.3-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:af3d4a73793725138d6b334d9d247ce7e5f084d96284ed23f22ee626a7b88e39"}, + {file = "typed_ast-1.4.3-cp38-cp38-win32.whl", hash = "sha256:f2362f3cb0f3172c42938946dbc5b7843c2a28aec307c49100c8b38764eb6927"}, + {file = "typed_ast-1.4.3-cp38-cp38-win_amd64.whl", hash = "sha256:dd4a21253f42b8d2b48410cb31fe501d32f8b9fbeb1f55063ad102fe9c425e40"}, + {file = "typed_ast-1.4.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f328adcfebed9f11301eaedfa48e15bdece9b519fb27e6a8c01aa52a17ec31b3"}, + {file = "typed_ast-1.4.3-cp39-cp39-manylinux1_i686.whl", hash = "sha256:2c726c276d09fc5c414693a2de063f521052d9ea7c240ce553316f70656c84d4"}, + {file = "typed_ast-1.4.3-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:cae53c389825d3b46fb37538441f75d6aecc4174f615d048321b716df2757fb0"}, + {file = "typed_ast-1.4.3-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:b9574c6f03f685070d859e75c7f9eeca02d6933273b5e69572e5ff9d5e3931c3"}, + {file = "typed_ast-1.4.3-cp39-cp39-win32.whl", hash = "sha256:209596a4ec71d990d71d5e0d312ac935d86930e6eecff6ccc7007fe54d703808"}, + {file = "typed_ast-1.4.3-cp39-cp39-win_amd64.whl", hash = "sha256:9c6d1a54552b5330bc657b7ef0eae25d00ba7ffe85d9ea8ae6540d2197a3788c"}, + {file = "typed_ast-1.4.3.tar.gz", hash = "sha256:fb1bbeac803adea29cedd70781399c99138358c26d05fcbd23c13016b7f5ec65"}, +] + +[[package]] +name = "typer" +version = "0.6.1" +description = "Typer, build great CLIs. Easy to code. Based on Python type hints." +optional = false +python-versions = ">=3.6" +files = [ + {file = "typer-0.6.1-py3-none-any.whl", hash = "sha256:54b19e5df18654070a82f8c2aa1da456a4ac16a2a83e6dcd9f170e291c56338e"}, + {file = "typer-0.6.1.tar.gz", hash = "sha256:2d5720a5e63f73eaf31edaa15f6ab87f35f0690f8ca233017d7d23d743a91d73"}, +] + +[package.dependencies] +click = ">=7.1.1,<9.0.0" +colorama = {version = ">=0.4.3,<0.5.0", optional = true, markers = "extra == \"all\""} +rich = {version = ">=10.11.0,<13.0.0", optional = true, markers = "extra == \"all\""} +shellingham = {version = ">=1.3.0,<2.0.0", optional = true, markers = "extra == \"all\""} + +[package.extras] +all = ["colorama (>=0.4.3,<0.5.0)", "rich (>=10.11.0,<13.0.0)", "shellingham (>=1.3.0,<2.0.0)"] +dev = ["autoflake (>=1.3.1,<2.0.0)", "flake8 (>=3.8.3,<4.0.0)", "pre-commit (>=2.17.0,<3.0.0)"] +doc = ["mdx-include (>=1.4.1,<2.0.0)", "mkdocs (>=1.1.2,<2.0.0)", "mkdocs-material (>=8.1.4,<9.0.0)"] +test = ["black (>=22.3.0,<23.0.0)", "coverage (>=5.2,<6.0)", "isort (>=5.0.6,<6.0.0)", "mypy (==0.910)", "pytest (>=4.4.0,<5.4.0)", "pytest-cov (>=2.10.0,<3.0.0)", "pytest-sugar (>=0.9.4,<0.10.0)", "pytest-xdist (>=1.32.0,<2.0.0)", "rich (>=10.11.0,<13.0.0)", "shellingham (>=1.3.0,<2.0.0)"] + +[[package]] +name = "typing-extensions" +version = "4.9.0" +description = "Backported and Experimental Type Hints for Python 3.8+" +optional = false +python-versions = ">=3.8" +files = [ + {file = "typing_extensions-4.9.0-py3-none-any.whl", hash = "sha256:af72aea155e91adfc61c3ae9e0e342dbc0cba726d6cba4b6c72c1f34e47291cd"}, + {file = "typing_extensions-4.9.0.tar.gz", hash = "sha256:23478f88c37f27d76ac8aee6c905017a143b0b1b886c3c9f66bc2fd94f9f5783"}, +] + +[[package]] +name = "tzdata" +version = "2023.3" +description = "Provider of IANA time zone data" +optional = false +python-versions = ">=2" +files = [ + {file = "tzdata-2023.3-py2.py3-none-any.whl", hash = "sha256:7e65763eef3120314099b6939b5546db7adce1e7d6f2e179e3df563c70511eda"}, + {file = "tzdata-2023.3.tar.gz", hash = "sha256:11ef1e08e54acb0d4f95bdb1be05da659673de4acbd21bf9c69e94cc5e907a3a"}, +] + +[[package]] +name = "ujson" +version = "5.9.0" +description = "Ultra fast JSON encoder and decoder for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "ujson-5.9.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:ab71bf27b002eaf7d047c54a68e60230fbd5cd9da60de7ca0aa87d0bccead8fa"}, + {file = "ujson-5.9.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:7a365eac66f5aa7a7fdf57e5066ada6226700884fc7dce2ba5483538bc16c8c5"}, + {file = "ujson-5.9.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e015122b337858dba5a3dc3533af2a8fc0410ee9e2374092f6a5b88b182e9fcc"}, + {file = "ujson-5.9.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:779a2a88c53039bebfbccca934430dabb5c62cc179e09a9c27a322023f363e0d"}, + {file = "ujson-5.9.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:10ca3c41e80509fd9805f7c149068fa8dbee18872bbdc03d7cca928926a358d5"}, + {file = "ujson-5.9.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:4a566e465cb2fcfdf040c2447b7dd9718799d0d90134b37a20dff1e27c0e9096"}, + {file = "ujson-5.9.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:f833c529e922577226a05bc25b6a8b3eb6c4fb155b72dd88d33de99d53113124"}, + {file = "ujson-5.9.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:b68a0caab33f359b4cbbc10065c88e3758c9f73a11a65a91f024b2e7a1257106"}, + {file = "ujson-5.9.0-cp310-cp310-win32.whl", hash = "sha256:7cc7e605d2aa6ae6b7321c3ae250d2e050f06082e71ab1a4200b4ae64d25863c"}, + {file = "ujson-5.9.0-cp310-cp310-win_amd64.whl", hash = "sha256:a6d3f10eb8ccba4316a6b5465b705ed70a06011c6f82418b59278fbc919bef6f"}, + {file = "ujson-5.9.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3b23bbb46334ce51ddb5dded60c662fbf7bb74a37b8f87221c5b0fec1ec6454b"}, + {file = "ujson-5.9.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:6974b3a7c17bbf829e6c3bfdc5823c67922e44ff169851a755eab79a3dd31ec0"}, + {file = "ujson-5.9.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b5964ea916edfe24af1f4cc68488448fbb1ec27a3ddcddc2b236da575c12c8ae"}, + {file = "ujson-5.9.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8ba7cac47dd65ff88571eceeff48bf30ed5eb9c67b34b88cb22869b7aa19600d"}, + {file = "ujson-5.9.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6bbd91a151a8f3358c29355a491e915eb203f607267a25e6ab10531b3b157c5e"}, + {file = "ujson-5.9.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:829a69d451a49c0de14a9fecb2a2d544a9b2c884c2b542adb243b683a6f15908"}, + {file = "ujson-5.9.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:a807ae73c46ad5db161a7e883eec0fbe1bebc6a54890152ccc63072c4884823b"}, + {file = "ujson-5.9.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:8fc2aa18b13d97b3c8ccecdf1a3c405f411a6e96adeee94233058c44ff92617d"}, + {file = "ujson-5.9.0-cp311-cp311-win32.whl", hash = "sha256:70e06849dfeb2548be48fdd3ceb53300640bc8100c379d6e19d78045e9c26120"}, + {file = "ujson-5.9.0-cp311-cp311-win_amd64.whl", hash = "sha256:7309d063cd392811acc49b5016728a5e1b46ab9907d321ebbe1c2156bc3c0b99"}, + {file = "ujson-5.9.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:20509a8c9f775b3a511e308bbe0b72897ba6b800767a7c90c5cca59d20d7c42c"}, + {file = "ujson-5.9.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:b28407cfe315bd1b34f1ebe65d3bd735d6b36d409b334100be8cdffae2177b2f"}, + {file = "ujson-5.9.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9d302bd17989b6bd90d49bade66943c78f9e3670407dbc53ebcf61271cadc399"}, + {file = "ujson-5.9.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9f21315f51e0db8ee245e33a649dd2d9dce0594522de6f278d62f15f998e050e"}, + {file = "ujson-5.9.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5635b78b636a54a86fdbf6f027e461aa6c6b948363bdf8d4fbb56a42b7388320"}, + {file = "ujson-5.9.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:82b5a56609f1235d72835ee109163c7041b30920d70fe7dac9176c64df87c164"}, + {file = "ujson-5.9.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:5ca35f484622fd208f55041b042d9d94f3b2c9c5add4e9af5ee9946d2d30db01"}, + {file = "ujson-5.9.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:829b824953ebad76d46e4ae709e940bb229e8999e40881338b3cc94c771b876c"}, + {file = "ujson-5.9.0-cp312-cp312-win32.whl", hash = "sha256:25fa46e4ff0a2deecbcf7100af3a5d70090b461906f2299506485ff31d9ec437"}, + {file = "ujson-5.9.0-cp312-cp312-win_amd64.whl", hash = "sha256:60718f1720a61560618eff3b56fd517d107518d3c0160ca7a5a66ac949c6cf1c"}, + {file = "ujson-5.9.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d581db9db9e41d8ea0b2705c90518ba623cbdc74f8d644d7eb0d107be0d85d9c"}, + {file = "ujson-5.9.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:ff741a5b4be2d08fceaab681c9d4bc89abf3c9db600ab435e20b9b6d4dfef12e"}, + {file = "ujson-5.9.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cdcb02cabcb1e44381221840a7af04433c1dc3297af76fde924a50c3054c708c"}, + {file = "ujson-5.9.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e208d3bf02c6963e6ef7324dadf1d73239fb7008491fdf523208f60be6437402"}, + {file = "ujson-5.9.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f4b3917296630a075e04d3d07601ce2a176479c23af838b6cf90a2d6b39b0d95"}, + {file = "ujson-5.9.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:0c4d6adb2c7bb9eb7c71ad6f6f612e13b264942e841f8cc3314a21a289a76c4e"}, + {file = "ujson-5.9.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:0b159efece9ab5c01f70b9d10bbb77241ce111a45bc8d21a44c219a2aec8ddfd"}, + {file = "ujson-5.9.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:f0cb4a7814940ddd6619bdce6be637a4b37a8c4760de9373bac54bb7b229698b"}, + {file = "ujson-5.9.0-cp38-cp38-win32.whl", hash = "sha256:dc80f0f5abf33bd7099f7ac94ab1206730a3c0a2d17549911ed2cb6b7aa36d2d"}, + {file = "ujson-5.9.0-cp38-cp38-win_amd64.whl", hash = "sha256:506a45e5fcbb2d46f1a51fead991c39529fc3737c0f5d47c9b4a1d762578fc30"}, + {file = "ujson-5.9.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d0fd2eba664a22447102062814bd13e63c6130540222c0aa620701dd01f4be81"}, + {file = "ujson-5.9.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:bdf7fc21a03bafe4ba208dafa84ae38e04e5d36c0e1c746726edf5392e9f9f36"}, + {file = "ujson-5.9.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e2f909bc08ce01f122fd9c24bc6f9876aa087188dfaf3c4116fe6e4daf7e194f"}, + {file = "ujson-5.9.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bd4ea86c2afd41429751d22a3ccd03311c067bd6aeee2d054f83f97e41e11d8f"}, + {file = "ujson-5.9.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:63fb2e6599d96fdffdb553af0ed3f76b85fda63281063f1cb5b1141a6fcd0617"}, + {file = "ujson-5.9.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:32bba5870c8fa2a97f4a68f6401038d3f1922e66c34280d710af00b14a3ca562"}, + {file = "ujson-5.9.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:37ef92e42535a81bf72179d0e252c9af42a4ed966dc6be6967ebfb929a87bc60"}, + {file = "ujson-5.9.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:f69f16b8f1c69da00e38dc5f2d08a86b0e781d0ad3e4cc6a13ea033a439c4844"}, + {file = "ujson-5.9.0-cp39-cp39-win32.whl", hash = "sha256:3382a3ce0ccc0558b1c1668950008cece9bf463ebb17463ebf6a8bfc060dae34"}, + {file = "ujson-5.9.0-cp39-cp39-win_amd64.whl", hash = "sha256:6adef377ed583477cf005b58c3025051b5faa6b8cc25876e594afbb772578f21"}, + {file = "ujson-5.9.0-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:ffdfebd819f492e48e4f31c97cb593b9c1a8251933d8f8972e81697f00326ff1"}, + {file = "ujson-5.9.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c4eec2ddc046360d087cf35659c7ba0cbd101f32035e19047013162274e71fcf"}, + {file = "ujson-5.9.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2fbb90aa5c23cb3d4b803c12aa220d26778c31b6e4b7a13a1f49971f6c7d088e"}, + {file = "ujson-5.9.0-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ba0823cb70866f0d6a4ad48d998dd338dce7314598721bc1b7986d054d782dfd"}, + {file = "ujson-5.9.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:4e35d7885ed612feb6b3dd1b7de28e89baaba4011ecdf995e88be9ac614765e9"}, + {file = "ujson-5.9.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:b048aa93eace8571eedbd67b3766623e7f0acbf08ee291bef7d8106210432427"}, + {file = "ujson-5.9.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:323279e68c195110ef85cbe5edce885219e3d4a48705448720ad925d88c9f851"}, + {file = "ujson-5.9.0-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9ac92d86ff34296f881e12aa955f7014d276895e0e4e868ba7fddebbde38e378"}, + {file = "ujson-5.9.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:6eecbd09b316cea1fd929b1e25f70382917542ab11b692cb46ec9b0a26c7427f"}, + {file = "ujson-5.9.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:473fb8dff1d58f49912323d7cb0859df5585cfc932e4b9c053bf8cf7f2d7c5c4"}, + {file = "ujson-5.9.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f91719c6abafe429c1a144cfe27883eace9fb1c09a9c5ef1bcb3ae80a3076a4e"}, + {file = "ujson-5.9.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7b1c0991c4fe256f5fdb19758f7eac7f47caac29a6c57d0de16a19048eb86bad"}, + {file = "ujson-5.9.0-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2a8ea0f55a1396708e564595aaa6696c0d8af532340f477162ff6927ecc46e21"}, + {file = "ujson-5.9.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:07e0cfdde5fd91f54cd2d7ffb3482c8ff1bf558abf32a8b953a5d169575ae1cd"}, + {file = "ujson-5.9.0.tar.gz", hash = "sha256:89cc92e73d5501b8a7f48575eeb14ad27156ad092c2e9fc7e3cf949f07e75532"}, +] + +[[package]] +name = "unidecode" +version = "1.3.7" +description = "ASCII transliterations of Unicode text" +optional = false +python-versions = ">=3.5" +files = [ + {file = "Unidecode-1.3.7-py3-none-any.whl", hash = "sha256:663a537f506834ed836af26a81b210d90cbde044c47bfbdc0fbbc9f94c86a6e4"}, + {file = "Unidecode-1.3.7.tar.gz", hash = "sha256:3c90b4662aa0de0cb591884b934ead8d2225f1800d8da675a7750cbc3bd94610"}, +] + +[[package]] +name = "urllib3" +version = "2.1.0" +description = "HTTP library with thread-safe connection pooling, file post, and more." +optional = false +python-versions = ">=3.8" +files = [ + {file = "urllib3-2.1.0-py3-none-any.whl", hash = "sha256:55901e917a5896a349ff771be919f8bd99aff50b79fe58fec595eb37bbc56bb3"}, + {file = "urllib3-2.1.0.tar.gz", hash = "sha256:df7aa8afb0148fa78488e7899b2c59b5f4ffcfa82e6c54ccb9dd37c1d7b52d54"}, +] + +[package.extras] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] +socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] +zstd = ["zstandard (>=0.18.0)"] + +[[package]] +name = "uvicorn" +version = "0.18.3" +description = "The lightning-fast ASGI server." +optional = false +python-versions = ">=3.7" +files = [ + {file = "uvicorn-0.18.3-py3-none-any.whl", hash = "sha256:0abd429ebb41e604ed8d2be6c60530de3408f250e8d2d84967d85ba9e86fe3af"}, + {file = "uvicorn-0.18.3.tar.gz", hash = "sha256:9a66e7c42a2a95222f76ec24a4b754c158261c4696e683b9dadc72b590e0311b"}, +] + +[package.dependencies] +click = ">=7.0" +h11 = ">=0.8" + +[package.extras] +standard = ["colorama (>=0.4)", "httptools (>=0.4.0)", "python-dotenv (>=0.13)", "pyyaml (>=5.1)", "uvloop (>=0.14.0,!=0.15.0,!=0.15.1)", "watchfiles (>=0.13)", "websockets (>=10.0)"] + +[[package]] +name = "validators" +version = "0.22.0" +description = "Python Data Validation for Humans™" +optional = false +python-versions = ">=3.8" +files = [ + {file = "validators-0.22.0-py3-none-any.whl", hash = "sha256:61cf7d4a62bbae559f2e54aed3b000cea9ff3e2fdbe463f51179b92c58c9585a"}, + {file = "validators-0.22.0.tar.gz", hash = "sha256:77b2689b172eeeb600d9605ab86194641670cdb73b60afd577142a9397873370"}, +] + +[package.extras] +docs-offline = ["myst-parser (>=2.0.0)", "pypandoc-binary (>=1.11)", "sphinx (>=7.1.1)"] +docs-online = ["mkdocs (>=1.5.2)", "mkdocs-git-revision-date-localized-plugin (>=1.2.0)", "mkdocs-material (>=9.2.6)", "mkdocstrings[python] (>=0.22.0)", "pyaml (>=23.7.0)"] +hooks = ["pre-commit (>=3.3.3)"] +package = ["build (>=1.0.0)", "twine (>=4.0.2)"] +runner = ["tox (>=4.11.1)"] +sast = ["bandit[toml] (>=1.7.5)"] +testing = ["pytest (>=7.4.0)"] +tooling = ["black (>=23.7.0)", "pyright (>=1.1.325)", "ruff (>=0.0.287)"] +tooling-extras = ["pyaml (>=23.7.0)", "pypandoc-binary (>=1.11)", "pytest (>=7.4.0)"] + +[[package]] +name = "wcwidth" +version = "0.2.12" +description = "Measures the displayed width of unicode strings in a terminal" +optional = false +python-versions = "*" +files = [ + {file = "wcwidth-0.2.12-py2.py3-none-any.whl", hash = "sha256:f26ec43d96c8cbfed76a5075dac87680124fa84e0855195a6184da9c187f133c"}, + {file = "wcwidth-0.2.12.tar.gz", hash = "sha256:f01c104efdf57971bcb756f054dd58ddec5204dd15fa31d6503ea57947d97c02"}, +] + +[[package]] +name = "xlrd" +version = "2.0.1" +description = "Library for developers to extract data from Microsoft Excel (tm) .xls spreadsheet files" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" +files = [ + {file = "xlrd-2.0.1-py2.py3-none-any.whl", hash = "sha256:6a33ee89877bd9abc1158129f6e94be74e2679636b8a205b43b85206c3f0bbdd"}, + {file = "xlrd-2.0.1.tar.gz", hash = "sha256:f72f148f54442c6b056bf931dbc34f986fd0c3b0b6b5a58d013c9aef274d0c88"}, +] + +[package.extras] +build = ["twine", "wheel"] +docs = ["sphinx"] +test = ["pytest", "pytest-cov"] + +[[package]] +name = "xlsxwriter" +version = "1.4.5" +description = "A Python module for creating Excel XLSX files." +optional = false +python-versions = "*" +files = [ + {file = "XlsxWriter-1.4.5-py2.py3-none-any.whl", hash = "sha256:f9335f1736e2c4fd80e940fe1b6d92d967bf454a1e5d639b0b7a4459ade790cc"}, + {file = "XlsxWriter-1.4.5.tar.gz", hash = "sha256:0956747859567ec01907e561a7d8413de18a7aae36860f979f9da52b9d58bc19"}, +] + +[[package]] +name = "xmltodict" +version = "0.13.0" +description = "Makes working with XML feel like you are working with JSON" +optional = false +python-versions = ">=3.4" +files = [ + {file = "xmltodict-0.13.0-py2.py3-none-any.whl", hash = "sha256:aa89e8fd76320154a40d19a0df04a4695fb9dc5ba977cbb68ab3e4eb225e7852"}, + {file = "xmltodict-0.13.0.tar.gz", hash = "sha256:341595a488e3e01a85a9d8911d8912fd922ede5fecc4dce437eb4b6c8d037e56"}, +] + +[[package]] +name = "yarl" +version = "1.9.4" +description = "Yet another URL library" +optional = false +python-versions = ">=3.7" +files = [ + {file = "yarl-1.9.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a8c1df72eb746f4136fe9a2e72b0c9dc1da1cbd23b5372f94b5820ff8ae30e0e"}, + {file = "yarl-1.9.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a3a6ed1d525bfb91b3fc9b690c5a21bb52de28c018530ad85093cc488bee2dd2"}, + {file = "yarl-1.9.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c38c9ddb6103ceae4e4498f9c08fac9b590c5c71b0370f98714768e22ac6fa66"}, + {file = "yarl-1.9.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d9e09c9d74f4566e905a0b8fa668c58109f7624db96a2171f21747abc7524234"}, + {file = "yarl-1.9.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b8477c1ee4bd47c57d49621a062121c3023609f7a13b8a46953eb6c9716ca392"}, + {file = "yarl-1.9.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d5ff2c858f5f6a42c2a8e751100f237c5e869cbde669a724f2062d4c4ef93551"}, + {file = "yarl-1.9.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:357495293086c5b6d34ca9616a43d329317feab7917518bc97a08f9e55648455"}, + {file = "yarl-1.9.4-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:54525ae423d7b7a8ee81ba189f131054defdb122cde31ff17477951464c1691c"}, + {file = "yarl-1.9.4-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:801e9264d19643548651b9db361ce3287176671fb0117f96b5ac0ee1c3530d53"}, + {file = "yarl-1.9.4-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e516dc8baf7b380e6c1c26792610230f37147bb754d6426462ab115a02944385"}, + {file = "yarl-1.9.4-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:7d5aaac37d19b2904bb9dfe12cdb08c8443e7ba7d2852894ad448d4b8f442863"}, + {file = "yarl-1.9.4-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:54beabb809ffcacbd9d28ac57b0db46e42a6e341a030293fb3185c409e626b8b"}, + {file = "yarl-1.9.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:bac8d525a8dbc2a1507ec731d2867025d11ceadcb4dd421423a5d42c56818541"}, + {file = "yarl-1.9.4-cp310-cp310-win32.whl", hash = "sha256:7855426dfbddac81896b6e533ebefc0af2f132d4a47340cee6d22cac7190022d"}, + {file = "yarl-1.9.4-cp310-cp310-win_amd64.whl", hash = "sha256:848cd2a1df56ddbffeb375535fb62c9d1645dde33ca4d51341378b3f5954429b"}, + {file = "yarl-1.9.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:35a2b9396879ce32754bd457d31a51ff0a9d426fd9e0e3c33394bf4b9036b099"}, + {file = "yarl-1.9.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4c7d56b293cc071e82532f70adcbd8b61909eec973ae9d2d1f9b233f3d943f2c"}, + {file = "yarl-1.9.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:d8a1c6c0be645c745a081c192e747c5de06e944a0d21245f4cf7c05e457c36e0"}, + {file = "yarl-1.9.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4b3c1ffe10069f655ea2d731808e76e0f452fc6c749bea04781daf18e6039525"}, + {file = "yarl-1.9.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:549d19c84c55d11687ddbd47eeb348a89df9cb30e1993f1b128f4685cd0ebbf8"}, + {file = "yarl-1.9.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a7409f968456111140c1c95301cadf071bd30a81cbd7ab829169fb9e3d72eae9"}, + {file = "yarl-1.9.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e23a6d84d9d1738dbc6e38167776107e63307dfc8ad108e580548d1f2c587f42"}, + {file = "yarl-1.9.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d8b889777de69897406c9fb0b76cdf2fd0f31267861ae7501d93003d55f54fbe"}, + {file = "yarl-1.9.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:03caa9507d3d3c83bca08650678e25364e1843b484f19986a527630ca376ecce"}, + {file = "yarl-1.9.4-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:4e9035df8d0880b2f1c7f5031f33f69e071dfe72ee9310cfc76f7b605958ceb9"}, + {file = "yarl-1.9.4-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:c0ec0ed476f77db9fb29bca17f0a8fcc7bc97ad4c6c1d8959c507decb22e8572"}, + {file = "yarl-1.9.4-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:ee04010f26d5102399bd17f8df8bc38dc7ccd7701dc77f4a68c5b8d733406958"}, + {file = "yarl-1.9.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:49a180c2e0743d5d6e0b4d1a9e5f633c62eca3f8a86ba5dd3c471060e352ca98"}, + {file = "yarl-1.9.4-cp311-cp311-win32.whl", hash = "sha256:81eb57278deb6098a5b62e88ad8281b2ba09f2f1147c4767522353eaa6260b31"}, + {file = "yarl-1.9.4-cp311-cp311-win_amd64.whl", hash = "sha256:d1d2532b340b692880261c15aee4dc94dd22ca5d61b9db9a8a361953d36410b1"}, + {file = "yarl-1.9.4-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0d2454f0aef65ea81037759be5ca9947539667eecebca092733b2eb43c965a81"}, + {file = "yarl-1.9.4-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:44d8ffbb9c06e5a7f529f38f53eda23e50d1ed33c6c869e01481d3fafa6b8142"}, + {file = "yarl-1.9.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:aaaea1e536f98754a6e5c56091baa1b6ce2f2700cc4a00b0d49eca8dea471074"}, + {file = "yarl-1.9.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3777ce5536d17989c91696db1d459574e9a9bd37660ea7ee4d3344579bb6f129"}, + {file = "yarl-1.9.4-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9fc5fc1eeb029757349ad26bbc5880557389a03fa6ada41703db5e068881e5f2"}, + {file = "yarl-1.9.4-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ea65804b5dc88dacd4a40279af0cdadcfe74b3e5b4c897aa0d81cf86927fee78"}, + {file = "yarl-1.9.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aa102d6d280a5455ad6a0f9e6d769989638718e938a6a0a2ff3f4a7ff8c62cc4"}, + {file = "yarl-1.9.4-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:09efe4615ada057ba2d30df871d2f668af661e971dfeedf0c159927d48bbeff0"}, + {file = "yarl-1.9.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:008d3e808d03ef28542372d01057fd09168419cdc8f848efe2804f894ae03e51"}, + {file = "yarl-1.9.4-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:6f5cb257bc2ec58f437da2b37a8cd48f666db96d47b8a3115c29f316313654ff"}, + {file = "yarl-1.9.4-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:992f18e0ea248ee03b5a6e8b3b4738850ae7dbb172cc41c966462801cbf62cf7"}, + {file = "yarl-1.9.4-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:0e9d124c191d5b881060a9e5060627694c3bdd1fe24c5eecc8d5d7d0eb6faabc"}, + {file = "yarl-1.9.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:3986b6f41ad22988e53d5778f91855dc0399b043fc8946d4f2e68af22ee9ff10"}, + {file = "yarl-1.9.4-cp312-cp312-win32.whl", hash = "sha256:4b21516d181cd77ebd06ce160ef8cc2a5e9ad35fb1c5930882baff5ac865eee7"}, + {file = "yarl-1.9.4-cp312-cp312-win_amd64.whl", hash = "sha256:a9bd00dc3bc395a662900f33f74feb3e757429e545d831eef5bb280252631984"}, + {file = "yarl-1.9.4-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:63b20738b5aac74e239622d2fe30df4fca4942a86e31bf47a81a0e94c14df94f"}, + {file = "yarl-1.9.4-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d7d7f7de27b8944f1fee2c26a88b4dabc2409d2fea7a9ed3df79b67277644e17"}, + {file = "yarl-1.9.4-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c74018551e31269d56fab81a728f683667e7c28c04e807ba08f8c9e3bba32f14"}, + {file = "yarl-1.9.4-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ca06675212f94e7a610e85ca36948bb8fc023e458dd6c63ef71abfd482481aa5"}, + {file = "yarl-1.9.4-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5aef935237d60a51a62b86249839b51345f47564208c6ee615ed2a40878dccdd"}, + {file = "yarl-1.9.4-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2b134fd795e2322b7684155b7855cc99409d10b2e408056db2b93b51a52accc7"}, + {file = "yarl-1.9.4-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d25039a474c4c72a5ad4b52495056f843a7ff07b632c1b92ea9043a3d9950f6e"}, + {file = "yarl-1.9.4-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:f7d6b36dd2e029b6bcb8a13cf19664c7b8e19ab3a58e0fefbb5b8461447ed5ec"}, + {file = "yarl-1.9.4-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:957b4774373cf6f709359e5c8c4a0af9f6d7875db657adb0feaf8d6cb3c3964c"}, + {file = "yarl-1.9.4-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:d7eeb6d22331e2fd42fce928a81c697c9ee2d51400bd1a28803965883e13cead"}, + {file = "yarl-1.9.4-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:6a962e04b8f91f8c4e5917e518d17958e3bdee71fd1d8b88cdce74dd0ebbf434"}, + {file = "yarl-1.9.4-cp37-cp37m-win32.whl", hash = "sha256:f3bc6af6e2b8f92eced34ef6a96ffb248e863af20ef4fde9448cc8c9b858b749"}, + {file = "yarl-1.9.4-cp37-cp37m-win_amd64.whl", hash = "sha256:ad4d7a90a92e528aadf4965d685c17dacff3df282db1121136c382dc0b6014d2"}, + {file = "yarl-1.9.4-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:ec61d826d80fc293ed46c9dd26995921e3a82146feacd952ef0757236fc137be"}, + {file = "yarl-1.9.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8be9e837ea9113676e5754b43b940b50cce76d9ed7d2461df1af39a8ee674d9f"}, + {file = "yarl-1.9.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:bef596fdaa8f26e3d66af846bbe77057237cb6e8efff8cd7cc8dff9a62278bbf"}, + {file = "yarl-1.9.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2d47552b6e52c3319fede1b60b3de120fe83bde9b7bddad11a69fb0af7db32f1"}, + {file = "yarl-1.9.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:84fc30f71689d7fc9168b92788abc977dc8cefa806909565fc2951d02f6b7d57"}, + {file = "yarl-1.9.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4aa9741085f635934f3a2583e16fcf62ba835719a8b2b28fb2917bb0537c1dfa"}, + {file = "yarl-1.9.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:206a55215e6d05dbc6c98ce598a59e6fbd0c493e2de4ea6cc2f4934d5a18d130"}, + {file = "yarl-1.9.4-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:07574b007ee20e5c375a8fe4a0789fad26db905f9813be0f9fef5a68080de559"}, + {file = "yarl-1.9.4-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:5a2e2433eb9344a163aced6a5f6c9222c0786e5a9e9cac2c89f0b28433f56e23"}, + {file = "yarl-1.9.4-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:6ad6d10ed9b67a382b45f29ea028f92d25bc0bc1daf6c5b801b90b5aa70fb9ec"}, + {file = "yarl-1.9.4-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:6fe79f998a4052d79e1c30eeb7d6c1c1056ad33300f682465e1b4e9b5a188b78"}, + {file = "yarl-1.9.4-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:a825ec844298c791fd28ed14ed1bffc56a98d15b8c58a20e0e08c1f5f2bea1be"}, + {file = "yarl-1.9.4-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8619d6915b3b0b34420cf9b2bb6d81ef59d984cb0fde7544e9ece32b4b3043c3"}, + {file = "yarl-1.9.4-cp38-cp38-win32.whl", hash = "sha256:686a0c2f85f83463272ddffd4deb5e591c98aac1897d65e92319f729c320eece"}, + {file = "yarl-1.9.4-cp38-cp38-win_amd64.whl", hash = "sha256:a00862fb23195b6b8322f7d781b0dc1d82cb3bcac346d1e38689370cc1cc398b"}, + {file = "yarl-1.9.4-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:604f31d97fa493083ea21bd9b92c419012531c4e17ea6da0f65cacdcf5d0bd27"}, + {file = "yarl-1.9.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8a854227cf581330ffa2c4824d96e52ee621dd571078a252c25e3a3b3d94a1b1"}, + {file = "yarl-1.9.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ba6f52cbc7809cd8d74604cce9c14868306ae4aa0282016b641c661f981a6e91"}, + {file = "yarl-1.9.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a6327976c7c2f4ee6816eff196e25385ccc02cb81427952414a64811037bbc8b"}, + {file = "yarl-1.9.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8397a3817d7dcdd14bb266283cd1d6fc7264a48c186b986f32e86d86d35fbac5"}, + {file = "yarl-1.9.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e0381b4ce23ff92f8170080c97678040fc5b08da85e9e292292aba67fdac6c34"}, + {file = "yarl-1.9.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:23d32a2594cb5d565d358a92e151315d1b2268bc10f4610d098f96b147370136"}, + {file = "yarl-1.9.4-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ddb2a5c08a4eaaba605340fdee8fc08e406c56617566d9643ad8bf6852778fc7"}, + {file = "yarl-1.9.4-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:26a1dc6285e03f3cc9e839a2da83bcbf31dcb0d004c72d0730e755b33466c30e"}, + {file = "yarl-1.9.4-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:18580f672e44ce1238b82f7fb87d727c4a131f3a9d33a5e0e82b793362bf18b4"}, + {file = "yarl-1.9.4-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:29e0f83f37610f173eb7e7b5562dd71467993495e568e708d99e9d1944f561ec"}, + {file = "yarl-1.9.4-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:1f23e4fe1e8794f74b6027d7cf19dc25f8b63af1483d91d595d4a07eca1fb26c"}, + {file = "yarl-1.9.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:db8e58b9d79200c76956cefd14d5c90af54416ff5353c5bfd7cbe58818e26ef0"}, + {file = "yarl-1.9.4-cp39-cp39-win32.whl", hash = "sha256:c7224cab95645c7ab53791022ae77a4509472613e839dab722a72abe5a684575"}, + {file = "yarl-1.9.4-cp39-cp39-win_amd64.whl", hash = "sha256:824d6c50492add5da9374875ce72db7a0733b29c2394890aef23d533106e2b15"}, + {file = "yarl-1.9.4-py3-none-any.whl", hash = "sha256:928cecb0ef9d5a7946eb6ff58417ad2fe9375762382f1bf5c55e61645f2c43ad"}, + {file = "yarl-1.9.4.tar.gz", hash = "sha256:566db86717cf8080b99b58b083b773a908ae40f06681e87e589a976faf8246bf"}, +] + +[package.dependencies] +idna = ">=2.0" +multidict = ">=4.0" + +[[package]] +name = "zipp" +version = "3.17.0" +description = "Backport of pathlib-compatible object wrapper for zip files" +optional = false +python-versions = ">=3.8" +files = [ + {file = "zipp-3.17.0-py3-none-any.whl", hash = "sha256:0e923e726174922dce09c53c59ad483ff7bbb8e572e00c7f7c46b88556409f31"}, + {file = "zipp-3.17.0.tar.gz", hash = "sha256:84e64a1c28cf7e91ed2078bb8cc8c259cb19b76942096c8d7b84947690cabaf0"}, +] + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"] +testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy (>=0.9.1)", "pytest-ruff"] + +[metadata] +lock-version = "2.0" +python-versions = "^3.8" +content-hash = "ab908b01bc3c36404fc161c761d3fc89c2cef5166468f99f1a81fb84996c97ec" diff --git a/pyproject.toml b/pyproject.toml index d9284001f..b85c6e9a7 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -116,7 +116,7 @@ pytest-mock = "^3.9.0" responses = "^0.21.0" snoop = "^0.4.2" # https://github.com/kvesteri/sqlalchemy-utils/issues/472 -sqlalchemy-utils = "~0.37" +sqlalchemy-utils = "~0.38.1" python-dotenv = "^0.21.0" # Starlette diff --git a/spinta/api/__init__.py b/spinta/api/__init__.py index 64d0c1caf..cb42e79eb 100644 --- a/spinta/api/__init__.py +++ b/spinta/api/__init__.py @@ -257,15 +257,21 @@ async def homepage(request: Request): config = context.get('config') UrlParams: Type[components.UrlParams] UrlParams = config.components['urlparams']['component'] - params: UrlParams = prepare(context, UrlParams(), Version(), request) + store = context.get('store') + + # Currently need to initialize the manifest and then add missing models + # otherwise, manifest never gets created and becomes infinite loop + manifest = commands.create_request_manifest(context, store.manifest) + context.set('request.manifest', manifest) + commands.load_for_request(context, manifest) + params: UrlParams = prepare(context, UrlParams(), Version(), request) context.attach('accesslog', create_accesslog, context, loaders=( - context.get('store'), + store, context.get("auth.token"), request, params, )) - return await create_http_response(context, params, request) diff --git a/spinta/api/inspect.py b/spinta/api/inspect.py index 15633742e..4406a82ee 100644 --- a/spinta/api/inspect.py +++ b/spinta/api/inspect.py @@ -170,7 +170,7 @@ async def inspect_api(context: Context, request: Request, params: UrlParams): only_url=True ) inspect_data.clean_up() - clean_up_source_for_return(manifest) + clean_up_source_for_return(context, manifest) return commands.render( context, @@ -183,8 +183,8 @@ async def inspect_api(context: Context, request: Request, params: UrlParams): raise e -def clean_up_source_for_return(manifest: Manifest): - for dataset in manifest.datasets.values(): +def clean_up_source_for_return(context: Context, manifest: Manifest): + for dataset in commands.get_datasets(context, manifest).values(): for resource in dataset.resources.values(): if resource.external and not ("http://" in resource.external or "https://" in resource.external): resource.external = f"https://get.data.gov.lt/{dataset.name}" diff --git a/spinta/backends/__init__.py b/spinta/backends/__init__.py index 6c53994c9..96de43972 100644 --- a/spinta/backends/__init__.py +++ b/spinta/backends/__init__.py @@ -612,7 +612,7 @@ def prepare_data_for_response( select=sel, ) for prop, val, sel in select_model_props( - ns.manifest.models['_ns'], + commands.get_model(context, ns.manifest, '_ns'), prop_names, value, select, diff --git a/spinta/backends/memory/commands/init.py b/spinta/backends/memory/commands/init.py index 01341a85d..c985f8b3b 100644 --- a/spinta/backends/memory/commands/init.py +++ b/spinta/backends/memory/commands/init.py @@ -8,6 +8,6 @@ @commands.prepare.register(Context, Memory, Manifest) def prepare(context: Context, backend: Memory, manifest: Manifest): - for model in manifest.models.values(): + for model in commands.get_models(context, manifest).values(): backend.create(get_table_name(model)) backend.create(get_table_name(model, TableType.CHANGELOG)) diff --git a/spinta/backends/postgresql/commands/init.py b/spinta/backends/postgresql/commands/init.py index 4411f0daf..c6ad3978d 100644 --- a/spinta/backends/postgresql/commands/init.py +++ b/spinta/backends/postgresql/commands/init.py @@ -22,7 +22,7 @@ @commands.prepare.register(Context, PostgreSQL, Manifest) def prepare(context: Context, backend: PostgreSQL, manifest: Manifest): # Prepare backend for models. - for model in manifest.models.values(): + for model in commands.get_models(context, manifest).values(): if model.backend and model.backend.name == backend.name: commands.prepare(context, backend, model) diff --git a/spinta/backends/postgresql/commands/manifest.py b/spinta/backends/postgresql/commands/manifest.py index fcf0691bd..aa68be140 100644 --- a/spinta/backends/postgresql/commands/manifest.py +++ b/spinta/backends/postgresql/commands/manifest.py @@ -49,7 +49,7 @@ def load( ) target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') commands.load(context, store.internal, into=target) diff --git a/spinta/backends/postgresql/commands/migrate.py b/spinta/backends/postgresql/commands/migrate.py index ab47600a8..89aa21b7d 100644 --- a/spinta/backends/postgresql/commands/migrate.py +++ b/spinta/backends/postgresql/commands/migrate.py @@ -48,12 +48,13 @@ def migrate(context: Context, manifest: Manifest, backend: PostgreSQL, migrate_m metadata.reflect() tables = [] + models = commands.get_models(context, manifest) for table in table_names: name = migrate_meta.rename.get_table_name(table) - if name not in manifest.models.keys(): + if name not in models.keys(): name = table tables.append(name) - sorted_models = sort_models_by_ref_and_base(list(manifest.models.values())) + sorted_models = sort_models_by_ref_and_base(list(models.values())) sorted_model_names = list([model.name for model in sorted_models]) # Do reversed zip, to ensure that sorted models get selected first models = zipitems( @@ -71,7 +72,7 @@ def migrate(context: Context, manifest: Manifest, backend: PostgreSQL, migrate_m old = NA if old_model: old = metadata.tables[migrate_meta.rename.get_old_table_name(old_model)] - new = manifest.models.get(new_model) if new_model else new_model + new = commands.get_model(context, manifest, new_model) if new_model else new_model commands.migrate(context, backend, inspector, old, new, handler, migrate_meta.rename) _handle_foreign_key_constraints(inspector, sorted_models, handler, migrate_meta.rename) _clean_up_file_type(inspector, sorted_models, handler, migrate_meta.rename) diff --git a/spinta/backends/postgresql/sqlalchemy.py b/spinta/backends/postgresql/sqlalchemy.py index fc2e6bf39..4e6f1f89c 100644 --- a/spinta/backends/postgresql/sqlalchemy.py +++ b/spinta/backends/postgresql/sqlalchemy.py @@ -10,6 +10,7 @@ class utcnow(FunctionElement): + inherit_cache = True type = sa.DateTime() diff --git a/spinta/cli/config.py b/spinta/cli/config.py index dd420e8d9..8c21904b3 100644 --- a/spinta/cli/config.py +++ b/spinta/cli/config.py @@ -35,5 +35,5 @@ def check( """Check configuration and manifests""" manifests = convert_str_to_manifest_path(manifests) context = configure_context(ctx.obj, manifests, mode=mode, check_names=check_names) - prepare_manifest(context, ensure_config_dir=True) + prepare_manifest(context, ensure_config_dir=True, full_load=True) echo("OK") diff --git a/spinta/cli/data.py b/spinta/cli/data.py index 9434d04bc..182fe1164 100644 --- a/spinta/cli/data.py +++ b/spinta/cli/data.py @@ -6,6 +6,7 @@ from typer import Context as TyperContext from typer import Option +from spinta import commands from spinta.cli.helpers.auth import require_auth from spinta.cli.helpers.store import prepare_manifest from spinta.cli.helpers.data import process_stream @@ -26,7 +27,7 @@ def import_( context = ctx.obj store = prepare_manifest(context) manifest = store.manifest - root = manifest.objects['ns'][''] + root = commands.get_namespace(context, manifest, '') with context: require_auth(context, auth) diff --git a/spinta/cli/helpers/store.py b/spinta/cli/helpers/store.py index e6c34920d..f7b83ed07 100644 --- a/spinta/cli/helpers/store.py +++ b/spinta/cli/helpers/store.py @@ -97,6 +97,7 @@ def load_manifest( ensure_config_dir: bool = False, rename_duplicates: bool = False, load_internal: bool = True, + full_load: bool = False ) -> Store: if store is None: store = load_store( @@ -120,6 +121,7 @@ def load_manifest( context, store.manifest, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) commands.link(context, store.manifest) commands.check(context, store.manifest) @@ -131,11 +133,13 @@ def prepare_manifest( *, verbose: bool = True, ensure_config_dir: bool = False, + full_load: bool = False ) -> Store: store = load_manifest( context, verbose=verbose, ensure_config_dir=ensure_config_dir, + full_load=full_load ) commands.wait(context, store) commands.prepare(context, store.manifest) diff --git a/spinta/cli/init.py b/spinta/cli/init.py index 9bf78dc15..595a586c0 100644 --- a/spinta/cli/init.py +++ b/spinta/cli/init.py @@ -2,10 +2,13 @@ from typer import Argument +from spinta.components import Context from spinta.manifests.tabular.helpers import write_tabular_manifest +from typer import Context as TyperContext def init( + ctx: TyperContext, manifest: Optional[str] = Argument(None, help="path to a manifest"), ): """Initialize a new manifest table @@ -14,5 +17,6 @@ def init( Depending of file extensions, a CSV of a XLSX format manifest will be created. """ - write_tabular_manifest(manifest) + context: Context = ctx.obj + write_tabular_manifest(context, manifest) diff --git a/spinta/cli/inspect.py b/spinta/cli/inspect.py index 75343ff8a..79df6e2f6 100644 --- a/spinta/cli/inspect.py +++ b/spinta/cli/inspect.py @@ -74,4 +74,4 @@ def inspect( fmt = config.exporters[output_type] commands.render(ctx.obj, manifest, fmt, path=output) else: - echo(render_tabular_manifest(manifest)) + echo(render_tabular_manifest(context, manifest)) diff --git a/spinta/cli/manifest.py b/spinta/cli/manifest.py index 2e1e91dd7..649e0da98 100644 --- a/spinta/cli/manifest.py +++ b/spinta/cli/manifest.py @@ -8,12 +8,12 @@ from typer import Typer from typer import echo -from spinta.cli.helpers.manifest import convert_str_to_manifest_path from spinta.cli.helpers.store import load_manifest from spinta.components import Context from spinta.core.context import configure_context from spinta.core.enums import Access -from spinta.manifests.components import ManifestPath +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.manifests.internal_sql.helpers import write_internal_sql_manifest from spinta.manifests.tabular.components import ManifestColumn from spinta.manifests.tabular.components import ManifestRow from spinta.manifests.tabular.helpers import datasets_to_tabular @@ -58,35 +58,106 @@ def copy( ): """Copy models from CSV manifest files into another CSV manifest file""" context: Context = ctx.obj + copy_manifest( + context, + source=source, + access=access, + format_names=format_names, + output=output, + columns=columns, + order_by=order_by, + rename_duplicates=rename_duplicates, + manifests=manifests + ) + + +def copy_manifest( + context: Context, + source: bool = True, + access: str = 'private', + format_names: bool = False, + output: Optional[str] = None, + columns: Optional[str] = None, + order_by: Optional[str] = None, + rename_duplicates: bool = False, + manifests: List[str] = None, + output_type: Optional[str] = None +): + """Copy models from CSV manifest files into another CSV manifest file""" access = get_enum_by_name(Access, access) cols = normalizes_columns(columns.split(',')) if columns else None - + internal = False verbose = True if not output: verbose = False + else: + if output_type: + if output_type == 'internal_sql': + internal = True + else: + internal = InternalSQLManifest.detect_from_path(output) - manifests = convert_str_to_manifest_path(manifests) + if internal: + rows = _read_and_return_manifest( + context, + manifests, + external=source, + access=access, + format_names=format_names, + order_by=order_by, + rename_duplicates=rename_duplicates, + verbose=verbose, + ) + else: + rows = _read_and_return_rows( + context, + manifests, + external=source, + access=access, + format_names=format_names, + order_by=order_by, + rename_duplicates=rename_duplicates, + verbose=verbose, + ) - rows = _read_csv_files( + if output: + if internal: + write_internal_sql_manifest(context, output, rows) + else: + write_tabular_manifest(context, output, rows) + else: + echo(render_tabular_manifest_rows(rows, cols)) + + +def _read_and_return_manifest( + context: Context, + manifests: List[str], + *, + external: bool = True, + access: Access = Access.private, + format_names: bool = False, + order_by: ManifestColumn = None, + rename_duplicates: bool = False, + verbose: bool = True, +) -> Iterator[ManifestRow]: + context = configure_context(context, manifests) + store = load_manifest( context, - manifests, - external=source, - access=access, - format_names=format_names, - order_by=order_by, rename_duplicates=rename_duplicates, + load_internal=False, verbose=verbose, + full_load=True ) - if output: - write_tabular_manifest(output, rows) - else: - echo(render_tabular_manifest_rows(rows, cols)) + if format_names: + reformat_names(context, store.manifest) + return store.manifest -def _read_csv_files( + +def _read_and_return_rows( context: Context, - manifests: List[ManifestPath], + manifests: List[str], *, external: bool = True, access: Access = Access.private, @@ -101,12 +172,14 @@ def _read_csv_files( rename_duplicates=rename_duplicates, load_internal=False, verbose=verbose, + full_load=True ) if format_names: reformat_names(context, store.manifest) yield from datasets_to_tabular( + context, store.manifest, external=external, access=access, diff --git a/spinta/cli/migrate.py b/spinta/cli/migrate.py index a9dd9b88a..b986c6fdd 100644 --- a/spinta/cli/migrate.py +++ b/spinta/cli/migrate.py @@ -14,6 +14,7 @@ from spinta.cli.helpers.manifest import convert_str_to_manifest_path from spinta.cli.helpers.store import load_store from spinta.cli.helpers.store import prepare_manifest +from spinta.components import Context from spinta.core.context import configure_context from spinta.exceptions import FileNotFound, ModelNotFound, PropertyNotFound from spinta.manifests.components import Manifest @@ -31,7 +32,7 @@ def bootstrap( """ manifests = convert_str_to_manifest_path(manifests) context = configure_context(ctx.obj, manifests) - store = prepare_manifest(context, ensure_config_dir=True) + store = prepare_manifest(context, ensure_config_dir=True, full_load=True) with context: require_auth(context) @@ -83,7 +84,7 @@ def migrate( path=rename ) ) - _validate_migrate_rename(migrate_meta.rename, manifest) + _validate_migrate_rename(context, migrate_meta.rename, manifest) if backend: context.attach(f'transaction.{backend.name}', backend.begin) @@ -188,12 +189,13 @@ def __init__(self, plan: bool, autocommit: bool, rename: MigrateRename): self.autocommit = autocommit -def _validate_migrate_rename(rename: MigrateRename, manifest: Manifest): +def _validate_migrate_rename(context: Context, rename: MigrateRename, manifest: Manifest): tables = rename.tables.values() for table in tables: - if table["new_name"] not in manifest.models.keys(): + models = commands.get_models(context, manifest) + if table["new_name"] not in models.keys(): raise ModelNotFound(model=table["new_name"]) - model = manifest.models[table["new_name"]] + model = commands.get_model(context, manifest, table["new_name"]) for column in table["columns"].values(): if column not in model.properties.keys(): raise PropertyNotFound(property=column) diff --git a/spinta/cli/pii.py b/spinta/cli/pii.py index 2a0601b1b..b97787aa2 100644 --- a/spinta/cli/pii.py +++ b/spinta/cli/pii.py @@ -16,6 +16,7 @@ from typer import Typer from typer import echo +from spinta import commands from spinta.cli.helpers.auth import require_auth from spinta.cli.helpers.data import ModelRow from spinta.cli.helpers.data import count_rows @@ -133,7 +134,7 @@ def _detect_nin_lt(value: Any): return is_nin_lt(str(value)) -def _detect_pii(manifest: Manifest, rows: Iterable[ModelRow]) -> None: +def _detect_pii(context: Context, manifest: Manifest, rows: Iterable[ModelRow]) -> None: """Detects PII and modifies given manifest in place""" detectors = [ @@ -160,7 +161,7 @@ def _detect_pii(manifest: Manifest, rows: Iterable[ModelRow]) -> None: # Update manifest. for model_name, props in result.items(): - model = manifest.models[model_name] + model = commands.get_model(context, manifest, model_name) for prop_place, matches in props.items(): prop = model.flatprops[prop_place] for uri, match in matches.items(): @@ -216,7 +217,7 @@ def detect( context.set('rc', rc.fork(config)) # Load manifest - store = prepare_manifest(context, verbose=False) + store = prepare_manifest(context, verbose=False, full_load=True) manifest = store.manifest with context: require_auth(context, auth) @@ -228,7 +229,7 @@ def detect( for backend in manifest.backends.values(): backends.add(backend.name) context.attach(f'transaction.{backend.name}', backend.begin) - for dataset in manifest.datasets.values(): + for dataset in commands.get_datasets(context, manifest).values(): for resource in dataset.resources.values(): if resource.backend and resource.backend.name not in backends: backends.add(resource.backend.name) @@ -236,10 +237,8 @@ def detect( for keymap in store.keymaps.values(): context.attach(f'keymap.{keymap.name}', lambda: keymap) - from spinta.types.namespace import traverse_ns_models - - ns = manifest.objects['ns'][''] - models = traverse_ns_models(context, ns, Action.SEARCH) + ns = commands.get_namespace(context, manifest, '') + models = commands.traverse_ns_models(context, ns, manifest, Action.SEARCH) models = sort_models_by_refs(models) models = list(reversed(list(models))) counts = count_rows(context, models, limit=limit) @@ -251,8 +250,8 @@ def detect( ) total = sum(counts.values()) rows = tqdm.tqdm(rows, 'PII DETECT', ascii=True, total=total) - _detect_pii(manifest, rows) + _detect_pii(context, manifest, rows) if output: - write_tabular_manifest(output, manifest) + write_tabular_manifest(context, output, manifest) else: - echo(render_tabular_manifest(manifest)) + echo(render_tabular_manifest(context, manifest)) diff --git a/spinta/cli/pull.py b/spinta/cli/pull.py index b88bf3bba..361b3a375 100644 --- a/spinta/cli/pull.py +++ b/spinta/cli/pull.py @@ -20,8 +20,8 @@ from spinta.manifests.components import Manifest -def _get_dataset_models(manifest: Manifest, dataset: Dataset): - for model in manifest.models.values(): +def _get_dataset_models(context: Context, manifest: Manifest, dataset: Dataset): + for model in commands.get_models(context, manifest).values(): if model.external and model.external.dataset and model.external.dataset.name == dataset.name: yield model @@ -50,8 +50,8 @@ def pull( context = ctx.obj store = prepare_manifest(context) manifest = store.manifest - if dataset in manifest.objects['dataset']: - dataset = manifest.objects['dataset'][dataset] + if commands.has_namespace(context, manifest, dataset): + dataset = commands.get_dataset(context, manifest, dataset) else: echo(str(exceptions.NodeNotFound(manifest, type='dataset', name=dataset))) raise Exit(code=1) @@ -59,12 +59,12 @@ def pull( if model: models = [] for model in model: - if model not in manifest.models: + if not commands.has_model(context, manifest, model): echo(str(exceptions.NodeNotFound(manifest, type='model', name=model))) raise Exit(code=1) - models.append(manifest.models[model]) + models.append(commands.get_model(context, manifest, model)) else: - models = _get_dataset_models(manifest, dataset) + models = _get_dataset_models(context, manifest, dataset) try: with context: @@ -77,7 +77,7 @@ def pull( stream = _pull_models(context, models) if push: - root = manifest.objects['ns'][''] + root = commands.get_namespace(context, manifest, '') stream = write(context, root, stream, changed=True) if export is None and push is False: diff --git a/spinta/cli/push.py b/spinta/cli/push.py index f51c4fd3e..cf2169b03 100644 --- a/spinta/cli/push.py +++ b/spinta/cli/push.py @@ -32,7 +32,7 @@ from typer import Exit from typer import echo -from spinta import exceptions +from spinta import exceptions, commands from spinta import spyna from spinta.auth import authorized from spinta.auth import get_client_id_from_name, get_clients_path @@ -148,7 +148,7 @@ def push( manifests = convert_str_to_manifest_path(manifests) context = configure_context(ctx.obj, manifests, mode=mode) - store = prepare_manifest(context) + store = prepare_manifest(context, full_load=True) config: Config = context.get('config') if credentials: @@ -168,11 +168,11 @@ def push( state = f'sqlite:///{state}' manifest = store.manifest - if dataset and dataset not in manifest.datasets: + if dataset and not commands.has_dataset(context, manifest, dataset): echo(str(exceptions.NodeNotFound(manifest, type='dataset', name=dataset))) raise Exit(code=1) - ns = manifest.namespaces[''] + ns = commands.get_namespace(context, manifest, '') echo(f"Get access token from {creds.server}") token = get_access_token(creds) @@ -190,9 +190,7 @@ def push( _attach_keymaps(context, store) error_counter = ErrorCounter(max_count=max_error_count) - from spinta.types.namespace import traverse_ns_models - - models = traverse_ns_models(context, ns, Action.SEARCH, dataset, source_check=True) + models = commands.traverse_ns_models(context, ns, manifest, Action.SEARCH, dataset_=dataset, source_check=True) models = sort_models_by_ref_and_base(list(models)) if state: @@ -479,7 +477,7 @@ def _attach_backends(context: Context, store: Store, manifest: Manifest) -> None for backend in manifest.backends.values(): backends.add(backend.name) context.attach(f'transaction.{backend.name}', backend.begin) - for dataset_ in manifest.datasets.values(): + for dataset_ in commands.get_datasets(context, manifest).values(): for resource in dataset_.resources.values(): if resource.backend and resource.backend.name not in backends: backends.add(resource.backend.name) diff --git a/spinta/cli/show.py b/spinta/cli/show.py index 7af532285..1898f2389 100644 --- a/spinta/cli/show.py +++ b/spinta/cli/show.py @@ -23,7 +23,7 @@ def show( """Show manifest as ascii table""" manifests = convert_str_to_manifest_path(manifests) context = configure_context(ctx.obj, manifests, mode=mode) - store = prepare_manifest(context, verbose=False) + store = prepare_manifest(context, verbose=False, full_load=True) manifest = store.manifest - echo(render_tabular_manifest(manifest)) + echo(render_tabular_manifest(context, manifest)) diff --git a/spinta/commands/__init__.py b/spinta/commands/__init__.py index f82ee9789..458247ab8 100644 --- a/spinta/commands/__init__.py +++ b/spinta/commands/__init__.py @@ -97,6 +97,7 @@ def load( rename_duplicates: bool = False, # If True, load internal manifest, if not loaded. load_internal: bool = True, + full_load: bool = False ) -> None: """Load primitive data structures to python-native objects. @@ -128,6 +129,38 @@ def load( """ +@command() +def load_for_request(): + pass + + +@command() +def fully_initialize_manifest( + context: Context, + manifest: Manifest +): + pass + + +@command() +def create_request_manifest( + context: Context, + manifest: Manifest +): + pass + + +@command() +def traverse_ns_models( + context: Context, + ns: Namespace, + manifest: Manifest, + action: Action, + **kwargs +): + pass + + @command() def decode(): """Decode given value from source backend format into target backend format. @@ -623,6 +656,19 @@ def getall( pass +@overload +def getall( + context: Context, + ns: Namespace, + request: Request, + manifest: Manifest, + *, + action: Action, + params: UrlParams, +) -> Response: + pass + + @overload def getall( context: Context, @@ -1155,3 +1201,95 @@ def get_column( **kwargs ): pass + + +@command() +def has_node_type(context: Context, manifest: Manifest, obj_type: str, **kwargs) -> bool: + """Check if manifest has specified node type""" + + +@command() +def has_node(context: Context, manifest: Manifest, obj_type: str, obj: str, **kwargs) -> bool: + """Check if manifest has specified node""" + + +@command() +def get_node(context: Context, manifest: Manifest, obj_type: str, obj: str, **kwargs) -> Node: + """Return node from manifest""" + + +@command() +def get_nodes(context: Context, manifest: Manifest, obj_type: str, **kwargs) -> Dict[str, Node]: + """Return all nodes from manifest""" + + +@command() +def set_node(context: Context, manifest: Manifest, obj_type: str, obj_name, obj: Node, **kwargs): + """Add node to manifest""" + + +@command() +def has_model(context: Context, manifest: Manifest, model: str, **kwargs) -> bool: + """Check if manifest has specified model""" + + +@command() +def get_model(context: Context, manifest: Manifest, model: str, **kwargs) -> Model: + """Return model from manifest""" + + +@command() +def get_models(context: Context, manifest: Manifest, **kwargs) -> Dict[str, Model]: + """Return all models from manifest""" + + +@command() +def set_model(context: Context, manifest: Manifest, model_name: str, model: Model, **kwargs): + """Add model to manifest""" + + +@command() +def set_models(context: Context, manifest: Manifest, models: Dict[str, Model], **kwargs): + """Sets all model to manifest""" + + +@command() +def has_namespace(context: Context, manifest: Manifest, namespace: str, **kwargs) -> bool: + """Check if manifest has specified namespace""" + + +@command() +def get_namespaces(context: Context, manifest: Manifest, **kwargs) -> Dict[str, Namespace]: + """Return all namespaces from manifest""" + + +@command() +def get_namespace(context: Context, manifest: Manifest, namespace: str, **kwargs) -> Namespace: + """Return namespace from manifest""" + + +@command() +def set_namespace(context: Context, manifest: Manifest, namespace: str, ns: Namespace, **kwargs): + """Add namespace to manifest""" + + +@command() +def has_dataset(context: Context, manifest: Manifest, dataset: str, **kwargs) -> bool: + """Check if manifest has specified dataset""" + + +@command() +def get_dataset(context: Context, manifest: Manifest, dataset: str, **kwargs) -> Dataset: + """Return dataset from manifest""" + + +@command() +def get_datasets(context: Context, manifest: Manifest, **kwargs) -> Dict[str, Dataset]: + """Return all datasets from manifest""" + + +@command() +def set_dataset(context: Context, manifest: Manifest, dataset_name: str, dataset: Dataset, **kwargs): + """Add dataset to manifest""" + + diff --git a/spinta/commands/manifest.py b/spinta/commands/manifest.py new file mode 100644 index 000000000..6ace4153e --- /dev/null +++ b/spinta/commands/manifest.py @@ -0,0 +1,140 @@ +from typing import TypedDict, Callable, Dict + +from spinta import commands +from spinta.components import Namespace, Model, Node, Context +from spinta.datasets.components import Dataset +from spinta.exceptions import DatasetNotFound, NamespaceNotFound, ModelNotFound, ManifestObjectNotDefined +from spinta.manifests.components import Manifest + + +class _FunctionTypes(TypedDict): + has: Callable + get: Callable + set: Callable + get_all: Callable + + +NODE_FUNCTION_MAPPER = { + 'model': _FunctionTypes( + has=commands.has_model, + get=commands.get_model, + set=commands.set_model, + get_all=commands.get_models + ), + 'ns': _FunctionTypes( + has=commands.has_namespace, + get=commands.get_namespace, + set=commands.set_namespace, + get_all=commands.get_namespaces + ), + 'dataset': _FunctionTypes( + has=commands.has_dataset, + get=commands.get_dataset, + set=commands.set_dataset, + get_all=commands.get_datasets + ) +} + + +@commands.has_node_type.register(Context, Manifest, str) +def has_object_type(context: Context, manifest: Manifest, obj_type: str, **kwargs): + return obj_type in manifest.get_objects() + + +@commands.has_node.register(Context, Manifest, str, str) +def has_object(context: Context, manifest: Manifest, obj_type: str, obj: str, **kwargs): + if obj_type in NODE_FUNCTION_MAPPER: + return NODE_FUNCTION_MAPPER[obj_type]['has'](context, manifest, obj, **kwargs) + raise ManifestObjectNotDefined(obj=obj_type) + + +@commands.get_node.register(Context, Manifest, str, str) +def get_node(context: Context, manifest: Manifest, obj_type: str, obj: str, **kwargs): + if obj_type in NODE_FUNCTION_MAPPER: + return NODE_FUNCTION_MAPPER[obj_type]['get'](context, manifest, obj, **kwargs) + raise ManifestObjectNotDefined(obj=obj_type) + + +@commands.get_nodes.register(Context, Manifest, str) +def get_nodes(context: Context, manifest: Manifest, obj_type: str, **kwargs): + if obj_type in NODE_FUNCTION_MAPPER: + return NODE_FUNCTION_MAPPER[obj_type]['get_all'](context, manifest, **kwargs) + raise ManifestObjectNotDefined(obj=obj_type) + + +@commands.set_node.register(Context, Manifest, str, str, Node) +def set_node(context: Context, manifest: Manifest, obj_type: str, obj_name, obj: Node, **kwargs): + if obj_type in NODE_FUNCTION_MAPPER: + return NODE_FUNCTION_MAPPER[obj_type]['set'](context, manifest, obj_name, obj, **kwargs) + raise ManifestObjectNotDefined(obj=obj_type) + + +@commands.has_model.register(Context, Manifest, str) +def has_model(context: Context, manifest: Manifest, model: str, **kwargs): + return model in manifest.get_objects()['model'] + + +@commands.get_model.register(Context, Manifest, str) +def get_model(context: Context, manifest: Manifest, model: str, **kwargs): + if has_model(context, manifest, model): + return manifest.get_objects()['model'][model] + raise ModelNotFound(model=model) + + +@commands.get_models.register(Context, Manifest) +def get_models(context: Context, manifest: Manifest, **kwargs): + return manifest.get_objects()['model'] + + +@commands.set_model.register(Context, Manifest, str, Model) +def set_model(context: Context, manifest: Manifest, model_name: str, model: Model, **kwargs): + manifest.get_objects()['model'][model_name] = model + + +@commands.set_models.register(Context, Manifest, dict) +def set_models(context: Context, manifest: Manifest, models: Dict[str, Model], **kwargs): + manifest.get_objects()['model'] = models + + +@commands.has_namespace.register(Context, Manifest, str) +def has_namespace(context: Context, manifest: Manifest, namespace: str, **kwargs): + return namespace in manifest.get_objects()['ns'] + + +@commands.get_namespace.register(Context, Manifest, str) +def get_namespace(context: Context, manifest: Manifest, namespace: str, **kwargs): + if has_namespace(context, manifest, namespace): + return manifest.get_objects()['ns'][namespace] + raise NamespaceNotFound(namespace=namespace) + + +@commands.get_namespaces.register(Context, Manifest) +def get_namespaces(context: Context, manifest: Manifest, **kwargs): + return manifest.get_objects()['ns'] + + +@commands.set_namespace.register(Context, Manifest, str, Namespace) +def set_namespace(context: Context, manifest: Manifest, namespace: str, ns: Namespace, **kwargs): + manifest.get_objects()['ns'][namespace] = ns + + +@commands.has_dataset.register(Context, Manifest, str) +def has_dataset(context: Context, manifest: Manifest, dataset: str, **kwargs): + return dataset in manifest.get_objects()['dataset'] + + +@commands.get_dataset.register(Context, Manifest, str) +def get_dataset(context: Context, manifest: Manifest, dataset: str, **kwargs): + if has_dataset(context, manifest, dataset): + return manifest.get_objects()['dataset'][dataset] + raise DatasetNotFound(dataset=dataset) + + +@commands.get_datasets.register(Context, Manifest) +def get_datasets(context: Context, manifest: Manifest, **kwargs): + return manifest.get_objects()['dataset'] + + +@commands.set_dataset.register(Context, Manifest, str, Dataset) +def set_dataset(context: Context, manifest: Manifest, dataset_name: str, dataset: Dataset, **kwargs): + manifest.get_objects()['dataset'][dataset_name] = dataset diff --git a/spinta/commands/write.py b/spinta/commands/write.py index 9fadf7dde..9356ddb6b 100644 --- a/spinta/commands/write.py +++ b/spinta/commands/write.py @@ -37,7 +37,6 @@ from spinta.utils.streams import splitlines from spinta.utils.schema import NotAvailable, NA from spinta.utils.data import take -from spinta.types.namespace import traverse_ns_models from spinta.core.ufuncs import asttoexpr from spinta.formats.components import Format from spinta.types.text.components import Text @@ -1356,7 +1355,7 @@ async def wipe( # noqa action: Action, params: UrlParams, ): - for model in traverse_ns_models(context, ns, action, internal=True): + for model in commands.traverse_ns_models(context, ns, ns.manifest, action, internal=True): commands.authorize(context, Action.WIPE, model) commands.wipe(context, ns, backend) response = {'wiped': True} @@ -1373,5 +1372,5 @@ def prepare_headers( headers = {} if action == Action.INSERT and not is_batch: server_url = context.get('config').server_url - headers['location'] = f'{server_url}{node.endpoint}/{resp["_id"]}' + headers['location'] = f'{server_url}{node.name}/{resp["_id"]}' return headers diff --git a/spinta/components.py b/spinta/components.py index 1a7e18b4f..b4c4e14e5 100644 --- a/spinta/components.py +++ b/spinta/components.py @@ -415,6 +415,13 @@ def get_eid_for_error_context(self): return str(self.eid) +class ExtraMetaData(Node): + id: str = None + schema = { + 'id': {'type': 'string'} + } + + class NamespaceGiven: access: str = None @@ -456,7 +463,7 @@ def is_root(self) -> bool: return isinstance(self.parent, Manifest) -class Base(Node): +class Base(ExtraMetaData): model: Model # a.base.b - here `model` is `b` parent: Model # a.base.b - here `parent` is `a` pk: List[Property] # a.base.b - list of properties of `a` model @@ -602,13 +609,11 @@ def __init__(self): class Model(MetaData): - id: str level: Level access: Access title: str description: str ns: Namespace - endpoint: str = None external: Entity = None properties: Dict[str, Property] mode: Mode = None @@ -629,7 +634,6 @@ class Model(MetaData): 'base': {}, 'link': {}, 'properties': {'default': {}}, - 'endpoint': {}, 'external': {}, 'level': { 'type': 'integer', @@ -697,8 +701,7 @@ class PrepareGiven(TypedDict): prepare: str -class Property(Node): - id: Any = None +class Property(ExtraMetaData): place: str = None # Dotted property path title: str = None description: str = None diff --git a/spinta/config.py b/spinta/config.py index ca70b3948..0d6b99021 100644 --- a/spinta/config.py +++ b/spinta/config.py @@ -48,7 +48,8 @@ 'sql': 'spinta.manifests.sql.components:SqlManifest', 'memory': 'spinta.manifests.memory.components:MemoryManifest', 'json': 'spinta.manifests.dict.components:JsonManifest', - 'xml': 'spinta.manifests.dict.components:XmlManifest' + 'xml': 'spinta.manifests.dict.components:XmlManifest', + 'internal': 'spinta.manifests.internal_sql.components:InternalSQLManifest', }, 'backends': { # In memory backends mostly usable in tests @@ -76,8 +77,6 @@ 'nodes': { 'ns': 'spinta.components:Namespace', 'model': 'spinta.components:Model', - 'owner': 'spinta.types.owner:Owner', - 'project': 'spinta.types.project:Project', 'dataset': 'spinta.datasets.components:Dataset', 'base': 'spinta.components:Base', }, diff --git a/spinta/core/access.py b/spinta/core/access.py index d32435eba..f346edf2a 100644 --- a/spinta/core/access.py +++ b/spinta/core/access.py @@ -26,7 +26,7 @@ def load_access_param( Property, EnumItem, ], - given_access: str, + given_access: Union[str, Access], parents: Iterable[Union[ Manifest, Dataset, @@ -35,7 +35,7 @@ def load_access_param( Property, ]] = (), ) -> None: - access = enum_by_name(component, 'access', Access, given_access) + access = enum_by_name(component, 'access', Access, given_access) if not isinstance(given_access, Access) else given_access # If child has higher access than parent, increase parent access. if access is not None: diff --git a/spinta/core/config.py b/spinta/core/config.py index 137c9fe5a..5a3301ba8 100644 --- a/spinta/core/config.py +++ b/spinta/core/config.py @@ -595,6 +595,7 @@ def configure_rc( backend: str = None, resources: List[ResourceTuple] = None, dataset: str = None, + manifest_type: str = 'inline' ) -> RawConfig: config: Dict[str, Any] = {} @@ -628,21 +629,6 @@ def configure_rc( if dataset: config['given_dataset_name'] = dataset - if manifests: - for i, path in enumerate(manifests): - manifest_name = f'manifest{i}' - manifest = parse_manifest_path(rc, path) - config[f'manifests.{manifest_name}'] = { - 'type': manifest.type, - 'path': manifest.path, - 'file': manifest.file, - } - if isinstance(path, ResourceTuple) and path.prepare: - parsed = spyna.parse(path.prepare) - converted = asttoexpr(parsed) - config[f'manifests.{manifest_name}']['prepare'] = converted - sync.append(manifest_name) - if resources: inline.append({ 'type': 'dataset', @@ -653,14 +639,41 @@ def configure_rc( }, }) - config['manifests.default'] = { - 'type': 'inline', - 'backend': 'default', - 'keymap': 'default', - 'mode': mode.value, - 'sync': sync, - 'manifest': inline, - } + if manifest_type != 'inline': + manifest = parse_manifest_path(rc, manifests[0]) + config['manifests.default'] = { + 'type': manifest_type, + 'backend': 'default', + 'keymap': 'default', + 'mode': mode.value, + 'path': manifest.path, + 'file': manifest.file, + 'manifest': inline + } + else: + if manifests: + for i, path in enumerate(manifests): + manifest_name = f'manifest{i}' + manifest = parse_manifest_path(rc, path) + config[f'manifests.{manifest_name}'] = { + 'type': manifest.type, + 'path': manifest.path, + 'file': manifest.file, + } + if isinstance(path, ResourceTuple) and path.prepare: + parsed = spyna.parse(path.prepare) + converted = asttoexpr(parsed) + config[f'manifests.{manifest_name}']['prepare'] = converted + sync.append(manifest_name) + + config['manifests.default'] = { + 'type': manifest_type, + 'backend': 'default', + 'keymap': 'default', + 'mode': mode.value, + 'sync': sync, + 'manifest': inline, + } config['manifest'] = 'default' if check_names is not None: diff --git a/spinta/datasets/commands/link.py b/spinta/datasets/commands/link.py index 56f887e0a..6f56993df 100644 --- a/spinta/datasets/commands/link.py +++ b/spinta/datasets/commands/link.py @@ -23,9 +23,9 @@ def link(context: Context, resource: Resource): @commands.link.register(Context, Entity) def link(context: Context, entity: Entity): - datasets = entity.model.manifest.datasets + manifest = entity.model.manifest if entity.dataset: - if entity.dataset not in datasets: + if not commands.has_dataset(context, manifest, entity.dataset): raise MissingReference( entity, param='dataset', @@ -33,7 +33,7 @@ def link(context: Context, entity: Entity): ) # XXX: https://gitlab.com/atviriduomenys/spinta/-/issues/44 dataset: str = entity.dataset - entity.dataset = datasets[dataset] + entity.dataset = commands.get_dataset(context, manifest, dataset) resources = entity.dataset.resources if entity.resource: diff --git a/spinta/datasets/commands/load.py b/spinta/datasets/commands/load.py index 4dc54ec29..18adb013b 100644 --- a/spinta/datasets/commands/load.py +++ b/spinta/datasets/commands/load.py @@ -58,7 +58,7 @@ def load( # Load resources dataset.resources = {} for name, params in (data.get('resources') or {}).items(): - resource = get_node(config, manifest, dataset.eid, data, parent=dataset, group='datasets', ctype='resource') + resource = get_node(context, config, manifest, dataset.eid, data, parent=dataset, group='datasets', ctype='resource') resource.type = params.get('type') resource.name = name resource.dataset = dataset diff --git a/spinta/datasets/components.py b/spinta/datasets/components.py index 67114544b..1ebb9b98d 100644 --- a/spinta/datasets/components.py +++ b/spinta/datasets/components.py @@ -8,20 +8,19 @@ from sqlalchemy.engine.base import Engine from spinta.backends.components import Backend -from spinta.components import EntryId +from spinta.components import EntryId, ExtraMetaData from spinta.components import Namespace +from spinta.dimensions.comments.components import Comment from spinta.dimensions.lang.components import LangData from spinta.components import MetaData from spinta.components import Model -from spinta.components import Node from spinta.components import Property from spinta.core.enums import Access from spinta.core.ufuncs import Expr from spinta.datasets.enums import Level from spinta.dimensions.prefix.components import UriPrefix from spinta.manifests.components import Manifest -from spinta.types.owner import Owner -from spinta.types.project import Project +from spinta.utils.schema import NA class DatasetGiven: @@ -35,13 +34,10 @@ class Dataset(MetaData): DCAT (Data Catalog Vocabulary) - https://w3c.github.io/dxwg/dcat/ """ - id: str manifest: Manifest - owner: Owner = None level: Level = 3 access: Access = Access.private website: str = None - projects: List[Project] = None resources: Dict[str, Resource] = None source: Optional[str] = None # metadata source title: str @@ -98,7 +94,7 @@ class ExternalBackend(Backend): schema: sa.MetaData = None -class External(Node): +class External(ExtraMetaData): pass @@ -108,7 +104,6 @@ class ResourceGiven: class Resource(External): - id: Any = None eid: EntryId = None title: str description: str @@ -117,7 +112,7 @@ class Resource(External): level: Level access: Access external: str - prepare: str + prepare: Expr models: Dict[str, Model] given: ResourceGiven lang: LangData = None @@ -126,7 +121,7 @@ class Resource(External): schema = { 'type': {'type': 'string'}, 'dataset': {'parent': True}, - 'prepare': {'type': 'spyna'}, + 'prepare': {'type': 'spyna', 'default': NA}, # Backend name specified in `ref` column, points to previously defined # backend or to a configured stored backend. @@ -155,6 +150,7 @@ class Resource(External): 'title': {'type': 'string'}, 'description': {'type': 'string'}, 'comments': {}, + 'lang': {'type': 'object'}, 'given_name': {'type': 'string', 'default': None}, } @@ -191,7 +187,7 @@ class Entity(External): 'dataset': {'type': 'ref', 'ref': 'context.nodes.dataset'}, 'resource': {'type': 'ref', 'ref': 'dataset.resources'}, 'name': {'type': 'string', 'default': None}, - 'prepare': {'type': 'spyna', 'default': None}, + 'prepare': {'type': 'spyna', 'default': NA}, 'params': { 'type': 'array', 'items': {'type': 'object'}, @@ -211,10 +207,10 @@ class Entity(External): class Attribute(External): prop: Property # property name: str # property.source - prepare: Expr = None # property.prepare + prepare: Expr = NA # property.prepare schema = { 'prop': {'parent': True}, 'name': {'default': None}, - 'prepare': {'type': 'spyna', 'default': None}, + 'prepare': {'type': 'spyna', 'default': NA}, } diff --git a/spinta/datasets/inspect/helpers.py b/spinta/datasets/inspect/helpers.py index 01ec63de2..61f2953be 100644 --- a/spinta/datasets/inspect/helpers.py +++ b/spinta/datasets/inspect/helpers.py @@ -58,7 +58,7 @@ def create_manifest_from_inspect( ) with context: require_auth(context, auth) - store = load_manifest(context, ensure_config_dir=True) + store = load_manifest(context, ensure_config_dir=True, full_load=True) old = store.manifest manifest = Manifest() init_manifest(context, manifest, 'inspect') @@ -66,7 +66,7 @@ def create_manifest_from_inspect( if not resources: resources = [] - for ds in old.datasets.values(): + for ds in commands.get_datasets(context, old).values(): for resource in ds.resources.values(): external = resource.external if external == '' and resource.backend: @@ -82,7 +82,7 @@ def create_manifest_from_inspect( # Sort models for render sorted_models = {} - for key, model in manifest.models.items(): + for key, model in commands.get_models(context, manifest).items(): if key not in sorted_models.keys(): if model.external and model.external.resource: resource = model.external.resource @@ -91,7 +91,7 @@ def create_manifest_from_inspect( sorted_models[resource_key] = resource_model else: sorted_models[key] = model - manifest.objects['model'] = sorted_models + commands.set_models(context, manifest, sorted_models) return context, manifest @@ -100,7 +100,7 @@ def _merge(context: Context, manifest: Manifest, old: Manifest, resource: Resour manifest_ = get_manifest_from_type(rc, resource.type) path = ManifestPath(type=manifest_.type, path=resource.external) context = configure_context(context, [path], mode=Mode.external, dataset=dataset) - store = load_manifest(context) + store = load_manifest(context, full_load=True) new = store.manifest commands.merge(context, manifest, old, new, has_manifest_priority) @@ -124,8 +124,8 @@ def merge(context: Context, manifest: Manifest, old: Manifest, new: Manifest, ha n.name = name merge(context, manifest, o, n) datasets = zipitems( - old.datasets.values(), - new.datasets.values(), + commands.get_datasets(context, old).values(), + commands.get_datasets(context, new).values(), _dataset_key, ) deduplicator = Deduplicator("{}") @@ -157,10 +157,10 @@ def merge(context: Context, manifest: Manifest, old: ExternalBackend, new: NotAv @commands.merge.register(Context, Manifest, NotAvailable, Dataset, bool) def merge(context: Context, manifest: Manifest, old: NotAvailable, new: Dataset, has_manifest_priority: bool) -> None: - manifest.datasets[new.name] = new + commands.set_dataset(context, manifest, new.name, new) _merge_resources(context, manifest, old, new) - dataset_models = _filter_models_for_dataset(new.manifest, new) + dataset_models = _filter_models_for_dataset(context, new.manifest, new) deduplicator = Deduplicator() for model in dataset_models: model.name = deduplicator(model.name) @@ -177,9 +177,7 @@ def merge(context: Context, manifest: Manifest, old: Dataset, new: Dataset, has_ old.name = coalesce(old.name, new.name) old.manifest = coalesce(old.manifest, new.manifest) old.website = coalesce(old.website, new.website) - old.projects = coalesce(old.projects, new.projects) old.source = coalesce(old.source, new.source) - old.owner = coalesce(old.owner, new.owner) old.given = coalesce(old.given, new.given) old.level = coalesce(old.level, new.level) old.access = coalesce(old.access, new.access) @@ -190,14 +188,14 @@ def merge(context: Context, manifest: Manifest, old: Dataset, new: Dataset, has_ commands.merge(context, manifest, old.ns, new.ns) else: old.ns = coalesce(old.ns, new.ns) - manifest.datasets[old.name] = old + commands.set_dataset(context, manifest, old.name, old) _merge_prefixes(context, manifest, old, new) _merge_resources(context, manifest, old, new) - dataset_models = _filter_models_for_dataset(manifest, old) + dataset_models = _filter_models_for_dataset(context, manifest, old) models = zipitems( dataset_models, - new.manifest.models.values(), + commands.get_models(context, new.manifest).values(), _model_key ) resource_list = [] @@ -229,7 +227,7 @@ def merge(context: Context, manifest: Manifest, old: Dataset, new: NotAvailable, @commands.merge.register(Context, Manifest, NotAvailable, UriPrefix) def merge(context: Context, manifest: Manifest, old: NotAvailable, new: UriPrefix) -> None: dataset = new.parent - manifest.datasets[dataset.name].prefixes[new.name] = new + commands.get_dataset(context, manifest, dataset.name).prefixes[new.name] = new @commands.merge.register(Context, Manifest, UriPrefix, UriPrefix) @@ -250,7 +248,7 @@ def merge(context: Context, manifest: Manifest, old: UriPrefix, new: NotAvailabl @commands.merge.register(Context, Manifest, NotAvailable, Namespace) def merge(context: Context, manifest: Manifest, old: NotAvailable, new: Namespace) -> None: - manifest.namespaces[new.name] = new + commands.set_namespace(context, manifest, new.name, new) @commands.merge.register(Context, Manifest, Namespace, Namespace) @@ -277,7 +275,7 @@ def merge(context: Context, manifest: Manifest, old: Namespace, new: NotAvailabl @commands.merge.register(Context, Manifest, NotAvailable, Resource) def merge(context: Context, manifest: Manifest, old: NotAvailable, new: Resource) -> None: - manifest.datasets[new.dataset.name].resources[new.name] = new + commands.get_dataset(context, manifest, new.dataset.name).resources[new.name] = new @commands.merge.register(Context, Manifest, Resource, Resource) @@ -329,7 +327,7 @@ def merge(context: Context, manifest: Manifest, old: NotAvailable, new: Model, h del old_res.models[old_name] old_res.models[old.name] = old old.manifest = manifest - manifest.models[old.name] = old + commands.set_model(context, manifest, old.name, old) _merge_model_properties(context, manifest, old, new, has_manifest_priority) @@ -350,7 +348,7 @@ def merge(context: Context, manifest: Manifest, old: Model, new: Model, has_mani old.external = coalesce(old.external, new.external) old.manifest = manifest - manifest.models[old.name] = old + commands.set_model(context, manifest, old.name, old) _merge_model_properties(context, manifest, old, new, has_manifest_priority) if old.external and new.external: @@ -419,7 +417,8 @@ def merge(context: Context, manifest: Manifest, old: Property, new: Property, ha def merge(context: Context, manifest: Manifest, old: Property, new: NotAvailable, has_manifest_priority: bool) -> None: if old.external: old.external.name = None - manifest.models[old.model.name].properties[old.name] = old + model = commands.get_model(context, manifest, old.model.name) + model.properties[old.name] = old @commands.merge.register(Context, Manifest, DataType, Array) @@ -438,7 +437,7 @@ def merge(context: Context, manifest: Manifest, old: DataType, new: Array) -> No merged.prepare = coalesce(old.prepare, new.prepare) models = zipitems( [merged.items.model], - manifest.models.values(), + commands.get_models(context, manifest).values(), _model_key ) for model in models: @@ -486,7 +485,7 @@ def merge(context: Context, manifest: Manifest, old: DataType, new: Object) -> N new_value = value models = zipitems( [value.model], - manifest.models.values(), + commands.get_models(context, manifest).values(), _model_key ) for model in models: @@ -533,7 +532,7 @@ def merge(context: Context, manifest: Manifest, old: DataType, new: Ref) -> None models = zipitems( [merged.model], - manifest.models.values(), + commands.get_models(context, manifest).values(), _model_key ) for model in models: @@ -583,7 +582,7 @@ def merge(context: Context, manifest: Manifest, old: DataType, new: Denorm) -> N models = zipitems( [merged.rel_prop.model], - manifest.models.values(), + commands.get_models(context, manifest).values(), _model_key ) for model in models: @@ -628,11 +627,12 @@ def merge(context: Context, manifest: Manifest, old: DataType, new: DataType) -> def _filter_models_for_dataset( + context: Context, manifest: Manifest, dataset: Dataset ) -> List[Model]: models = [] - for model in manifest.models.values(): + for model in commands.get_models(context, manifest).values(): if model.external: if model.external.dataset is dataset: models.append(model) diff --git a/spinta/dimensions/enum/components.py b/spinta/dimensions/enum/components.py index 1a6868ef7..102c66330 100644 --- a/spinta/dimensions/enum/components.py +++ b/spinta/dimensions/enum/components.py @@ -2,8 +2,7 @@ from typing import Dict -from spinta.components import Model -from spinta.components import Node +from spinta.components import Model, ExtraMetaData from spinta.core.enums import Access from spinta.core.ufuncs import Env from spinta.core.ufuncs import Expr @@ -19,7 +18,7 @@ class EnumValueGiven: access: str = None -class EnumItem(Node): +class EnumItem(ExtraMetaData): source: str prepare: Expr access: Access = None diff --git a/spinta/dimensions/prefix/helpers.py b/spinta/dimensions/prefix/helpers.py index 6a774c778..7fcef9946 100644 --- a/spinta/dimensions/prefix/helpers.py +++ b/spinta/dimensions/prefix/helpers.py @@ -20,6 +20,7 @@ def load_prefixes( loaded = {} for name, data in prefixes.items(): prefix: UriPrefix = get_node( + context, config, manifest, data['eid'], diff --git a/spinta/exceptions.py b/spinta/exceptions.py index 741315488..3d1a4ca89 100644 --- a/spinta/exceptions.py +++ b/spinta/exceptions.py @@ -206,6 +206,16 @@ class ItemDoesNotExist(UserError): template = "Resource {id!r} not found." +class DatasetNotFound(UserError): + status_code = 404 + template = "Dataset {dataset!r} not found." + + +class NamespaceNotFound(UserError): + status_code = 404 + template = "Namespace {namespace!r} not found." + + class ModelNotFound(UserError): status_code = 404 template = "Model {model!r} not found." @@ -818,3 +828,11 @@ class InheritPropertyValueMissmatch(UserError): class OutOfMemoryMigrate(UserError): template = "Ran out of shared memory while migrating. Use '--autocommit' tag, or increase 'max_locks_per_transaction'." + + +class ManifestObjectNotDefined(UserError): + template = "Object {obj!r} is not defined in manifest objects list." + + +class InvalidIdType(UserError): + template = "Id {id!r} of {id_type!r} type is invalid." diff --git a/spinta/formats/ascii/components.py b/spinta/formats/ascii/components.py index cb0e37497..24e15ff17 100644 --- a/spinta/formats/ascii/components.py +++ b/spinta/formats/ascii/components.py @@ -1,6 +1,7 @@ import operator import itertools +from spinta import commands from spinta.components import Context, Action, UrlParams, Model from spinta.formats.ascii.helpers import get_widths, get_displayed_cols, draw_border, draw_header, draw_row from spinta.manifests.components import Manifest @@ -53,7 +54,7 @@ def __call__( for name, group in groups: if name: yield f'\n\nTable: {name}\n' - model = manifest.models[name] + model = commands.get_model(context, manifest, name) rows = flatten(group) rows = rename_page_col(rows) diff --git a/spinta/formats/csv/commands.py b/spinta/formats/csv/commands.py index c03451075..22dd5d0e1 100644 --- a/spinta/formats/csv/commands.py +++ b/spinta/formats/csv/commands.py @@ -59,7 +59,7 @@ def render( headers: Dict[str, str] = None, path: str = None ) -> Response: - rows = datasets_to_tabular(manifest) + rows = datasets_to_tabular(context, manifest) rows = ({c: row[c] for c in DATASET} for row in rows) if not path: headers = headers or {} diff --git a/spinta/formats/html/helpers.py b/spinta/formats/html/helpers.py index 194bf6136..e511ff2d4 100644 --- a/spinta/formats/html/helpers.py +++ b/spinta/formats/html/helpers.py @@ -8,6 +8,7 @@ from typing import Tuple from typing import TypedDict +from spinta import commands from spinta.components import Config from spinta.components import Context from spinta.components import Model @@ -28,6 +29,7 @@ def get_current_location( + context: Context, config: Config, model: Model, params: UrlParams, @@ -40,7 +42,7 @@ def get_current_location( elif config.root.startswith(path): path = '' - parts = _split_path(model.manifest, config.root, path) + parts = _split_path(context, model.manifest, config.root, path) if len(parts) > 0: parts, last = parts[:-1], parts[-1] else: @@ -212,6 +214,7 @@ class PathInfo(NamedTuple): def _split_path( + context: Context, manifest: Manifest, base: str, orig_path: str, @@ -222,10 +225,10 @@ def _split_path( base = [base] if base else [] for i, part in enumerate(parts, 1): path = '/'.join(base + parts[:i]) - if i == last and path in manifest.models: - title = manifest.models[path].title - elif path in manifest.namespaces: - title = manifest.namespaces[path].title + if i == last and commands.has_model(context, manifest, path): + title = commands.get_model(context, manifest, path).title + elif commands.has_namespace(context, manifest, path): + title = commands.get_namespace(context, manifest, path).title else: title = '' title = title or part @@ -241,7 +244,7 @@ def _split_path( def get_template_context(context: Context, model, params: UrlParams): config: Config = context.get('config') return { - 'location': get_current_location(config, model, params), + 'location': get_current_location(context, config, model, params), } diff --git a/spinta/formats/rdf/commands.py b/spinta/formats/rdf/commands.py index 9cb37dcd1..7d516b7ba 100644 --- a/spinta/formats/rdf/commands.py +++ b/spinta/formats/rdf/commands.py @@ -30,7 +30,6 @@ from spinta.types.datatype import Time from spinta.types.datatype import DateTime from spinta.types.datatype import Number -from spinta.types.namespace import traverse_ns_models from spinta.utils.encoding import encode_page_values from spinta.utils.schema import NotAvailable @@ -39,13 +38,13 @@ DESCRIPTION = "Description" -def _get_available_prefixes(model: Model) -> dict: +def _get_available_prefixes(context: Context, model: Model) -> dict: prefixes = { RDF: "http://www.w3.org/1999/02/22-rdf-syntax-ns#", PAV: "http://purl.org/pav/" } - if model.manifest.datasets.get(model.ns.name): - manifest_prefixes = model.manifest.datasets.get(model.ns.name).prefixes + if commands.has_dataset(context, model.manifest, model.ns.name): + manifest_prefixes = commands.get_dataset(context, model.manifest, model.ns.name).prefixes for key, val in manifest_prefixes.items(): if isinstance(val, UriPrefix): prefixes[key] = val.uri @@ -218,6 +217,7 @@ def render( return StreamingResponse( _stream( + context, request, model, action, @@ -259,13 +259,14 @@ def render( async def _stream( + context: Context, request: Request, model: Model, action: Action, data ): namespaces = [] - prefixes = _get_available_prefixes(model) + prefixes = _get_available_prefixes(context, model) root_name = _get_attribute_name(RDF.upper(), RDF, prefixes) for key, val in prefixes.items(): namespaces.append(f'xmlns:{key}="{val}"') @@ -292,9 +293,10 @@ async def _stream_namespace( data ): namespaces = [] - models = traverse_ns_models( + models = commands.traverse_ns_models( context, ns, + ns.manifest, action, internal=True, ) @@ -346,7 +348,7 @@ def prepare_data_for_response( value = value.copy() reserved = get_model_reserved_props(action, model) - available_prefixes = _get_available_prefixes(model) + available_prefixes = _get_available_prefixes(context, model) value['_available_prefixes'] = available_prefixes value['_about_name'] = _get_attribute_name('about', RDF, available_prefixes) diff --git a/spinta/formats/xlsx/commands.py b/spinta/formats/xlsx/commands.py index f38ce5905..cd67c812a 100644 --- a/spinta/formats/xlsx/commands.py +++ b/spinta/formats/xlsx/commands.py @@ -61,7 +61,7 @@ def render( headers: Dict[str, str] = None, path: str = None ) -> Response: - rows = datasets_to_tabular(manifest) + rows = datasets_to_tabular(context, manifest) rows = ({c: row[c] for c in DATASET} for row in rows) if not path: headers = headers or {} diff --git a/spinta/manifests/backend/commands/load.py b/spinta/manifests/backend/commands/load.py index 41eeee95c..18040810e 100644 --- a/spinta/manifests/backend/commands/load.py +++ b/spinta/manifests/backend/commands/load.py @@ -13,6 +13,7 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): assert freezed, ( "BackendManifest does not have unfreezed version of manifest." diff --git a/spinta/manifests/backend/commands/sync.py b/spinta/manifests/backend/commands/sync.py index 1741bc869..24577c4d5 100644 --- a/spinta/manifests/backend/commands/sync.py +++ b/spinta/manifests/backend/commands/sync.py @@ -18,5 +18,5 @@ async def sync( sources: List[Manifest] = None, ): stream = read_sync_versions(context, manifest) - stream = versions_to_dstream(manifest, stream) + stream = versions_to_dstream(context, manifest, stream) await adrain(push_stream(context, stream)) diff --git a/spinta/manifests/backend/helpers.py b/spinta/manifests/backend/helpers.py index 20766fc5a..6494f10d7 100644 --- a/spinta/manifests/backend/helpers.py +++ b/spinta/manifests/backend/helpers.py @@ -32,7 +32,7 @@ async def run_bootstrap(context: Context, manifest: BackendManifest): # Sync versions stream = read_sync_versions(context, manifest) - stream = versions_to_dstream(manifest, stream, applied=True) + stream = versions_to_dstream(context, manifest, stream, applied=True) await adrain(push_stream(context, stream)) # Update schemas to last version @@ -43,12 +43,12 @@ async def run_bootstrap(context: Context, manifest: BackendManifest): async def run_migrations(context: Context, manifest: BackendManifest): # Sync versions stream = read_sync_versions(context, manifest) - stream = versions_to_dstream(manifest, stream) + stream = versions_to_dstream(context, manifest, stream) await adrain(push_stream(context, stream)) # Apply unapplied versions store = manifest.store - model = manifest.objects['ns'][''] + model = commands.get_namespace(context, manifest, '') backends = {} versions = read_unapplied_versions(context, manifest) versions = itertools.groupby(versions, key=lambda v: v.get('backend', 'default')) @@ -90,7 +90,7 @@ def read_unapplied_versions( context: Context, manifest: Manifest, ): - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') query = Expr( 'and', Expr('select', bind('id'), bind('_id'), bind('parents')), @@ -113,13 +113,14 @@ def read_sync_versions(context: Context, manifest: Manifest): async def versions_to_dstream( + context: Context, manifest: BackendManifest, versions: Iterable[dict], *, applied: bool = False, ) -> AsyncIterator[DataItem]: now = datetime.datetime.now(datetime.timezone.utc) - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') for version in versions: payload = { '_op': 'upsert', @@ -142,7 +143,7 @@ async def versions_to_dstream( def list_schemas(context: Context, manifest: BackendManifest): - model = manifest.objects['model']['_schema'] + model = commands.get_model(context, manifest, '_schema') query = { 'select': ['_id'], } @@ -151,7 +152,7 @@ def list_schemas(context: Context, manifest: BackendManifest): def read_schema(context: Context, manifest: BackendManifest, eid: str): - model = manifest.objects['model']['_schema'] + model = commands.get_model(context, manifest, '_schema') row = commands.getone(context, model, model.backend, id_=eid) return row['schema'] @@ -160,7 +161,7 @@ def list_sorted_unapplied_versions( context: Context, manifest: Manifest, ) -> Iterator[Tuple[str, str]]: - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') query = { 'select': ['id', '_id', 'parents'], 'query': [ @@ -182,7 +183,7 @@ def read_lastest_version_schemas( context: Context, manifest: Manifest, ) -> Iterator[Tuple[str, str]]: - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') query = Expr( 'and', Expr('select', bind('id'), bind('_id'), bind('parents')), @@ -204,7 +205,7 @@ def get_last_version_eid( manifest: Manifest, schema_eid: str, ) -> Iterator[Tuple[str, str]]: - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') query = Expr( 'and', Expr('select', bind('_id'), bind('parents')), @@ -224,13 +225,13 @@ def get_version_schema( manifest: Manifest, version_eid: str, ) -> Iterator[Tuple[str, str]]: - model = manifest.objects['model']['_schema/Version'] + model = commands.get_model(context, manifest, '_schema/Version') version = commands.getone(context, model, model.backend, id_=version_eid) return version['schema'] async def update_schema_version(context: Context, manifest: Manifest, schema: dict): - model = manifest.objects['model']['_schema'] + model = commands.get_model(context, manifest, '_schema') data = DataItem(model, action=Action.UPSERT, payload={ '_op': 'upsert', '_where': '_id="%s"' % schema['id'], diff --git a/spinta/manifests/commands/auth.py b/spinta/manifests/commands/auth.py new file mode 100644 index 000000000..c2f81ded5 --- /dev/null +++ b/spinta/manifests/commands/auth.py @@ -0,0 +1,9 @@ +from spinta import commands +from spinta.auth import authorized +from spinta.components import Context, Action, Namespace +from spinta.manifests.components import Manifest + + +@commands.authorize.register(Context, Action, Namespace, Manifest) +def authorize(context: Context, action: Action, ns: Namespace, manifest: Manifest): + authorized(context, ns, action, throw=True) diff --git a/spinta/manifests/commands/check.py b/spinta/manifests/commands/check.py index 01d87d78c..a9a7c711b 100644 --- a/spinta/manifests/commands/check.py +++ b/spinta/manifests/commands/check.py @@ -1,19 +1,10 @@ from spinta import commands from spinta.components import Context -from spinta.manifests.components import Manifest +from spinta.manifests.components import Manifest, get_manifest_object_names @commands.check.register(Context, Manifest) def check(context: Context, manifest: Manifest): - for objects in manifest.objects.values(): - for obj in objects.values(): + for node in get_manifest_object_names(): + for obj in commands.get_nodes(context, manifest, node).values(): check(context, obj) - - # Check endpoints. - names = set(manifest.models) - for model in manifest.models.values(): - if model.endpoint == model.name or model.endpoint in names: - raise Exception( - f"Endpoint name can't overshadow existing model names and " - f"{model.endpoint!r} is already a model name." - ) diff --git a/spinta/manifests/commands/inspect.py b/spinta/manifests/commands/inspect.py index dcc107564..b3ff48a02 100644 --- a/spinta/manifests/commands/inspect.py +++ b/spinta/manifests/commands/inspect.py @@ -17,7 +17,7 @@ def inspect( manifest: Manifest, source: None, ) -> Iterator[ManifestSchema]: - for dataset in manifest.datasets.values(): + for dataset in commands.get_datasets(context, manifest).values(): yield from commands.inspect(context, manifest.backend, dataset, None) diff --git a/spinta/manifests/commands/link.py b/spinta/manifests/commands/link.py index 7c0b5bfbd..13f05a59c 100644 --- a/spinta/manifests/commands/link.py +++ b/spinta/manifests/commands/link.py @@ -1,13 +1,13 @@ from spinta import commands from spinta.components import Context, MetaData -from spinta.manifests.components import Manifest +from spinta.manifests.components import Manifest, get_manifest_object_names @commands.link.register(Context, Manifest) def link(context: Context, manifest: Manifest): - for nodes in manifest.objects.values(): - for node in nodes.values(): - commands.link(context, node) + for node in get_manifest_object_names(): + for obj in commands.get_nodes(context, manifest, node).values(): + commands.link(context, obj) @commands.link.register(Context, MetaData) diff --git a/spinta/manifests/commands/load.py b/spinta/manifests/commands/load.py new file mode 100644 index 000000000..d99ee1107 --- /dev/null +++ b/spinta/manifests/commands/load.py @@ -0,0 +1,18 @@ +from spinta import commands +from spinta.components import Context +from spinta.manifests.components import Manifest + + +@commands.load_for_request.register(Context, Manifest) +def load_for_request(context: Context, manifest: Manifest): + pass + + +@commands.fully_initialize_manifest.register(Context, Manifest) +def fully_initialize_manifest(context: Context, manifest: Manifest): + pass + + +@commands.create_request_manifest.register(Context, Manifest) +def create_request_manifest(context: Context, manifest: Manifest): + return manifest diff --git a/spinta/manifests/commands/read.py b/spinta/manifests/commands/read.py new file mode 100644 index 000000000..3e86c0f80 --- /dev/null +++ b/spinta/manifests/commands/read.py @@ -0,0 +1,189 @@ +from typing import Optional, NamedTuple, Union, Dict, Any, Iterable +from starlette.requests import Request +from starlette.responses import Response +from spinta import commands +from spinta.accesslog import log_response +from spinta.backends.helpers import get_select_tree, get_select_prop_names +from spinta.renderer import render +from spinta.compat import urlparams_to_expr +from spinta.components import Context, Namespace, Action, UrlParams, Model +from spinta.manifests.components import Manifest +from spinta.types.namespace import _model_matches_params, check_if_model_has_backend_and_source +from spinta.utils import itertools + + +@commands.traverse_ns_models.register(Context, Namespace, Manifest, Action) +def traverse_ns_models( + context: Context, + ns: Namespace, + manifest: Manifest, + action: Action, + dataset_: Optional[str] = None, + resource: Optional[str] = None, + internal: bool = False, + source_check: bool = False, + **kwargs +): + models = (ns.models or {}) + for model in models.values(): + if not (source_check and not check_if_model_has_backend_and_source(model)): + if _model_matches_params(context, model, action, dataset_, resource, internal): + yield model + for ns_ in ns.names.values(): + if not internal and ns_.name.startswith('_'): + continue + yield from commands.traverse_ns_models( + context, + ns_, + manifest, + action, + dataset_=dataset_, + resource=resource, + internal=internal, + source_check=source_check + ) + + +@commands.getall.register(Context, Namespace, Request, Manifest) +def getall( + context: Context, + ns: Namespace, + request: Request, + manifest: Manifest, + *, + action: Action, + params: UrlParams +): + if params.all and params.ns: + + for model in commands.traverse_ns_models(context, ns, manifest, action, internal=True): + commands.authorize(context, action, model) + return _get_ns_content( + context, + request, + ns, + params, + action, + recursive=True, + ) + elif params.all: + accesslog = context.get('accesslog') + + prepare_data_for_response_kwargs = {} + for model in commands.traverse_ns_models(context, ns, manifest, action, internal=True): + commands.authorize(context, action, model) + select_tree = get_select_tree(context, action, params.select) + prop_names = get_select_prop_names( + context, + model, + model.properties, + action, + select_tree, + ) + prepare_data_for_response_kwargs[model.model_type()] = { + 'select': select_tree, + 'prop_names': prop_names, + } + expr = urlparams_to_expr(params) + rows = commands.getall(context, ns, action=action, query=expr) + rows = ( + commands.prepare_data_for_response( + context, + commands.get_model(context, manifest, row['_type']), + params.fmt, + row, + action=action, + **prepare_data_for_response_kwargs[row['_type']], + ) + for row in rows + ) + rows = log_response(context, rows) + return render(context, request, ns, params, rows, action=action) + else: + return _get_ns_content(context, request, ns, params, action) + + +def _get_ns_content( + context: Context, + request: Request, + ns: Namespace, + params: UrlParams, + action: Action, + *, + recursive: bool = False, + dataset_: Optional[str] = None, + resource: Optional[str] = None, +) -> Response: + if recursive: + data = _get_ns_content_data_recursive(context, ns, action, dataset_, resource) + else: + data = _get_ns_content_data(context, ns, action, dataset_, resource) + + data = sorted(data, key=lambda x: (x.data['_type'] != 'ns', x.data['name'])) + + model = commands.get_model(context, ns.manifest, '_ns') + select = params.select or ['name', 'title', 'description'] + select_tree = get_select_tree(context, action, select) + prop_names = get_select_prop_names( + context, + model, + model.properties, + action, + select_tree, + auth=False, + ) + rows = ( + commands.prepare_data_for_response( + context, + model, + params.fmt, + row.data, + action=action, + select=select_tree, + prop_names=prop_names, + ) + for row in data + ) + + rows = log_response(context, rows) + + return render(context, request, model, params, rows, action=action) + + +class _NodeAndData(NamedTuple): + node: Union[Namespace, Model] + data: Dict[str, Any] + + +def _get_ns_content_data_recursive( + context: Context, + ns: Namespace, + action: Action, + dataset_: Optional[str] = None, + resource: Optional[str] = None, +) -> Iterable[_NodeAndData]: + yield from _get_ns_content_data(context, ns, action, dataset_, resource) + for name in ns.names.values(): + yield from _get_ns_content_data_recursive(context, name, action, dataset_, resource) + + +def _get_ns_content_data( + context: Context, + ns: Namespace, + action: Action, + dataset_: Optional[str] = None, + resource: Optional[str] = None, +) -> Iterable[_NodeAndData]: + items: Iterable[Union[Namespace, Model]] = itertools.chain( + ns.names.values(), + ns.models.values(), + ) + + for item in items: + if _model_matches_params(context, item, action, dataset_, resource): + yield _NodeAndData(item, { + '_type': item.node_type(), + 'name': item.model_type(), + 'title': item.title, + 'description': item.description, + }) diff --git a/spinta/manifests/components.py b/spinta/manifests/components.py index f22c4cb4b..361f063b7 100644 --- a/spinta/manifests/components.py +++ b/spinta/manifests/components.py @@ -13,7 +13,7 @@ from typing import TypedDict from typing import Union -from spinta.components import Component +from spinta.components import Component, Context from spinta.components import Mode from spinta.components import Model from spinta.components import Namespace @@ -34,6 +34,10 @@ class MetaDataContainer(TypedDict): model: Dict[str, Model] +def get_manifest_object_names(): + return MetaDataContainer.__annotations__.keys() + + class ManifestGiven: access: str = None @@ -45,16 +49,12 @@ class Manifest(Component): backend: Backend = None parent: Component = None store: Store = None - objects: MetaDataContainer = None + _objects: MetaDataContainer = None path: str = None access: Access = Access.protected prefixes: Dict[str, UriPrefix] enums: Enums - # {: } mapping. There can be multiple model types, - # but name and endpoint for all of them should match. - endpoints: Dict[str, str] = None - # Backends defined in the manifest. backends: Dict[str, Backend] = None @@ -91,28 +91,8 @@ def __eq__(self, other: Union[Manifest, str]): else: super().__eq__(other) - def add_model_endpoint(self, model): - endpoint = model.endpoint - if endpoint: - if endpoint not in self.endpoints: - self.endpoints[endpoint] = model.name - elif self.endpoints[endpoint] != model.name: - raise Exception( - f"Same endpoint, but different model name, " - f"endpoint={endpoint!r}, model.name={model.name!r}." - ) - - @property - def models(self) -> Dict[str, Model]: - return self.objects['model'] - - @property - def datasets(self) -> Dict[str, Dataset]: - return self.objects['dataset'] - - @property - def namespaces(self) -> Dict[str, Namespace]: - return self.objects['ns'] + def get_objects(self) -> dict: + return self._objects NodeSchema = Optional[Dict[str, Any]] diff --git a/spinta/manifests/dict/commands/load.py b/spinta/manifests/dict/commands/load.py index dd4d2ea9b..2dc23be34 100644 --- a/spinta/manifests/dict/commands/load.py +++ b/spinta/manifests/dict/commands/load.py @@ -15,12 +15,13 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) if manifest.path is None: return @@ -40,4 +41,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/manifests/helpers.py b/spinta/manifests/helpers.py index 2c7e03b7b..13c3ebd2e 100644 --- a/spinta/manifests/helpers.py +++ b/spinta/manifests/helpers.py @@ -37,9 +37,8 @@ def init_manifest(context: Context, manifest: Manifest, name: str): manifest.name = name manifest.store = context.get('store') manifest.parent = None - manifest.endpoints = {} manifest.backends = {} - manifest.objects = {name: {} for name in config.components['nodes']} + manifest._objects = {name: {} for name in config.components['nodes']} manifest.sync = [] manifest.prefixes = {} manifest.enums = {} @@ -136,11 +135,11 @@ def load_manifest_nodes( _load_manifest(context, manifest, schema, eid) else: node = _load_manifest_node(context, config, manifest, source, eid, schema) - manifest.objects[node.type][node.name] = node + commands.set_node(context, manifest, node.type, node.name, node) if link: to_link.append(node) - if '' not in manifest.namespaces: + if not commands.has_namespace(context, manifest, '', loaded=True): # Root namespace must always be present in manifest event if manifest is # empty. load_namespace_from_name(context, manifest, '', drop=False) @@ -154,8 +153,12 @@ def _load_manifest_backends( context: Context, manifest: Manifest, backends: Dict[str, Dict[str, str]], + reset: bool = True ) -> None: - manifest.backends = {} + if reset: + manifest.backends = {} + elif not manifest.backends: + manifest.backends = {} for name, data in backends.items(): manifest.backends[name] = load_backend( context, @@ -171,9 +174,10 @@ def _load_manifest( manifest: Manifest, data: Dict[str, Any], eid: EntryId, + reset: bool = True ) -> None: if 'backends' in data: - _load_manifest_backends(context, manifest, data['backends']) + _load_manifest_backends(context, manifest, data['backends'], reset=reset) if 'prefixes' in data: prefixes = load_prefixes(context, manifest, manifest, data['prefixes']) manifest.prefixes.update(prefixes) @@ -191,7 +195,7 @@ def _load_manifest_node( eid: EntryId, data: dict, ) -> MetaData: - node = get_node(config, manifest, eid, data) + node = get_node(context, config, manifest, eid, data) node.eid = eid node.type = data['type'] node.parent = manifest diff --git a/spinta/manifests/internal/commands/load.py b/spinta/manifests/internal/commands/load.py index 3de785ef8..614a43b0b 100644 --- a/spinta/manifests/internal/commands/load.py +++ b/spinta/manifests/internal/commands/load.py @@ -20,6 +20,7 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): if freezed: if into: diff --git a/spinta/manifests/internal_sql/__init__.py b/spinta/manifests/internal_sql/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/spinta/manifests/internal_sql/commands/__init__.py b/spinta/manifests/internal_sql/commands/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/spinta/manifests/internal_sql/commands/auth.py b/spinta/manifests/internal_sql/commands/auth.py new file mode 100644 index 000000000..cccf92a27 --- /dev/null +++ b/spinta/manifests/internal_sql/commands/auth.py @@ -0,0 +1,125 @@ +from typing import List +import sqlalchemy as sa + +from spinta import commands +from spinta.auth import get_client_id_from_name, get_clients_path +from spinta.components import Context, Action, Config, Namespace +from spinta.core.enums import Access +from spinta.exceptions import AuthorizedClientsOnly +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.utils.enums import get_enum_by_name +from spinta.utils.scopes import name_to_scope + + +def get_transaction_connection(context: Context): + if context.has('transaction.manifest'): + return context.get('transaction.manifest').connection + return None + + +def get_namespace_highest_access(context: Context, manifest: InternalSQLManifest, namespace: str): + conn = get_transaction_connection(context) + highest = None + if conn is not None: + table = manifest.table + results = conn.execute(sa.select(table.c.access, sa.func.min(table.c.mpath).label('mpath')).where( + sa.and_( + table.c.mpath.startswith(namespace), + sa.or_( + table.c.dim == 'ns', + table.c.dim == 'dataset', + table.c.dim == 'model', + table.c.dim == 'property' + ), + ) + ).group_by(table.c.access)) + + for result in results: + if result['access'] is not None: + enum = get_enum_by_name(Access, result['access']) + if highest is None or enum > highest: + highest = enum + else: + objs = manifest.get_objects() + if namespace in objs['ns']: + highest = objs['ns'][namespace].access + return highest if highest is not None else manifest.access + + +def internal_authorized( + context: Context, + name: str, + access: Access, + action: Action, + parents: List[str], + *, + throw: bool = False, +): + config: Config = context.get('config') + token = context.get('auth.token') + + # Unauthorized clients can only access open nodes. + unauthorized = token.get_client_id() == get_client_id_from_name(get_clients_path(config), config.default_auth_client) + + open_node = access >= Access.open + if unauthorized and not open_node: + if throw: + raise AuthorizedClientsOnly() + else: + return False + + # Private nodes can only be accessed with explicit node scope. + scopes = [name] + + # Protected and higher level nodes can be accessed with parent nodes scopes. + if access > Access.private: + scopes.extend(parents) + + if not isinstance(action, (list, tuple)): + action = [action] + scopes = [ + internal_scope_formatter(context, scope, act) + for act in action + for scope in scopes + ] + + # Check if client has at least one of required scopes. + if throw: + token.check_scope(scopes, operator='OR') + else: + return token.valid_scope(scopes, operator='OR') + + +def internal_scope_formatter( + context: Context, + name: str, + action: Action, +) -> str: + config = context.get('config') + + return name_to_scope( + '{prefix}{name}_{action}' if name else '{prefix}{action}', + name, + maxlen=config.scope_max_length, + params={ + 'prefix': config.scope_prefix, + 'action': action.value, + }, + ) + + +@commands.authorize.register(Context, Action, Namespace, InternalSQLManifest) +def authorize(context: Context, action: Action, ns: Namespace, manifest: InternalSQLManifest): + parents = [parent.name for parent in ns.parents()] + return internal_authorized( + context, + ns.name, + get_namespace_highest_access( + context, + manifest, + ns.name + ), + action, + parents, + throw=True + ) diff --git a/spinta/manifests/internal_sql/commands/bootstrap.py b/spinta/manifests/internal_sql/commands/bootstrap.py new file mode 100644 index 000000000..e75ef0dee --- /dev/null +++ b/spinta/manifests/internal_sql/commands/bootstrap.py @@ -0,0 +1,22 @@ +from spinta import commands +from spinta.components import Context +from spinta.manifests.internal_sql.components import InternalSQLManifest +import sqlalchemy as sa + +from spinta.manifests.internal_sql.helpers import get_table_structure + + +@commands.bootstrap.register(Context, InternalSQLManifest) +def bootstrap(context: Context, manifest: InternalSQLManifest): + store = context.get('store') + url = sa.engine.make_url(manifest.path) + url.get_dialect() + engine = sa.create_engine(url) + inspector = sa.inspect(engine) + meta = sa.MetaData(engine) + if not inspector.has_table('_manifest'): + table = get_table_structure(meta) + table.create() + + for backend in store.backends.values(): + commands.bootstrap(context, backend) diff --git a/spinta/manifests/internal_sql/commands/configure.py b/spinta/manifests/internal_sql/commands/configure.py new file mode 100644 index 000000000..912df7b23 --- /dev/null +++ b/spinta/manifests/internal_sql/commands/configure.py @@ -0,0 +1,20 @@ +from typing import Optional +import sqlalchemy as sa + +from spinta import commands +from spinta.components import Context +from spinta.core.config import RawConfig +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.manifests.internal_sql.helpers import get_table_structure + + +@commands.configure.register(Context, InternalSQLManifest) +def configure(context: Context, manifest: InternalSQLManifest): + rc: RawConfig = context.get('rc') + path: Optional[str] = rc.get('manifests', manifest.name, 'path') + manifest.path = path + url = sa.engine.make_url(manifest.path) + engine = sa.create_engine(url) + manifest.engine = engine + meta = sa.MetaData(engine) + manifest.table = get_table_structure(meta) diff --git a/spinta/manifests/internal_sql/commands/load.py b/spinta/manifests/internal_sql/commands/load.py new file mode 100644 index 000000000..67793e0fe --- /dev/null +++ b/spinta/manifests/internal_sql/commands/load.py @@ -0,0 +1,80 @@ +import logging + +from spinta import commands +from spinta.components import Context, Namespace +from spinta.manifests.helpers import init_manifest, _configure_manifest, load_manifest_nodes +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.manifests.components import Manifest +from spinta.manifests.internal_sql.helpers import read_initial_schema, load_internal_manifest_nodes, read_schema + +log = logging.getLogger(__name__) + + +@commands.create_request_manifest.register(Context, InternalSQLManifest) +def create_request_manifest(context: Context, manifest: InternalSQLManifest): + old = manifest + store = manifest.store + manifest = old.__class__() + rc = context.get('rc') + init_manifest(context, manifest, old.name) + _configure_manifest( + context, rc, store, manifest, + backend=store.manifest.backend.name if store.manifest.backend else None, + ) + commands.load(context, manifest) + commands.link(context, manifest) + return manifest + + +@commands.load_for_request.register(Context, InternalSQLManifest) +def load_for_request(context: Context, manifest: InternalSQLManifest): + context.attach('transaction.manifest', manifest.transaction) + schemas = read_initial_schema(context, manifest) + load_internal_manifest_nodes(context, manifest, schemas, link=True) + + +@commands.load.register(Context, InternalSQLManifest) +def load( + context: Context, + manifest: InternalSQLManifest, + *, + into: Manifest = None, + freezed: bool = True, + rename_duplicates: bool = False, + load_internal: bool = True, + full_load=False +): + if load_internal: + target = into or manifest + if '_schema' not in target.get_objects()['model']: + store = context.get('store') + commands.load(context, store.internal, into=target, full_load=full_load) + + if full_load: + schemas = read_schema(manifest.path) + if into: + log.info( + 'Loading freezed manifest %r into %r from %s.', + manifest.name, + into.name, + manifest.path, + ) + load_manifest_nodes(context, into, schemas, source=manifest) + else: + log.info( + 'Loading freezed manifest %r from %s.', + manifest.name, + manifest.path, + ) + load_manifest_nodes(context, manifest, schemas) + + for source in manifest.sync: + commands.load( + context, source, + into=into or manifest, + freezed=freezed, + rename_duplicates=rename_duplicates, + load_internal=load_internal, + full_load=full_load + ) + diff --git a/spinta/manifests/internal_sql/commands/manifest.py b/spinta/manifests/internal_sql/commands/manifest.py new file mode 100644 index 000000000..b7ba1dd89 --- /dev/null +++ b/spinta/manifests/internal_sql/commands/manifest.py @@ -0,0 +1,418 @@ +from typing import Dict +import sqlalchemy as sa +from spinta import commands +from spinta.components import Model, Namespace, Context +from spinta.datasets.components import Dataset +from spinta.exceptions import ModelNotFound, NamespaceNotFound, DatasetNotFound +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.manifests.internal_sql.helpers import internal_to_schema, load_internal_manifest_nodes, get_object_from_id, \ + select_full_table, update_schema_with_external, load_required_models, get_manifest, get_transaction_connection +from spinta.types.namespace import load_namespace_from_name + + +def get_model_name_list(context: Context, manifest: InternalSQLManifest, loaded: bool, namespace: str = None, recursive: bool = False): + manifest = get_manifest(context, manifest) + table = manifest.table + conn = get_transaction_connection(context) + objs = manifest.get_objects() + if conn is None or loaded: + if 'model' and objs and objs['model']: + if namespace: + for model_name, model in objs['model'].items(): + if model.ns.name == namespace: + yield model_name + else: + yield from objs['model'].keys() + else: + if namespace == '': + for model in objs['model'].values(): + if model.name.startswith('_'): + yield model.name + + if namespace: + stmt = sa.select(table.c.path).where( + sa.and_( + table.c.path.startswith(namespace), + table.c.dim == 'model' + ) + ) + else: + stmt = sa.select(table.c.path).where( + table.c.dim == 'model' + ) + rows = conn.execute(stmt) + for row in rows: + if not recursive and namespace: + # Check if path is actually right after ns, + # ex: namespace = 'dataset/test' + # models: 'dataset/test/gov/Model', 'dataset/test/Model' + # This will filter out first model, since it belongs to gov namespace + fixed_path = row['path'].replace(namespace, '') + if fixed_path.startswith('/'): + fixed_path = fixed_path[1:] + if len(fixed_path.split('/')) == 1: + yield row['path'] + else: + yield row['path'] + + +def get_namespace_name_list(context: Context, manifest: InternalSQLManifest, loaded: bool, namespace: str = None): + manifest = get_manifest(context, manifest) + table = manifest.table + conn = get_transaction_connection(context) + if conn is None or loaded: + objs = manifest.get_objects() + if 'ns' and objs and objs['ns']: + if namespace is not None: + for ns_name, ns in objs['ns'].items(): + if ns.parent and isinstance(ns.parent, Namespace) and ns.parent.name == namespace: + yield ns_name + else: + yield from objs['ns'].keys() + else: + if not namespace: + stmt = sa.select(table.c.mpath).where( + sa.or_( + table.c.dim == 'namespace', + table.c.dim == 'dataset' + ) + ).order_by(table.c.mpath) + else: + stmt = sa.select(table.c.mpath).where( + sa.and_( + table.c.mpath.startswith(namespace), + table.c.mpath != namespace, + sa.or_( + table.c.dim == 'namespace', + table.c.dim == 'dataset' + ) + ) + + ).order_by(table.c.index) + rows = conn.execute(stmt) + yielded = [] + for row in rows: + if namespace: + # Fix namespace path, ex given namespace is 'dataset/test' + # Fetched namespaces are: 'dataset/test/gov', 'dataset/test/other/gov' + # it will return 'dataset/test/gov' and 'dataset/test/other' + fixed_path = row['mpath'].replace(namespace, '') + if fixed_path.startswith('/'): + fixed_path = fixed_path[1:] + fixed_path = f'{namespace}/{fixed_path.split("/")[0]}' + if fixed_path not in yielded: + yielded.append(fixed_path) + yield fixed_path + else: + yield row['mpath'] + + +def _get_dataset_name_list(context: Context, manifest: InternalSQLManifest, loaded: bool): + manifest = get_manifest(context, manifest) + table = manifest.table + conn = get_transaction_connection(context) + if conn is None or loaded: + objs = manifest.get_objects() + if 'dataset' and objs and objs['dataset']: + yield from objs['dataset'].keys() + else: + stmt = sa.select(table.c.path).where( + table.c.dim == 'dataset' + ).order_by(table.c.path) + rows = conn.execute(stmt) + for row in rows: + yield row['path'] + + +@commands.has_model.register(Context, InternalSQLManifest, str) +def has_model(context: Context, manifest: InternalSQLManifest, model: str, loaded: bool = False, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + if model in manifest.get_objects()['model']: + return True + elif not loaded and conn is not None: + table = manifest.table + ns = conn.execute( + sa.select(table).where( + sa.and_( + table.c.path == model, + table.c.dim == 'model' + ) + ) + ) + if any(ns): + return True + return False + + +@commands.get_model.register(Context, InternalSQLManifest, str) +def get_model(context: Context, manifest: InternalSQLManifest, model: str, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + objects = manifest.get_objects() + if has_model(context, manifest, model): + if model in objects['model']: + m = objects['model'][model] + return m + elif conn is not None: + schemas = [] + table = manifest.table + m = conn.execute( + select_full_table(table).where( + sa.and_( + table.c.path == model, + table.c.dim == 'model' + ) + ).limit(1) + ) + model_obj = None + props = [] + for item in m: + model_obj = item + props = conn.execute( + select_full_table(table).where( + sa.and_( + table.c.path.startswith(model), + table.c.dim != 'model' + ) + ).order_by(table.c.index) + ) + + parent_id = model_obj['parent'] + parent_dataset = None + parent_resource = None + parent_schemas = [] + while parent_id is not None: + parent_obj = get_object_from_id(context, manifest, parent_id) + if parent_obj is None: + break + + parent_schemas.append(parent_obj) + parent_id = parent_obj['parent'] + + if parent_obj['dim'] == 'dataset': + parent_dataset = parent_obj['name'] + dataset = commands.get_dataset(context, manifest, parent_dataset) + if parent_resource: + get_dataset_resource(context, manifest, dataset, parent_resource) + break + elif parent_obj['dim'] == 'resource' and not parent_resource: + parent_resource = parent_obj['name'] + + schemas.extend(reversed(parent_schemas)) + schemas.append(model_obj) + schemas.extend(props) + required_models = [model] + + schemas = internal_to_schema(manifest, schemas) + schemas = update_schema_with_external(schemas, { + 'dataset': parent_dataset, + 'resource': parent_resource + }) + schemas = load_required_models(context, manifest, schemas, required_models) + load_internal_manifest_nodes(context, manifest, schemas, link=True, ignore_types=['dataset', 'resource']) + if model in objects['model']: + return objects['model'][model] + raise ModelNotFound(model=model) + + +@commands.get_models.register(Context, InternalSQLManifest) +def get_models(context: Context, manifest: InternalSQLManifest, loaded: bool = False, **kwargs): + model_names = get_model_name_list(context, manifest, loaded) + objs = manifest.get_objects() + for name in model_names: + # get_model loads the model if it has not been loaded + if name not in objs['model']: + commands.get_model(context, manifest, name) + return objs['model'] + + +@commands.set_model.register(Context, InternalSQLManifest, str, Model) +def set_model(context: Context, manifest: InternalSQLManifest, model_name: str, model: Model, **kwargs): + manifest.get_objects()['model'][model_name] = model + + +@commands.set_models.register(Context, InternalSQLManifest, dict) +def set_models(context: Context, manifest: InternalSQLManifest, models: Dict[str, Model], **kwargs): + manifest.get_objects()['model'] = models + + +@commands.has_namespace.register(Context, InternalSQLManifest, str) +def has_namespace(context: Context, manifest: InternalSQLManifest, namespace: str, loaded: bool = False, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + if namespace in manifest.get_objects()['ns']: + return True + elif conn is not None and not loaded: + table = manifest.table + ns = conn.execute( + sa.select(table).where(table.c.mpath.startswith(namespace)).limit(1) + ) + if any(ns): + return True + return False + + +@commands.get_namespace.register(Context, InternalSQLManifest, str) +def get_namespace(context: Context, manifest: InternalSQLManifest, namespace: str, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + objects = manifest.get_objects() + + if has_namespace(context, manifest, namespace): + if namespace in objects['ns']: + ns = objects['ns'][namespace] + return ns + elif conn is not None: + table = manifest.table + ns = conn.execute( + select_full_table(table).where( + sa.and_( + table.c.name == namespace, + table.c.dim == 'ns' + ) + ) + ) + schemas = internal_to_schema(manifest, ns) + load_internal_manifest_nodes(context, manifest, schemas, link=True) + if namespace in objects['ns']: + return objects['ns'][namespace] + + ns = load_namespace_from_name(context, manifest, namespace, drop=False) + return ns + + raise NamespaceNotFound(namespace=namespace) + + +@commands.get_namespaces.register(Context, InternalSQLManifest) +def get_namespaces(context: Context, manifest: InternalSQLManifest, loaded: bool = False, **kwargs): + ns_names = get_namespace_name_list(context, manifest, loaded) + objs = manifest.get_objects() + for name in ns_names: + # get_namespace loads the namespace if it has not been loaded + if name not in objs['ns']: + commands.get_namespace(context, manifest, name) + return objs['ns'] + + +@commands.set_namespace.register(Context, InternalSQLManifest, str, Namespace) +def set_namespace(context: Context, manifest: InternalSQLManifest, namespace: str, ns: Namespace, **kwargs): + manifest = get_manifest(context, manifest) + manifest.get_objects()['ns'][namespace] = ns + + +@commands.has_dataset.register(Context, InternalSQLManifest, str) +def has_dataset(context: Context, manifest: InternalSQLManifest, dataset: str, loaded: bool = False, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + if dataset in manifest.get_objects()['dataset']: + return True + elif conn is not None and not loaded: + table = manifest.table + ds = conn.execute( + sa.select(table).where( + sa.and_( + table.c.mpath == dataset, + table.c.dim == 'dataset' + ) + + ).limit(1) + ) + if any(ds): + return True + return False + + +def has_dataset_resource(context: Context, manifest: InternalSQLManifest, dataset: Dataset, resource: str, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + if resource in dataset.resources: + return True + elif conn is not None: + table = manifest.table + ds = conn.execute( + sa.select(table).where( + sa.and_( + table.c.path == dataset.name, + table.c.dim == 'resource', + table.c.name == resource + ) + + ).limit(1) + ) + if any(ds): + return True + return False + + +def get_dataset_resource(context: Context, manifest: InternalSQLManifest, dataset: Dataset, resource: str, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + if has_dataset_resource(context, manifest, dataset, resource, **kwargs): + if resource in dataset.resources: + return dataset.resources[resource] + elif conn is not None: + table = manifest.table + resources = conn.execute( + sa.select(table).where( + sa.and_( + table.c.path == dataset.name, + sa.or_( + sa.and_( + table.c.name == resource, + table.c.dim == 'resource' + ), + table.c.dim == 'dataset' + ) + ) + ) + ) + schemas = internal_to_schema(manifest, resources) + load_internal_manifest_nodes(context, manifest, schemas, link=True) + if resource in dataset.resources: + return dataset.resources[resource] + + +@commands.get_dataset.register(Context, InternalSQLManifest, str) +def get_dataset(context: Context, manifest: InternalSQLManifest, dataset: str, **kwargs): + manifest = get_manifest(context, manifest) + conn = get_transaction_connection(context) + objects = manifest.get_objects() + + if has_dataset(context, manifest, dataset): + if dataset in objects['dataset']: + return objects['dataset'][dataset] + elif conn is not None: + table = manifest.table + ds = conn.execute( + select_full_table(table).where( + sa.and_( + table.c.path == dataset, + table.c.dim != 'base', + table.c.dim != 'resource', + ) + ) + ) + schemas = internal_to_schema(manifest, ds) + load_internal_manifest_nodes(context, manifest, schemas, link=True) + if dataset in objects['dataset']: + return objects['dataset'][dataset] + + raise DatasetNotFound(dataset=dataset) + + +@commands.get_datasets.register(Context, InternalSQLManifest) +def get_datasets(context: Context, manifest: InternalSQLManifest, loaded: bool = False, **kwargs): + dataset_names = _get_dataset_name_list(context, manifest, loaded) + objs = manifest.get_objects() + for name in dataset_names: + # get_dataset loads the dataset if it has not been loaded + if name not in objs['dataset']: + commands.get_dataset(context, manifest, name) + return objs['dataset'] + + +@commands.set_dataset.register(Context, InternalSQLManifest, str, Dataset) +def set_dataset(context: Context, manifest: InternalSQLManifest, dataset_name: str, dataset: Dataset, **kwargs): + manifest.get_objects()['dataset'][dataset_name] = dataset + diff --git a/spinta/manifests/internal_sql/commands/read.py b/spinta/manifests/internal_sql/commands/read.py new file mode 100644 index 000000000..ce6805597 --- /dev/null +++ b/spinta/manifests/internal_sql/commands/read.py @@ -0,0 +1,144 @@ +from typing import Optional + +from starlette.requests import Request +from starlette.responses import Response +from spinta import commands +from spinta.accesslog import log_response +from spinta.backends.helpers import get_select_tree, get_select_prop_names +from spinta.compat import urlparams_to_expr +from spinta.manifests.internal_sql.commands.manifest import get_model_name_list, get_namespace_name_list +from spinta.renderer import render +from spinta.components import Context, Namespace, Action, UrlParams +from spinta.manifests.internal_sql.components import InternalSQLManifest +from spinta.manifests.internal_sql.helpers import get_namespace_partial_data +from spinta.types.namespace import check_if_model_has_backend_and_source, _model_matches_params + + +@commands.traverse_ns_models.register(Context, Namespace, InternalSQLManifest, Action) +def traverse_ns_models( + context: Context, + ns: Namespace, + manifest: InternalSQLManifest, + action: Action, + dataset_: Optional[str] = None, + resource: Optional[str] = None, + internal: bool = False, + source_check: bool = False, + loaded: bool = False, + **kwargs +): + models = get_model_name_list(context, manifest, loaded, namespace=ns.name, recursive=True) + for model_name in models: + model = commands.get_model(context, manifest, model_name) + if not (source_check and not check_if_model_has_backend_and_source(model)): + if _model_matches_params(context, model, action, dataset_, resource, internal): + yield model + + +@commands.getall.register(Context, Namespace, Request, InternalSQLManifest) +def getall( + context: Context, + ns: Namespace, + request: Request, + manifest: InternalSQLManifest, + *, + action: Action, + params: UrlParams +): + if params.all and params.ns: + # for model in traverse_ns_models(context, ns, action, internal=True): + # commands.authorize(context, action, model) + return _get_internal_ns_content( + context, + request, + ns, + manifest, + params, + action, + recursive=True + ) + elif params.all: + accesslog = context.get('accesslog') + + prepare_data_for_response_kwargs = {} + for model in commands.traverse_ns_models(context, ns, manifest, action, internal=True): + commands.authorize(context, action, model) + select_tree = get_select_tree(context, action, params.select) + prop_names = get_select_prop_names( + context, + model, + model.properties, + action, + select_tree, + ) + prepare_data_for_response_kwargs[model.model_type()] = { + 'select': select_tree, + 'prop_names': prop_names, + } + expr = urlparams_to_expr(params) + rows = commands.getall(context, ns, action=action, query=expr) + rows = ( + commands.prepare_data_for_response( + context, + commands.get_model(context, manifest, row['_type']), + params.fmt, + row, + action=action, + **prepare_data_for_response_kwargs[row['_type']], + ) + for row in rows + ) + rows = log_response(context, rows) + return render(context, request, ns, params, rows, action=action) + else: + return _get_internal_ns_content( + context, + request, + ns, + manifest, + params, + action + ) + + +def _get_internal_ns_content( + context: Context, + request: Request, + ns: Namespace, + manifest: InternalSQLManifest, + params: UrlParams, + action: Action, + *, + recursive: bool = False, +) -> Response: + parents = [parent.name for parent in ns.parents()] + partial_data = get_namespace_partial_data(context, manifest, ns.name, parents=parents, recursive=recursive, action=action) + + data = sorted(partial_data, key=lambda x: (x['_type'] != 'ns', x['name'])) + model = commands.get_model(context, ns.manifest, '_ns') + select = params.select or ['name', 'title', 'description'] + select_tree = get_select_tree(context, action, select) + prop_names = get_select_prop_names( + context, + model, + model.properties, + action, + select_tree, + auth=False, + ) + rows = ( + commands.prepare_data_for_response( + context, + model, + params.fmt, + row, + action=action, + select=select_tree, + prop_names=prop_names, + ) + for row in data + ) + + rows = log_response(context, rows) + + return render(context, request, model, params, data=rows, action=action) diff --git a/spinta/manifests/internal_sql/components.py b/spinta/manifests/internal_sql/components.py new file mode 100644 index 000000000..c3ce2543a --- /dev/null +++ b/spinta/manifests/internal_sql/components.py @@ -0,0 +1,97 @@ +import contextlib +from typing import Optional, Dict, List, Final, Literal + +from spinta.manifests.components import Manifest +import sqlalchemy as sa + + +class InternalSQLManifest(Manifest): + type = 'internal' + path: Optional[str] = None + engine: sa.engine.Engine = None + table = None + + @staticmethod + def detect_from_path(path: str) -> bool: + try: + url = sa.engine.make_url(path) + if not url: + return False + url.get_dialect() + engine = sa.create_engine(url) + inspector = sa.inspect(engine) + return inspector.has_table('_manifest') + except sa.exc.SQLAlchemyError: + return False + + @contextlib.contextmanager + def transaction(self): + with self.engine.begin() as connection: + yield Transaction(connection) + + +class Transaction: + id: str + errors: int + + def __init__(self, connection): + self.connection = connection + + +INDEX: Final = 'index' +ID: Final = 'id' +PARENT: Final = 'parent' +DEPTH: Final = 'depth' +PATH: Final = 'path' +MPATH: Final = 'mpath' +DIM: Final = 'dim' +NAME: Final = 'name' +TYPE: Final = 'type' +REF: Final = 'ref' +SOURCE: Final = 'source' +PREPARE: Final = 'prepare' +LEVEL: Final = 'level' +ACCESS: Final = 'access' +URI: Final = 'uri' +TITLE: Final = 'title' +DESCRIPTION: Final = 'description' +InternalManifestColumn = Literal[ + 'index', + 'id', + 'parent', + 'depth', + 'path', + 'mpath', + 'dim', + 'name', + 'type', + 'ref', + 'source', + 'prepare', + 'level', + 'access', + 'uri', + 'title', + 'description', +] +INTERNAL_MANIFEST_COLUMNS: List[InternalManifestColumn] = [ + INDEX, + ID, + PARENT, + DEPTH, + PATH, + MPATH, + DIM, + NAME, + TYPE, + REF, + SOURCE, + PREPARE, + LEVEL, + ACCESS, + URI, + TITLE, + DESCRIPTION, +] + +InternalManifestRow = Dict[InternalManifestColumn, str] diff --git a/spinta/manifests/internal_sql/helpers.py b/spinta/manifests/internal_sql/helpers.py new file mode 100644 index 000000000..9e9bc73ba --- /dev/null +++ b/spinta/manifests/internal_sql/helpers.py @@ -0,0 +1,1421 @@ +import uuid +from operator import itemgetter +from typing import Optional, List, Iterator, Dict, Any, Tuple, Iterable + +import sqlalchemy as sa +from sqlalchemy.sql.elements import Null + +from spinta import commands +from spinta.backends import Backend +from spinta.backends.components import BackendOrigin +from spinta.components import Namespace, Base, Model, Property, Context, Config, EntryId, MetaData, Action +from spinta.core.enums import Access +from spinta.core.ufuncs import Expr +from spinta.datasets.components import Dataset, Resource +from spinta.dimensions.comments.components import Comment +from spinta.dimensions.enum.components import Enums +from spinta.dimensions.lang.components import LangData +from spinta.dimensions.prefix.components import UriPrefix +from spinta.exceptions import InvalidIdType +from spinta.manifests.components import Manifest, ManifestSchema +from spinta.manifests.helpers import _load_manifest +from spinta.manifests.internal_sql.commands.auth import internal_authorized, get_namespace_highest_access +from spinta.manifests.internal_sql.components import InternalManifestRow, INTERNAL_MANIFEST_COLUMNS, \ + InternalManifestColumn, InternalSQLManifest +from spinta.manifests.tabular.components import ManifestRow, MANIFEST_COLUMNS +from spinta.manifests.tabular.helpers import ENUMS_ORDER_BY, sort, MODELS_ORDER_BY, DATASETS_ORDER_BY, \ + to_relative_model_name, PROPERTIES_ORDER_BY, _get_type_repr, _read_tabular_manifest_rows +from sqlalchemy_utils import UUIDType + +from spinta.nodes import get_node +from spinta.spyna import unparse +from spinta.types.datatype import Ref, Array, BackRef, Object +from spinta.types.text.components import Text +from spinta.utils.data import take +from spinta.utils.enums import get_enum_by_name +from spinta.utils.schema import NotAvailable, NA +from spinta.utils.types import is_str_uuid + + +def get_manifest(context: Context, manifest: InternalSQLManifest): + if context.has('request.manifest'): + return context.get('request.manifest') + return manifest + + +def get_transaction_connection(context: Context): + if context.has('transaction.manifest'): + return context.get('transaction.manifest').connection + return None + + +def can_return_namespace_data(context: Context, manifest: InternalSQLManifest, full_name: str, item, parents: list, action: Action): + if full_name.startswith('_'): + return False + if not internal_authorized( + context, + full_name, + get_namespace_highest_access( + context, + manifest, + full_name + ), + action, + parents + ): + return False + + return True + + +def select_full_table(table, extra_cols=None): + if extra_cols is None: + extra_cols = [] + + return sa.select([ + table, + sa.literal_column("prepare IS NULL").label("prepare_is_null")], + *extra_cols + ) + + +def read_initial_schema(context: Context, manifest: InternalSQLManifest): + conn = context.get('transaction.manifest').connection + table = manifest.table + stmt = select_full_table(table).where(table.c.path == None).order_by(table.c.index) + rows = conn.execute(stmt) + yield from internal_to_schema(manifest, rows) + + +def internal_to_schema(manifest: InternalSQLManifest, rows): + converted = convert_sql_to_tabular_rows(list(rows)) + yield from _read_tabular_manifest_rows(path=manifest.path, rows=converted, allow_updates=True) + + +def read_schema(path: str): + engine = sa.create_engine(path) + with engine.connect() as conn: + yield from _read_all_sql_manifest_rows(path, conn) + + +def get_object_from_id(context: Context, manifest: InternalSQLManifest, uid): + conn = context.get('transaction.manifest').connection + table = manifest.table + results = conn.execute(select_full_table(table).where( + table.c.id == uid + ).limit(1)) + for item in results: + return item + return None + + +def update_schema_with_external(schema, external: dict): + for id_, item in schema: + if item['type'] == 'model': + if external['dataset']: + if '/' not in item['name']: + item['name'] = f'{external["dataset"]}/{item["name"]}' + item['external']['dataset'] = external['dataset'] + if external['resource']: + item['external']['resource'] = external['resource'] + if item['properties']: + for prop in item['properties'].values(): + if 'model' in prop and prop['model']: + if '/' not in prop['model'] and external['dataset']: + prop['model'] = f'{external["dataset"]}/{prop["model"]}' + if item['base']: + base = item['base'] + if base['parent']: + if '/' not in base['parent'] and external['dataset']: + base['parent'] = f'{external["dataset"]}/{base["parent"]}' + yield id_, item + + +def _traverse_properties(prop: dict, ds: str): + if prop['type'] not in ['backref', 'array_backref']: + if 'model' in prop: + model_name = prop['model'] + if '/' in model_name or not ds: + yield model_name + else: + yield f'{ds}/{model_name}' + elif 'items' in prop: + yield from _traverse_properties(prop['items'], ds) + elif 'properties' in prop and isinstance(prop['properties'], dict): + for prop_ in prop['properties'].values(): + yield from _traverse_properties(prop_, ds) + + +def load_required_models(context: Context, manifest: InternalSQLManifest, schema, model_list: list): + for id_, item in schema: + if item['type'] == 'model': + ds = item['external']['dataset'] if item['external'] and item['external']['dataset'] else None + if item['properties']: + for prop in item['properties'].values(): + for model_prop in _traverse_properties(prop, ds): + if model_prop not in model_list: + model_list.append(model_prop) + commands.get_model(context, manifest, model_prop) + if item['base']: + if item['base']['parent'] not in model_list: + model_list.append(item['base']['parent']) + commands.get_model(context, manifest, item['base']['parent']) + yield id_, item + + +def get_namespace_partial_data( + context: Context, + manifest: InternalSQLManifest, + namespace: str, + parents: list, + action: Action, + recursive: bool = False, +): + conn = context.get('transaction.manifest').connection + table = manifest.table + parents = parents.copy() + parents.append(namespace) + + results = conn.execute(select_full_table(table).where( + sa.and_( + table.c.mpath.startswith(namespace), + table.c.mpath != namespace, + sa.or_( + table.c.dim == 'ns', + table.c.dim == 'dataset', + table.c.dim == 'model' + ) + ) + ).order_by(table.c.mpath)) + result = [] + recursive_list = [] + for item in results: + item = item._asdict() + if item['path'] == namespace or item['mpath'] == namespace: + continue + + type_ = 'ns' + if item['dim'] == 'ns' or item['dim'] == 'dataset': + name = item['name'] + else: + type_ = 'model' + name = item['path'] + + name = name[len(namespace):] + if name[0] == '/': + name = name[1:] + split = name.split('/') + full_name = f'{namespace}/{split[0]}' if namespace else split[0] + if len(split) == 1: + result.append(split[0]) + if can_return_namespace_data(context, manifest, full_name, item, parents, action=action): + if recursive and type_ == 'ns' and full_name not in recursive_list: + recursive_list.append(full_name) + yield { + '_type': type_, + 'name': f'{full_name}/:ns' if type_ == 'ns' else full_name, + 'title': item['title'] or '', + 'description': item['description'] or '' + } + elif split[0] not in result: + result.append(split[0]) + if can_return_namespace_data(context, manifest, full_name, item, parents, action=action): + if recursive and full_name not in recursive_list: + recursive_list.append(full_name) + yield { + '_type': 'ns', + 'name': f'{full_name}/:ns', + 'title': '', + 'description': '' + } + + if recursive and recursive_list: + for item in recursive_list: + yield from get_namespace_partial_data( + context, + manifest, + item, + recursive=recursive, + parents=parents, + action=action + ) + + +def load_internal_manifest_nodes( + context: Context, + manifest: InternalSQLManifest, + schemas: Iterable[ManifestSchema], + *, + link: bool = False, + ignore_types: list = [] +) -> None: + to_link = [] + config = context.get('config') + for eid, schema in schemas: + if schema.get('type') == 'manifest': + _load_manifest(context, manifest, schema, eid, reset=False) + else: + node = _load_internal_manifest_node(context, config, manifest, None, eid, schema) + if node.type not in ignore_types: + commands.set_node(context, manifest, node.type, node.name, node) + if link: + to_link.append(node) + + if to_link: + for node in to_link: + commands.link(context, node) + + +def _load_internal_manifest_node( + context: Context, + config: Config, + manifest: Manifest, + source: Optional[Manifest], + eid: EntryId, + data: dict, +) -> MetaData: + node = get_node(context, config, manifest, eid, data, check=False) + node.eid = eid + node.type = data['type'] + node.parent = manifest + node.manifest = manifest + commands.load(context, node, data, manifest, source=source) + return node + + +def load_internal_namespace_from_name( + context: Context, + manifest: InternalSQLManifest, + path: str, + *, + # Drop last element from path which is usually a model name. + drop: bool = True, +) -> Namespace: + ns: Optional[Namespace] = None + parent: Optional[Namespace] = None + parts: List[str] = [] + parts_ = [p for p in path.split('/') if p] + if drop: + parts_ = parts_[:-1] + objects = manifest.get_objects() + for part in parts_: + parts.append(part) + name = '/'.join(parts) + + if name not in objects['ns']: + ns = Namespace() + data = { + 'type': 'ns', + 'name': name, + 'title': '', + 'description': '', + } + commands.load(context, ns, data, manifest) + ns.generated = True + else: + ns = objects['ns'][name] + pass + + if parent: + if ns.name == parent.name: + raise RuntimeError(f"Self reference in {path!r}.") + + ns.parent = parent or manifest + + if part and part not in parent.names: + parent.names[part] = ns + else: + ns.parent = manifest + + parent = ns + + return ns + + +def get_table_structure(meta: sa.MetaData): + table = sa.Table( + '_manifest', + meta, + sa.Column("index", sa.BIGINT), + sa.Column("id", UUIDType, primary_key=True), + sa.Column("parent", UUIDType), + sa.Column("depth", sa.Integer), + sa.Column("path", sa.String), + sa.Column("mpath", sa.String), + sa.Column("dim", sa.String), + sa.Column("name", sa.String), + sa.Column("type", sa.String), + sa.Column("ref", sa.String), + sa.Column("source", sa.String), + sa.Column("prepare", sa.JSON), + sa.Column("level", sa.Integer), + sa.Column("access", sa.String), + sa.Column("uri", sa.String), + sa.Column("title", sa.String), + sa.Column("description", sa.String), + ) + return table + + +def _read_all_sql_manifest_rows( + path: Optional[str], + conn: sa.engine.Connection, + *, + rename_duplicates: bool = True +): + meta = sa.MetaData(conn) + table = get_table_structure(meta) + full_table = select_full_table(table) + stmt = full_table.order_by( + table.c.index + ) + rows = conn.execute(stmt) + converted = convert_sql_to_tabular_rows(list(rows)) + yield from _read_tabular_manifest_rows(path=path, rows=converted, rename_duplicates=rename_duplicates) + + +def write_internal_sql_manifest(context: Context, dsn: str, manifest: Manifest): + engine = sa.create_engine(dsn) + inspect = sa.inspect(engine) + with engine.connect() as conn: + meta = sa.MetaData(conn) + table = get_table_structure(meta) + if inspect.has_table('_manifest'): + conn.execute(table.delete()) + else: + table.create() + rows = datasets_to_sql(context, manifest) + index = 0 + for row in rows: + row['index'] = index + index += 1 + conn.execute(table.insert().values(row)) + + +def _handle_id(item_id: Any): + if item_id: + if is_str_uuid(item_id): + return uuid.UUID(item_id, version=4) + elif isinstance(item_id, uuid.UUID): + return item_id + else: + raise InvalidIdType(id=item_id, id_type=type(item_id)) + return uuid.uuid4() + + +def datasets_to_sql( + context: Context, + manifest: Manifest, + *, + external: bool = True, # clean content of source and prepare + access: Access = Access.private, + internal: bool = False, # internal models with _ prefix like _txn + order_by: InternalManifestColumn = None, +) -> Iterator[InternalManifestRow]: + yield from _prefixes_to_sql(manifest.prefixes) + yield from _backends_to_sql(manifest.backends) + yield from _namespaces_to_sql(commands.get_namespaces(context, manifest)) + yield from _enums_to_sql( + manifest.enums, + external=external, + access=access, + order_by=order_by, + ) + + seen_datasets = set() + dataset = { + "id": None, + "path": None, + "mpath": None, + "item": None, + "depth": 0 + } + resource = { + "id": None, + "path": None, + "mpath": None, + "item": None, + "depth": 0 + } + base = { + "id": None, + "path": None, + "mpath": None, + "item": None, + "depth": 0 + } + models = commands.get_models(context, manifest) + models = models if internal else take(models) + models = sort(MODELS_ORDER_BY, models.values(), order_by) + + for model in models: + if model.access < access: + continue + + if model.external: + if dataset["item"] is None or (model.external.dataset and dataset["item"].name != model.external.dataset.name): + dataset["item"] = model.external.dataset + if dataset["item"]: + seen_datasets.add(dataset["item"].name) + resource["item"] = None + for item in _dataset_to_sql( + dataset["item"], + external=external, + access=access, + order_by=order_by, + ): + yield item + if item["dim"] == "dataset": + dataset["id"] = item["id"] + dataset["path"] = item["path"] + dataset["mpath"] = item["mpath"] + dataset["depth"] = item["depth"] + if not model.external.dataset: + dataset["item"] = None + resource["item"] = None + base["item"] = None + + if external and model.external.resource and ( + resource["item"] is None or + resource["item"].name != model.external.resource.name + ): + resource["item"] = model.external.resource + if resource["item"]: + parent_id = None + depth = 0 + path = '' + mpath = '' + if dataset["item"]: + parent_id = dataset["id"] + depth = dataset["depth"] + 1 + path = dataset["path"] + mpath = dataset["mpath"] + for item in _resource_to_sql( + resource["item"], + external=external, + access=access, + parent_id=parent_id, + path=path, + mpath=mpath, + depth=depth, + ): + yield item + if item["dim"] == "resource": + resource["id"] = item["id"] + resource["path"] = item["path"] + resource["mpath"] = item["mpath"] + resource["depth"] = item["depth"] + elif external: + if not model.external.resource: + resource["item"] = None + base["item"] = None + + if model.base and (not base["item"] or model.base.name != base["item"].name): + base["item"] = model.base + parent_id = None + depth = 0 + path = '' + mpath = '' + if resource["item"]: + parent_id = resource["id"] + depth = resource["depth"] + 1 + path = resource["path"] + mpath = resource["mpath"] + elif dataset["item"]: + parent_id = dataset["id"] + depth = dataset["depth"] + 1 + path = dataset["path"] + mpath = dataset["mpath"] + for item in _base_to_sql( + base=base["item"], + parent_id=parent_id, + depth=depth, + path=path, + mpath=mpath, + ): + yield item + if item["dim"] == "base": + base["id"] = item["id"] + base["path"] = item["path"] + base["mpath"] = item["mpath"] + base["depth"] = item["depth"] + elif base["item"] and not model.base: + base["item"] = None + parent_id = None + depth = 0 + path = dataset["path"] if dataset["item"] else "" + mpath = '' + if base["item"]: + parent_id = base["id"] + depth = base["depth"] + 1 + mpath = base["mpath"] + elif resource["item"]: + parent_id = resource["id"] + depth = resource["depth"] + 1 + mpath = resource["mpath"] + elif dataset["item"]: + parent_id = dataset["id"] + depth = dataset["depth"] + 1 + mpath = dataset["mpath"] + yield from _model_to_sql( + model, + external=external, + access=access, + order_by=order_by, + parent_id=parent_id, + depth=depth, + path=path, + mpath=mpath + ) + + datasets = sort(DATASETS_ORDER_BY, commands.get_datasets(context, manifest).values(), order_by) + for dataset in datasets: + if dataset.name in seen_datasets: + continue + parent_id = None + depth = 0 + path = '' + mpath = '' + for item in _dataset_to_sql( + dataset, + external=external, + access=access, + order_by=order_by, + ): + yield item + if item["dim"] == "dataset": + parent_id = item["id"] + depth = int(item["depth"]) + 1 + path = item["path"] + mpath = item["mpath"] + + for resource in dataset.resources.values(): + yield from _resource_to_sql( + resource, + parent_id=parent_id, + depth=depth, + path=path, + mpath=mpath + ) + + +def _prefixes_to_sql( + prefixes: Dict[str, UriPrefix], + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + for name, prefix in prefixes.items(): + item_id = _handle_id(prefix.id) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, name] if mpath else [name]), + 'dim': 'prefix', + 'name': name, + 'type': prefix.type, + 'ref': name, + 'uri': prefix.uri, + 'title': prefix.title, + 'description': prefix.description, + 'prepare': _handle_prepare(NA) + }) + + +def _namespaces_to_sql( + namespaces: Dict[str, Namespace], + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + namespaces = { + k: ns + for k, ns in namespaces.items() if not ns.generated + } + + for name, ns in namespaces.items(): + item_id = _handle_id(ns.id) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, name] if mpath else [name]), + 'dim': 'ns', + 'name': name, + 'type': ns.type, + 'ref': name, + 'title': ns.title, + 'description': ns.description, + 'prepare': _handle_prepare(NA) + }) + + +def _enums_to_sql( + enums: Optional[Enums], + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None, + external: bool = True, + access: Access = Access.private, + order_by: InternalManifestColumn = None, +) -> Iterator[InternalManifestRow]: + if enums is None: + return + for name, enum in enums.items(): + items = sort(ENUMS_ORDER_BY, enum.values(), order_by) + new_parent_id = _handle_id("") + mpath_name = name if name else str(new_parent_id) + new_mpath = '/'.join([mpath, mpath_name] if mpath else [mpath_name]) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': new_parent_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': new_mpath, + 'dim': 'enum', + 'name': name, + 'type': 'enum', + 'ref': name, + 'prepare': _handle_prepare(NA) + }) + for item in items: + if item.access is not None and item.access < access: + continue + new_item_id = _handle_id(item.id) + new_item_mpath = '/'.join([new_mpath, str(new_item_id)] if new_mpath else [str(new_item_id)]) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': new_item_id, + 'parent': new_parent_id, + 'depth': depth + 1, + 'path': path, + 'mpath': new_item_mpath, + 'dim': 'enum.item', + 'source': item.source if external else None, + 'prepare': _handle_prepare(item.prepare), + 'access': item.given.access, + 'title': item.title, + 'description': item.description, + }) + yield from _lang_to_sql(item.lang, path=path, mpath=new_item_mpath, depth=depth + 2, parent_id=new_item_id) + + +def _lang_to_sql( + lang: Optional[LangData], + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None, +) -> Iterator[InternalManifestRow]: + if lang is None: + return + for name, data in sorted(lang.items(), key=itemgetter(0)): + item_id = _handle_id(data.get("id")) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, name] if mpath else [name]), + 'dim': 'lang', + 'name': name, + 'type': 'lang', + 'ref': name, + 'title': data['title'], + 'description': data['description'], + 'prepare': _handle_prepare(NA) + }) + + +def _comments_to_sql( + comments: Optional[List[Comment]], + access: Access = Access.private, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + if comments is None: + return + for comment in comments: + if comment.access < access: + return + new_id = _handle_id(comment.id) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': new_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, str(new_id)] if mpath else [str(new_id)]), + 'name': comment.parent if comment.parent else None, + 'dim': 'comment', + 'type': 'comment', + 'ref': comment.parent, + 'source': comment.author, + 'access': comment.given.access, + 'title': comment.created, + 'description': comment.comment, + 'prepare': _handle_prepare(NA) + }) + + +def _backends_to_sql( + backends: Dict[str, Backend], + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + for name, backend in backends.items(): + new_id = _handle_id(backend.config.get("id")) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': new_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, name] if mpath else [name]), + 'dim': 'resource', + 'name': name, + 'type': backend.type, + 'source': backend.config.get('dsn'), + 'prepare': _handle_prepare(NA) + }) + + +def _dataset_to_sql( + dataset: Dataset, + parent_id: uuid.UUID = None, + path: str = None, + mpath: str = None, + depth: int = 0, + external: bool = True, + access: Access = Access.private, + order_by: InternalManifestColumn = None, +) -> Iterator[InternalManifestRow]: + dataset_id = _handle_id(dataset.id) + new_path = '/'.join([path, dataset.name] if path else [dataset.name]) + new_mpath = '/'.join([mpath, dataset.name] if mpath else [dataset.name]) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': dataset_id, + 'parent': parent_id, + 'depth': depth, + 'path': new_path, + 'mpath': new_mpath, + 'dim': 'dataset', + 'name': dataset.name, + 'dataset': dataset.name, + 'level': dataset.level, + 'access': dataset.given.access, + 'title': dataset.title, + 'description': dataset.description, + 'prepare': _handle_prepare(NA) + }) + yield from _lang_to_sql(dataset.lang, parent_id=dataset_id, depth=depth + 1, path=new_path, mpath=new_mpath) + yield from _prefixes_to_sql(dataset.prefixes, parent_id=dataset_id, depth=depth + 1, path=new_path, mpath=new_mpath) + yield from _enums_to_sql( + dataset.ns.enums, + external=external, + access=access, + order_by=order_by, + parent_id=dataset_id, + depth=depth + 1, + path=new_path, + mpath=new_mpath + ) + + +def _params_to_sql( + params_data: dict, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + if not params_data: + return + for param, values in params_data.items(): + param_base_id = _handle_id("") + new_mpath = '/'.join([mpath, param] if mpath else [param]) + for i in range(len(values["source"])): + new_id = _handle_id('') + if "id" in values.keys(): + new_id = _handle_id(values["id"][i]) + prepare = _handle_prepare(values["prepare"][i]) + if not (isinstance(values["prepare"][i], NotAvailable) and values['source'][i] is None): + if i == 0: + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': param_base_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': new_mpath, + 'dim': 'param', + 'name': param, + 'type': 'param', + 'ref': param, + 'source': values["source"][i], + 'title': values["title"], + 'description': values["description"] + }) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': new_id, + 'parent': param_base_id, + 'depth': depth + 1, + 'path': path, + 'mpath': '/'.join([new_mpath, str(new_id)] if new_mpath else [str(new_id)]), + 'dim': 'param.item', + 'source': values["source"][i], + 'prepare': prepare + }) + + +def _resource_to_sql( + resource: Resource, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None, + external: bool = True, + access: Access = Access.private, +) -> Iterator[InternalManifestRow]: + backend = resource.backend + new_mpath = '/'.join([mpath, resource.name] if mpath else [resource.name]) + item_id = _handle_id(resource.id) + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': new_mpath, + 'dim': 'resource', + 'name': resource.name, + 'source': resource.external if external else None, + 'prepare': _handle_prepare(resource.prepare), + 'type': resource.type, + 'ref': ( + backend.name + if ( + external and + backend and + backend.origin != BackendOrigin.resource + ) + else None + ), + 'level': resource.level, + 'access': resource.given.access, + 'title': resource.title, + 'description': resource.description, + }) + yield from _params_to_sql(resource.params, parent_id=item_id, depth=depth + 1, path=path, mpath=new_mpath) + yield from _comments_to_sql(resource.comments, access=access, parent_id=item_id, depth=depth + 1, path=path, + mpath=new_mpath) + yield from _lang_to_sql(resource.lang, parent_id=item_id, depth=depth + 1, path=path, mpath=new_mpath) + + +def _base_to_sql( + base: Base, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + item_id = _handle_id(base.id) + new_path = '/'.join([path, base.name] if path else [base.name]) + new_mpath = '/'.join([mpath, base.name] if mpath else [base.name]) + data = { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': new_path, + 'mpath': new_mpath, + 'dim': 'base', + 'name': base.name, + 'level': base.level.value if base.level else None, + 'prepare': _handle_prepare(NA) + } + if base.pk: + data['ref'] = ', '.join([pk.place for pk in base.pk]) + yield to_row(INTERNAL_MANIFEST_COLUMNS, data) + yield from _lang_to_sql(base.lang, parent_id=item_id, depth=depth + 1, path=new_path, mpath=new_mpath) + + +def _model_to_sql( + model: Model, + external: bool = True, + access: Access = Access.private, + order_by: InternalManifestColumn = None, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + item_id = _handle_id(model.id) + name = model.name + if model.external and model.external.dataset: + name = to_relative_model_name( + model, + model.external.dataset, + ) + new_mpath = '/'.join([mpath, name] if mpath else [name]) + new_path = '/'.join([path, name] if path else [name]) + data = { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': new_path, + 'mpath': new_mpath, + 'dim': 'model', + 'name': name, + 'level': model.level.value if model.level else None, + 'access': model.given.access, + 'title': model.title, + 'description': model.description, + 'uri': model.uri if model.uri else None, + } + + if external and model.external: + data.update({ + 'source': model.external.name, + 'prepare': _handle_prepare(model.external.prepare), + }) + if ( + not model.external.unknown_primary_key and + all(p.access >= access for p in model.external.pkeys) + ): + # Add `ref` only if all properties are available in the + # resulting manifest. + data['ref'] = ', '.join([ + p.name for p in model.external.pkeys + ]) + + hide_list = [] + if model.external: + if not model.external.unknown_primary_key: + hide_list = [model.external.pkeys] + yield to_row(INTERNAL_MANIFEST_COLUMNS, data) + yield from _params_to_sql(model.params, parent_id=item_id, depth=depth + 1, path=new_path, mpath=new_mpath) + yield from _comments_to_sql(model.comments, access=access, parent_id=item_id, depth=depth + 1, path=new_path, + mpath=new_mpath) + yield from _lang_to_sql(model.lang, parent_id=item_id, depth=depth + 1, path=new_path, mpath=new_mpath) + yield from _unique_to_sql(model.unique, hide_list, parent_id=item_id, depth=depth + 1, path=new_path, + mpath=new_mpath) + + props = sort(PROPERTIES_ORDER_BY, model.properties.values(), order_by) + for prop in props: + yield from _property_to_sql( + prop, + external=external, + access=access, + order_by=order_by, + parent_id=item_id, + depth=depth + 1, + path=new_path, + mpath=new_mpath + ) + + +def _unique_to_sql( + model_unique_data, + hide_list: List, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + if not model_unique_data: + return + for row in model_unique_data: + if row not in hide_list: + item_id = _handle_id("") + yield to_row(INTERNAL_MANIFEST_COLUMNS, { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': path, + 'mpath': '/'.join([mpath, str(item_id)] if mpath else [item_id]), + 'dim': 'unique', + 'type': 'unique', + 'ref': ', '.join([r.name for r in row]), + 'prepare': _handle_prepare(NA) + }) + + +def _property_to_sql( + prop: Property, + external: bool = True, + access: Access = Access.private, + order_by: InternalManifestColumn = None, + parent_id: uuid.UUID = None, + depth: int = 0, + path: str = None, + mpath: str = None +) -> Iterator[InternalManifestRow]: + if prop.name.startswith('_'): + return + + if prop.access and prop.access < access: + return + + item_id = _handle_id(prop.id) + new_path = '/'.join([path, prop.name] if path else [prop.name]) + new_mpath = '/'.join([mpath, prop.name] if mpath else [prop.name]) + type_ = _get_type_repr(prop.dtype) + data = { + 'id': item_id, + 'parent': parent_id, + 'depth': depth, + 'path': new_path, + 'mpath': new_mpath, + 'dim': 'property', + 'name': prop.given.name, + 'type': type_, + 'level': prop.level.value if prop.level else None, + 'access': prop.given.access, + 'uri': prop.uri, + 'title': prop.title, + 'description': prop.description, + } + + if external and prop.external: + if isinstance(prop.external, list): + # data['source'] = ', '.join(x.name for x in prop.external) + # data['prepare'] = ', '.join( + # unparse(x.prepare or NA) + # for x in prop.external if x.prepare + # ) + raise DeprecationWarning( + "Source can't be a list, use prepare instead." + ) + elif prop.external: + data['source'] = prop.external.name + data['prepare'] = _handle_prepare(prop.external.prepare) + yield_rows = [] + if isinstance(prop.dtype, Array): + yield_array_row = prop.dtype.items + yield_rows.append(yield_array_row) + if isinstance(prop.dtype, Ref): + model = prop.model + if model.external and model.external.dataset: + data['ref'] = prop.dtype.model.name + pkeys = prop.dtype.model.external.pkeys + rkeys = prop.dtype.refprops + if rkeys and pkeys != rkeys: + rkeys = ', '.join([p.place for p in rkeys]) + data['ref'] += f'[{rkeys}]' + else: + data['ref'] = prop.dtype.model.name + + if prop.dtype.properties: + for obj_prop in prop.dtype.properties.values(): + yield_rows.append(obj_prop) + elif isinstance(prop.dtype, BackRef): + model = prop.model + if model.external and model.external.dataset: + data['ref'] = prop.dtype.model.name + rkey = prop.dtype.refprop.place + if prop.dtype.explicit: + data['ref'] += f'[{rkey}]' + else: + data['ref'] = prop.dtype.model.name + elif isinstance(prop.dtype, Object): + for obj_prop in prop.dtype.properties.values(): + yield_rows.append(obj_prop) + elif isinstance(prop.dtype, Text): + for lang_prop in prop.dtype.langs.values(): + yield_rows.append(lang_prop) + + elif prop.enum is not None: + data['ref'] = prop.given.enum + elif prop.unit is not None: + data['ref'] = prop.given.unit + + if data['name']: + yield to_row(INTERNAL_MANIFEST_COLUMNS, data) + yield from _comments_to_sql(prop.comments, access=access, parent_id=item_id, depth=depth + 1, path=new_path, + mpath=new_mpath) + yield from _lang_to_sql(prop.lang, parent_id=item_id, depth=depth + 1, path=new_path, mpath=new_mpath) + yield from _enums_to_sql( + prop.enums, + external=external, + access=access, + order_by=order_by, + parent_id=item_id, + depth=depth + 1, + path=new_path, + mpath=new_mpath + ) + if yield_rows: + for yield_row in yield_rows: + if yield_row: + yield from _property_to_sql(yield_row, external=external, access=access, order_by=order_by, parent_id=item_id, depth=depth + 1, path=new_path, mpath=new_mpath) + + +def _value_or_null(value: Any): + if isinstance(value, Null) or value or value is False or value == 0: + return value + return None + + +def _value_or_empty(value: Any): + if isinstance(value, Null) or value or value is False or value == 0: + return value + return '' + + +def to_row(keys, values) -> InternalManifestRow: + return {k: _value_or_null(values.get(k)) for k in keys} + + +def to_row_tabular(keys, values) -> ManifestRow: + result = {} + for key in keys: + if key == "prepare" and values.get("prepare_is_null") is not None and not values.get("prepare_is_null"): + result[key] = unparse(values.get("prepare")) + else: + result[key] = _value_or_empty(values.get(key)) + return result + + +def _handle_prepare(prepare: Any): + if isinstance(prepare, NotAvailable): + prepare = sa.null() + elif isinstance(prepare, Expr): + prepare = prepare.todict() + return prepare + + +def convert_sql_to_tabular_rows(rows: list) -> Iterator[Tuple[str, List[str]]]: + previous_row = {} + data_row = {} + enum_data = {} + param_data = {} + meta_dimensions = { + "current": None, + "previous": None, + 'dataset': None, + 'resource': None, + 'base': None + } + yield 1, MANIFEST_COLUMNS + line = 2 + for i, row in enumerate(rows): + row = dict(row) + _update_meta_dimensions(meta_dimensions, row) + is_first = row["dim"] != previous_row.get("dim") + base, resource, dataset = _requires_end_marker(row, meta_dimensions) + if dataset: + yield line, list(to_row_tabular(MANIFEST_COLUMNS, {'dataset': '/'}).values()) + line += 1 + elif resource: + yield line, list(to_row_tabular(MANIFEST_COLUMNS, {'resource': '/'}).values()) + line += 1 + elif base: + yield line, list(to_row_tabular(MANIFEST_COLUMNS, {'base': '/'}).values()) + line += 1 + if _requires_seperator(row, previous_row, meta_dimensions): + yield line, list(to_row_tabular(MANIFEST_COLUMNS, {}).values()) + line += 1 + dimension = row["dim"] + if dimension == "prefix": + data_row = _convert_prefixes(row, is_first) + elif dimension == "ns": + data_row = _convert_namespaces(row, is_first) + elif dimension == "lang": + data_row = _convert_lang(row, is_first) + elif dimension == "comment": + data_row = _convert_comment(row, is_first) + elif dimension == "resource": + data_row = _convert_resource(row) + elif dimension == "enum": + enum_data = _convert_enum(row) + elif dimension == "enum.item": + data_row = _convert_enum_item(row, enum_data, is_first) + elif dimension == "unique": + data_row = _convert_unique(row) + elif dimension == "param": + param_data = row + elif dimension == "param.item": + data_row = _convert_param(row, param_data, is_first) + elif dimension == "dataset": + data_row = _convert_dataset(row) + elif dimension == "base": + data_row = _convert_base(row) + elif dimension == "property": + data_row = _convert_property(row) + elif dimension == "model": + data_row = _convert_model(row) + previous_row = row + if dimension != "enum" and dimension != "param": + yield line, list(data_row.values()) + line += 1 + + +def _update_meta_dimensions(meta_dimensions: dict, row: InternalManifestRow): + if row["dim"] == "dataset": + meta_dimensions["dataset"] = row["depth"] + meta_dimensions["resource"] = None + meta_dimensions["base"] = None + elif row["dim"] == "resource": + meta_dimensions["resource"] = row["depth"] + meta_dimensions["base"] = None + elif row["dim"] == "base": + meta_dimensions["base"] = row["depth"] + if meta_dimensions["current"] != row["dim"] and row["dim"] in [ + "dataset", "resource", "base", "model", "property" + ]: + meta_dimensions["previous"] = meta_dimensions["current"] + meta_dimensions["current"] = row["dim"] + + +def _requires_end_marker(row: InternalManifestRow, meta_dimensions: dict): + base_end_marker = False + resource_end_marker = False + dataset_end_marker = False + if row["dim"] not in ["dataset", "resource", "base", "property", "enum.item", "param.item"]: + depth = row["depth"] + if meta_dimensions["base"] is not None: + if depth <= meta_dimensions["base"]: + meta_dimensions["base"] = None + base_end_marker = True + if meta_dimensions["resource"]: + if depth <= meta_dimensions["resource"] is not None: + meta_dimensions["resource"] = None + resource_end_marker = True + if meta_dimensions["dataset"] is not None: + if depth <= meta_dimensions["dataset"]: + meta_dimensions["dataset"] = None + dataset_end_marker = True + return base_end_marker, resource_end_marker, dataset_end_marker + + +def _requires_seperator(row: InternalManifestRow, previous_row: InternalManifestRow, meta_dimensions: dict): + primary_list = (None, "dataset", "resource", "base", "model", "property") + + def is_primary_dimension(dimension: str): + if dimension in primary_list: + return primary_list.index(dimension) + return -1 + + if row["dim"] == previous_row.get("dim"): + return False + + previous_dim = is_primary_dimension(previous_row.get("dim")) + current_dim = is_primary_dimension(row["dim"]) + if previous_row.get("dim") == "base": + return False + elif row["dim"] == "resource": + return False + elif current_dim > previous_dim != -1: + return False + elif current_dim == -1 and previous_dim != -1: + return False + elif current_dim == -1 and previous_dim == -1: + if previous_row.get("dim") == "lang" or row.get("dim") == "lang": + return False + if (previous_row.get("dim") == "enum" and row["dim"] == "enum.item") or (previous_row.get("dim") == "param" and row["dim"] == "param.item"): + return False + elif current_dim != -1 and primary_list[current_dim] == "property" and previous_dim == -1: + return False + elif previous_dim == -1 and meta_dimensions["previous"] == meta_dimensions["current"]: + return False + return True + + +def _convert_model(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["model"] = row["name"] + return new + + +def _convert_property(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["property"] = row["name"] + return new + + +def _convert_base(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["base"] = row["name"] + return new + + +def _convert_dataset(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["dataset"] = row["name"] + return new + + +def _convert_param(row: InternalManifestRow, param_data: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + new["ref"] = '' + new["title"] = '' + new["description"] = '' + else: + new["type"] = 'param' + new["ref"] = _value_or_empty(param_data["ref"]) + new["title"] = _value_or_empty(param_data["title"]) + new["description"] = _value_or_empty(param_data["description"]) + return new + + +def _convert_unique(row: InternalManifestRow): + return to_row_tabular(MANIFEST_COLUMNS, row) + + +def _convert_enum_item(row: InternalManifestRow, enum_data: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + new["ref"] = '' + else: + new["type"] = "enum" + new["ref"] = enum_data["ref"] + return new + + +def _convert_resource(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["resource"] = row["name"] + return new + + +def _convert_comment(row: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + return new + + +def _convert_lang(row: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + new["ref"] = '' + return new + + +def _convert_namespaces(row: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + return new + + +def _convert_prefixes(row: InternalManifestRow, first: bool = False): + new = to_row_tabular(MANIFEST_COLUMNS, row) + if not first: + new["type"] = '' + return new + + +def _convert_enum(row: InternalManifestRow): + new = to_row_tabular(MANIFEST_COLUMNS, row) + new["ref"] = _value_or_empty(new["ref"]) + return new diff --git a/spinta/manifests/memory/commands/load.py b/spinta/manifests/memory/commands/load.py index 76975e8b4..e2054aabc 100644 --- a/spinta/manifests/memory/commands/load.py +++ b/spinta/manifests/memory/commands/load.py @@ -14,12 +14,13 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) for source in manifest.sync: commands.load( @@ -28,4 +29,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/manifests/rdf/commands/load.py b/spinta/manifests/rdf/commands/load.py index e483f9023..2ca7c5cc1 100644 --- a/spinta/manifests/rdf/commands/load.py +++ b/spinta/manifests/rdf/commands/load.py @@ -15,12 +15,13 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) if manifest.path is None: return @@ -40,4 +41,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/manifests/sql/commands/load.py b/spinta/manifests/sql/commands/load.py index 98ca93c99..615d02596 100644 --- a/spinta/manifests/sql/commands/load.py +++ b/spinta/manifests/sql/commands/load.py @@ -19,6 +19,7 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): assert freezed, ( "SqlManifest does not have unfreezed version of manifest." @@ -26,9 +27,9 @@ def load( if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) dataset_name = context.get("rc").get("given_dataset_name") schemas = read_schema(context, manifest.path, manifest.prepare, dataset_name) @@ -56,4 +57,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/manifests/sql/components.py b/spinta/manifests/sql/components.py index b3279ebe0..c4f8f084a 100644 --- a/spinta/manifests/sql/components.py +++ b/spinta/manifests/sql/components.py @@ -14,7 +14,11 @@ class SqlManifest(Manifest): def detect_from_path(path: str) -> bool: try: url = sa.engine.make_url(path) + if not url: + return False url.get_dialect() - return True - except: + engine = sa.create_engine(url) + inspector = sa.inspect(engine) + return not inspector.has_table('_manifest') + except sa.exc.SQLAlchemyError: return False diff --git a/spinta/manifests/tabular/commands/load.py b/spinta/manifests/tabular/commands/load.py index e6789be1d..e49556674 100644 --- a/spinta/manifests/tabular/commands/load.py +++ b/spinta/manifests/tabular/commands/load.py @@ -21,6 +21,7 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load=False ): assert freezed, ( "TabularManifest does not have unfreezed version of manifest." @@ -28,7 +29,7 @@ def load( if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') commands.load(context, store.internal, into=target) @@ -77,4 +78,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/manifests/tabular/components.py b/spinta/manifests/tabular/components.py index 20d798203..4c45868ff 100644 --- a/spinta/manifests/tabular/components.py +++ b/spinta/manifests/tabular/components.py @@ -138,6 +138,7 @@ class DatasetRow(TypedDict, total=False): class ResourceRow(ManifestRow): + id: str backend: str external: str lang: LangData @@ -145,6 +146,7 @@ class ResourceRow(ManifestRow): class BackendRow(TypedDict, total=False): + id: str type: str name: str dsn: str @@ -153,6 +155,7 @@ class BackendRow(TypedDict, total=False): class BaseRow(TypedDict, total=False): + id: str name: str model: str pk: List[str] @@ -161,6 +164,7 @@ class BaseRow(TypedDict, total=False): class ParamRow(TypedDict): + id: str name: str # param name source: List[str] # list of `self` for prepare formulas prepare: List[Any] # list of formulas @@ -198,6 +202,7 @@ class ModelExternalRow(TypedDict, total=False): class EnumRow(TypedDict, total=False): + id: str name: str source: str prepare: Optional[Dict[str, Any]] @@ -208,6 +213,7 @@ class EnumRow(TypedDict, total=False): class PropertyRow(TypedDict, total=False): + id: str type: str type_args: List[str] prepare: Optional[Dict[str, Any]] diff --git a/spinta/manifests/tabular/helpers.py b/spinta/manifests/tabular/helpers.py index 49c7eb132..722169878 100644 --- a/spinta/manifests/tabular/helpers.py +++ b/spinta/manifests/tabular/helpers.py @@ -4,6 +4,7 @@ import pathlib import logging import textwrap +import uuid from operator import itemgetter from itertools import zip_longest from typing import Any @@ -38,7 +39,7 @@ from spinta.components import Namespace from spinta.components import Property from spinta.core.enums import Access -from spinta.core.ufuncs import unparse +from spinta.core.ufuncs import unparse, Expr from spinta.datasets.components import Dataset from spinta.dimensions.enum.components import Enums from spinta.dimensions.lang.components import LangData @@ -173,18 +174,21 @@ class TabularReader: data: ManifestRow # Used when `appendable` is False rows: List[Dict[str, Any]] # Used when `appendable` is True appendable: bool = False # Tells if reader is appendable. + allow_updates: bool = False # Tells if manifest supports structure updates def __init__( self, state: State, path: str, line: str, + allow_updates: bool = False ): self.state = state self.path = path self.line = line self.data = {} self.rows = [] + self.allow_updates = allow_updates def __str__(self): return f"<{type(self).__name__} name={self.name!r}>" @@ -193,10 +197,11 @@ def read(self, row: Dict[str, str]) -> None: raise NotImplementedError def append(self, row: Dict[str, str]) -> None: - if any(row.values()): - self.error( - f"Updates are not supported in context of {self.type!r}." - ) + if not self.allow_updates: + if any(row.values()): + self.error( + f"Updates are not supported in context of {self.type!r}." + ) def release(self, reader: TabularReader = None) -> bool: raise NotImplementedError @@ -313,6 +318,7 @@ def read_backend(self, row: Dict[str, str]) -> None: ) self.data = { + 'id': row['id'], 'type': row['type'], 'name': self.name, 'dsn': row['source'], @@ -330,6 +336,7 @@ def read_resource(self, row: Dict[str, str]) -> None: self.error("Resource with the same name already defined in ") self.data = { + 'id': row['id'], 'type': row['type'], 'backend': row['ref'], 'external': row['source'], @@ -344,10 +351,14 @@ def read_resource(self, row: Dict[str, str]) -> None: dataset['resources'][self.name] = self.data def release(self, reader: TabularReader = None) -> bool: + if self.state.dataset is None: + return True return reader is None or isinstance(reader, ( ManifestReader, DatasetReader, ResourceReader, + EnumReader, + PrefixReader )) or (isinstance(reader, ModelReader) and self.name == '/') def enter(self) -> None: @@ -371,6 +382,7 @@ def read(self, row: Dict[str, str]) -> None: dataset = dataset.data if dataset else None self.data = { + 'id': row['id'], 'name': self.name, 'model': get_relative_model_name(dataset, row['base']), 'pk': ( @@ -425,6 +437,7 @@ def read(self, row: Dict[str, str]) -> None: 'id': row['id'], 'name': name, 'base': { + 'id': base.data["id"], 'name': base.name, 'parent': base.data['model'], 'pk': base.data['pk'], @@ -633,6 +646,7 @@ def _append_prepare(self, row: Dict[str, str], prepare: str): def _initial_normal_property_schema(given_name: str, dtype: dict, row: dict): return { + 'id': row.get('id'), 'type': dtype['type'], 'type_args': dtype['type_args'], 'prepare': row.get(PREPARE), @@ -850,12 +864,12 @@ def _text_datatype_handler(reader: PropertyReader, row: dict): 'access': row['access'], })) temp_data['type'] = 'string' - temp_data['external'] = new_data['external'] + temp_data['external'] = new_data['external'] if 'external' in new_data else {} if result: new_data['langs'] = result['langs'] if new_data['level'] and int(new_data['level']) <= 3: new_data['langs']['C'] = temp_data - if new_data['external']: + if 'external' in new_data and new_data['external']: new_data['external'] = {} result.update(new_data) return result @@ -864,7 +878,7 @@ def _text_datatype_handler(reader: PropertyReader, row: dict): new_data['langs'] = { 'C': temp_data } - if new_data['external']: + if 'external' in new_data and new_data['external']: new_data['external'] = {} return new_data @@ -1304,6 +1318,7 @@ def read(self, row: ManifestRow) -> None: self.error(f"Enum's do not have a level, but level {row[LEVEL]!r} is given.") self.data = { + 'id': row[ID], 'name': self.name, 'source': row[SOURCE], 'prepare': _parse_spyna(self, row[PREPARE]), @@ -1361,6 +1376,7 @@ def read(self, row: ManifestRow) -> None: ModelReader, PropertyReader, EnumReader, + LangReader )): self.error(f'Language metadata is not supported on {reader.type}.') return @@ -1533,6 +1549,7 @@ def _read_tabular_manifest_rows( rows: Iterator[Tuple[str, List[str]]], *, rename_duplicates: bool = True, + allow_updates: bool = False ) -> Iterator[ParsedRow]: _, header = next(rows, (None, None)) if header is None: @@ -1544,7 +1561,7 @@ def _read_tabular_manifest_rows( state = State() state.rename_duplicates = rename_duplicates - reader = ManifestReader(state, path, '1') + reader = ManifestReader(state, path, '1', allow_updates=allow_updates) reader.read({}) yield from state.release(reader) @@ -1554,7 +1571,7 @@ def _read_tabular_manifest_rows( row = {**defaults, **row} dimension = _detect_dimension(path, line, row) Reader = READERS[dimension] - reader = Reader(state, path, line) + reader = Reader(state, path, line, allow_updates=allow_updates) reader.read(row) yield from state.release(reader) @@ -1766,6 +1783,8 @@ def get_relative_model_name(dataset: [str, dict], name: str) -> str: return name.replace(dataset, '') if name.startswith('/'): return name[1:] + elif '/' in name: + return name elif dataset is None: return name else: @@ -1779,9 +1798,8 @@ def to_relative_model_name(model: Model, dataset: Dataset = None) -> str: """Convert absolute model `name` to relative.""" if dataset is None: return model.name - if model.name.startswith(dataset.name): - prefix = dataset.name - return model.name[len(prefix) + 1:] + if model.name == f'{dataset.name}/{model.basename}': + return model.basename else: return '/' + model.name @@ -1886,6 +1904,7 @@ def _backends_to_tabular( ) -> Iterator[ManifestRow]: for name, backend in backends.items(): yield torow(DATASET, { + 'id': backend.config.get('id'), 'type': backend.type, 'resource': name, 'source': backend.config.get('dsn'), @@ -1907,6 +1926,7 @@ def _namespaces_to_tabular( first = True for name, ns in namespaces.items(): yield torow(DATASET, { + 'id': ns.id, 'type': ns.type if first else '', 'ref': name, 'title': ns.title, @@ -1944,6 +1964,7 @@ def _enums_to_tabular( if item.access is not None and item.access < access: continue yield torow(DATASET, { + 'id': item.id, 'type': 'enum' if first else '', 'ref': name if first else '', 'source': item.source if external else '', @@ -1970,6 +1991,7 @@ def _lang_to_tabular( first = True for name, data in sorted(lang.items(), key=itemgetter(0)): yield torow(DATASET, { + 'id': data['id'], 'type': 'lang' if first else '', 'ref': name if first else '', 'title': data['title'], @@ -1985,6 +2007,7 @@ def _text_to_tabular( return for lang in prop.dtype.langs: yield torow(DATASET, { + 'id': prop.id, 'property': prop.name + '@' + lang, 'type': prop.dtype.name, 'level': prop.level.value if prop.level is not None else '', @@ -2085,6 +2108,7 @@ def _resource_to_tabular( ) -> Iterator[ManifestRow]: backend = resource.backend yield torow(DATASET, { + 'id': resource.id, 'resource': resource.name, 'source': resource.external if external else '', 'prepare': unparse(resource.prepare or NA) if external else '', @@ -2114,6 +2138,7 @@ def _base_to_tabular( data = { 'base': base.name, 'level': base.level.value if base.level else "", + 'id': base.id, } if base.pk: data['ref'] = ', '.join([pk.place for pk in base.pk]) @@ -2136,6 +2161,7 @@ def _property_to_tabular( data = { 'property': prop.given.name or prop.name, + 'id': prop.id, 'type': _get_type_repr(prop.dtype), 'level': prop.level.value if prop.level else "", 'access': prop.given.access, @@ -2192,8 +2218,6 @@ def _property_to_tabular( else: data['ref'] = prop.dtype.model.name - for denorm_prop in prop.dtype.properties.values(): - yield_rows.append(denorm_prop) elif isinstance(prop.dtype, Object): for obj_prop in prop.dtype.properties.values(): yield_rows.append(obj_prop) @@ -2295,6 +2319,7 @@ def _model_to_tabular( def datasets_to_tabular( + context: Context, manifest: Manifest, *, external: bool = True, # clean content of source and prepare @@ -2304,7 +2329,7 @@ def datasets_to_tabular( ) -> Iterator[ManifestRow]: yield from _prefixes_to_tabular(manifest.prefixes, separator=True) yield from _backends_to_tabular(manifest.backends, separator=True) - yield from _namespaces_to_tabular(manifest.namespaces, separator=True) + yield from _namespaces_to_tabular(commands.get_namespaces(context, manifest), separator=True) yield from _enums_to_tabular( manifest.enums, external=external, @@ -2317,7 +2342,8 @@ def datasets_to_tabular( dataset = None resource = None base = None - models = manifest.models if internal else take(manifest.models) + models = commands.get_models(context, manifest) + models = models if internal else take(models) models = sort(MODELS_ORDER_BY, models.values(), order_by) separator = False @@ -2384,7 +2410,7 @@ def datasets_to_tabular( order_by=order_by, ) - datasets = sort(DATASETS_ORDER_BY, manifest.datasets.values(), order_by) + datasets = sort(DATASETS_ORDER_BY, commands.get_datasets(context, manifest).values(), order_by) for dataset in datasets: if dataset.name in seen_datasets: continue @@ -2407,12 +2433,13 @@ def torow(keys, values) -> ManifestRow: def render_tabular_manifest( + context: Context, manifest: Manifest, cols: List[ManifestColumn] = None, *, sizes: Dict[ManifestColumn, int] = None, ) -> str: - rows = datasets_to_tabular(manifest) + rows = datasets_to_tabular(context, manifest) return render_tabular_manifest_rows(rows, cols, sizes=sizes) @@ -2454,7 +2481,10 @@ def render_tabular_manifest_rows( for row in rows: if ID in cols: - line = [row[ID][:2] if row[ID] else ' '] + value = row[ID] + if isinstance(value, uuid.UUID): + value = str(value) + line = [value[:2] if value else ' '] else: line = [] @@ -2526,6 +2556,7 @@ def normalizes_columns( def write_tabular_manifest( + context: Context, path: str, rows: Union[ Manifest, @@ -2539,7 +2570,7 @@ def write_tabular_manifest( if rows is None: rows = [] elif isinstance(rows, Manifest): - rows = datasets_to_tabular(rows) + rows = datasets_to_tabular(context, rows) rows = ({c: row[c] for c in cols} for row in rows) if path.endswith('.csv'): diff --git a/spinta/manifests/yaml/commands/freeze.py b/spinta/manifests/yaml/commands/freeze.py index 94ce01571..38c9d4fde 100644 --- a/spinta/manifests/yaml/commands/freeze.py +++ b/spinta/manifests/yaml/commands/freeze.py @@ -1,4 +1,5 @@ from spinta import commands +from spinta.manifests.components import get_manifest_object_names from spinta.migrations import get_new_schema_version from spinta.components import Context from spinta.manifests.yaml.components import YamlManifest @@ -22,18 +23,18 @@ def freeze(context: Context, current: YamlManifest): # Read current nodes, get freezed versions and freeze changes between # current and previously freezed node into a new version. - for ntype, nodes in current.objects.items(): + for ntype in get_manifest_object_names(): if ntype == 'ns': # Namespaces do not have physical form, yet, se there is no need to # freeze them. continue - for name, cnode in nodes.items(): + for name, cnode in commands.get_nodes(context, current, ntype).items(): # Get freezed node - if name in freezed.objects[ntype]: - fnode = freezed.objects[ntype][name] + if commands.has_node(context, freezed, ntype, name): + fnode = commands.get_node(context, freezed, ntype, name) else: fnode = None diff --git a/spinta/manifests/yaml/commands/load.py b/spinta/manifests/yaml/commands/load.py index 6ee8c6465..2a1f78acf 100644 --- a/spinta/manifests/yaml/commands/load.py +++ b/spinta/manifests/yaml/commands/load.py @@ -22,12 +22,13 @@ def load( freezed: bool = False, rename_duplicates: bool = False, load_internal: bool = True, + full_load: bool = False ): if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) if freezed: if into: @@ -75,6 +76,7 @@ def load( freezed: bool = True, rename_duplicates: bool = False, load_internal: bool = True, + full_load: bool = False ): assert freezed, ( "InlineManifest does not have unfreezed version of manifest." @@ -82,9 +84,9 @@ def load( if load_internal: target = into or manifest - if '_schema' not in target.models: + if not commands.has_model(context, target, '_schema'): store = context.get('store') - commands.load(context, store.internal, into=target) + commands.load(context, store.internal, into=target, full_load=full_load) if into: log.info( @@ -111,4 +113,5 @@ def load( freezed=freezed, rename_duplicates=rename_duplicates, load_internal=load_internal, + full_load=full_load ) diff --git a/spinta/naming/helpers.py b/spinta/naming/helpers.py index 8e2406ae3..ecd9cbae6 100644 --- a/spinta/naming/helpers.py +++ b/spinta/naming/helpers.py @@ -1,6 +1,7 @@ from typing import Any from typing import Union +from spinta import commands from spinta.components import Context from spinta.components import Model from spinta.components import Property @@ -134,10 +135,11 @@ def _format_model(model: Model) -> Model: def reformat_names(context: Context, manifest: Manifest): - for model in manifest.models.values(): + models = commands.get_models(context, manifest) + for model in models.values(): _format_model_expr(context, model) - manifest.objects['model'] = { + commands.set_models(context, manifest, { model.name: model - for model in map(_format_model, manifest.models.values()) - } + for model in map(_format_model, models.values()) + }) diff --git a/spinta/nodes.py b/spinta/nodes.py index 5d8635924..c48485473 100644 --- a/spinta/nodes.py +++ b/spinta/nodes.py @@ -19,6 +19,7 @@ def get_node( + context: Context, config: Config, manifest: Manifest, # MetaData entry ID, for yaml manifests it's filename, for backend manifests @@ -56,7 +57,7 @@ def get_node( # If parent is given, that means we are loading a node whose parent is # not manifest, that means we can't do checks on manifest.objects. - if ctype not in manifest.objects: + if not commands.has_node_type(context, manifest, ctype): raise exceptions.InvalidManifestFile( manifest=manifest.name, eid=eid, @@ -71,9 +72,9 @@ def get_node( prop='name', ) - if data['name'] in manifest.objects[ctype]: + if commands.has_node(context, manifest, ctype, data['name'], loaded=True): name = data['name'] - other = manifest.objects[ctype][name].eid + other = commands.get_node(context, manifest, ctype, name).eid raise exceptions.InvalidManifestFile( manifest=manifest.name, eid=eid, diff --git a/spinta/testing/config.py b/spinta/testing/config.py index 88d0b6501..a0b48c4c2 100644 --- a/spinta/testing/config.py +++ b/spinta/testing/config.py @@ -2,6 +2,7 @@ from typing import Optional from spinta.auth import gen_auth_server_keys, get_clients_path +from spinta.components import Context from spinta.core.config import RawConfig from spinta.testing.datasets import Sqlite from spinta.testing.tabular import create_tabular_manifest @@ -38,12 +39,13 @@ def create_config_path(path: pathlib.Path) -> pathlib.Path: def configure( + context: Context, rc: RawConfig, db: Optional[Sqlite], # TODO: move backend configuration to manifest path: pathlib.Path, # manifest file path manifest: str, # manifest as ascii table string ) -> RawConfig: - create_tabular_manifest(path, striptable(manifest)) + create_tabular_manifest(context, path, striptable(manifest)) if db: return rc.fork({ 'manifests': { diff --git a/spinta/testing/context.py b/spinta/testing/context.py index 6b74f3e44..4651e8cf9 100644 --- a/spinta/testing/context.py +++ b/spinta/testing/context.py @@ -61,13 +61,13 @@ def transaction(self: TestContext, *, write=False): def wipe(self: TestContext, model: Union[str, Node]): if isinstance(model, str): store = self.get('store') - model = store.manifest.models[model] + model = commands.get_model(self, store.manifest, model) with self.transaction() as context: commands.wipe(context, model, model.backend) def wipe_all(self: TestContext): store = self.get('store') - self.wipe(store.manifest.objects['ns']['']) + self.wipe(commands.get_namespace(self, store.manifest, '')) def load( self: TestContext, diff --git a/spinta/testing/dtypes.py b/spinta/testing/dtypes.py index 1affbc1d0..1ab411d33 100644 --- a/spinta/testing/dtypes.py +++ b/spinta/testing/dtypes.py @@ -5,6 +5,7 @@ def path(model: str): parts = model.split('/') + parts = list([part.lower() for part in parts]) if len(parts) > 3 and (parts[0], parts[2]) == ('backends', 'dtypes'): parts = parts[3:] parts = [ @@ -17,11 +18,13 @@ def path(model: str): def nest(model: str, data: dict): parts = model.split('/') + parts = list([part.lower() for part in parts]) + name = parts[-1] if len(parts) > 3 and (parts[0], parts[2]) == ('backends', 'dtypes'): parts = parts[3:] if parts[-1] in data: d = data = data.copy() - value = d.pop(parts[-1]) + value = d.pop(name) for k in parts[:-1]: if k == 'array': v = [] @@ -35,12 +38,14 @@ def nest(model: str, data: dict): if isinstance(d, list): d.append(value) else: - d[parts[-1]] = value + d[name] = value return data def flat(model, data): parts = model.split('/') + parts = list([part.lower() for part in parts]) + name = parts[-1] if len(parts) > 3 and (parts[0], parts[2]) == ('backends', 'dtypes'): parts = parts[3:] if parts[0] in data: @@ -60,12 +65,12 @@ def flat(model, data): break data.pop(parts[0]) if v is not NA: - data[parts[-1]] = v + data[name] = v return data def post(app, model: str, value: str, *, status: int = 201): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() data = nest(model, {name: value}) resp = app.post(f'/{model}', json=data) assert resp.status_code == status, resp.json() @@ -84,7 +89,7 @@ def post(app, model: str, value: str, *, status: int = 201): def upsert(app, model: str, where: str, value: str, *, status: int): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() data = nest(model, { '_op': 'upsert', '_where': f'%s={where!r}' % path(model), @@ -107,7 +112,7 @@ def upsert(app, model: str, where: str, value: str, *, status: int): def put(app, model: str, pk: str, rev: str, value: str = NA): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() data = nest(model, take(all, { '_revision': rev, name: value, @@ -129,7 +134,7 @@ def put(app, model: str, pk: str, rev: str, value: str = NA): def patch(app, model: str, pk: str, rev: str, value: str = NA): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() data = nest(model, take(all, { '_revision': rev, name: value, @@ -161,7 +166,7 @@ def delete(app, model: str, pk: str, rev: str): def get(app, model, pk, rev, status=200): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() resp = app.get(f'{model}/{pk}?expand()') data = resp.json() assert resp.status_code == status, data @@ -180,7 +185,7 @@ def get(app, model, pk, rev, status=200): def search(app, model, pk, rev, val=NA, by=None): - name = model.split('/')[-1] + name = model.split('/')[-1].lower() place = path(model) if val is None: val = 'null' diff --git a/spinta/testing/manifest.py b/spinta/testing/manifest.py index b01064824..a760d35c8 100644 --- a/spinta/testing/manifest.py +++ b/spinta/testing/manifest.py @@ -7,12 +7,14 @@ from spinta import commands from spinta.cli.helpers.store import load_store +from spinta.cli.manifest import copy_manifest from spinta.components import Context from spinta.components import Store from spinta.core.config import RawConfig from spinta.core.config import configure_rc from spinta.manifests.components import Manifest from spinta.manifests.components import ManifestPath +from spinta.manifests.helpers import detect_manifest_from_path from spinta.manifests.tabular.helpers import normalizes_columns from spinta.manifests.tabular.helpers import render_tabular_manifest from spinta.manifests.tabular.helpers import striptable @@ -20,42 +22,88 @@ from spinta.testing.context import create_test_context -def compare_manifest(manifest: Manifest, expected: str) -> Tuple[str, str]: +def compare_manifest(manifest: Manifest, expected: str, context: Context = None) -> Tuple[str, str]: expected = striptable(expected) + if not context: + context = Context('empty') if expected: header = expected.splitlines()[0] cols = normalizes_columns(header.split('|')) sizes = {c: len(x) - 2 for c, x in zip(cols, f' {header} '.split('|'))} - actual = render_tabular_manifest(manifest, cols, sizes=sizes) + actual = render_tabular_manifest(context, manifest, cols, sizes=sizes) else: - actual = render_tabular_manifest(manifest) + actual = render_tabular_manifest(context, manifest) return actual, expected +def _create_file_path_for_type(tmp_path: pathlib.Path, file_name: str, manifest_type: str): + if manifest_type != 'internal_sql' and tmp_path is None: + raise Exception(f"TMP_PATH IS REQUIRED FOR {manifest_type} MANIFEST") + + if manifest_type == 'internal_sql': + return 'sqlite:///' + str(tmp_path / f'{file_name}.sqlite') + elif manifest_type in ['csv', 'xml', 'xlsx', 'yaml']: + return str(tmp_path / f'{file_name}.{manifest_type}') + elif manifest_type in ['ascii', 'tabular']: + return str(tmp_path / f'{file_name}.txt') + else: + raise Exception(f"NO SUPPORT FOR {manifest_type} MANIFEST") + + def load_manifest_get_context( rc: RawConfig, manifest: Union[pathlib.Path, str] = None, *, load_internal: bool = False, request: FixtureRequest = None, + manifest_type: str = '', + tmp_path: pathlib.Path = None, + full_load: bool = True, ensure_config_dir: bool = False, **kwargs, ) -> TestContext: - if isinstance(manifest, pathlib.Path): - manifests = [str(manifest)] - elif isinstance(manifest, str) and '|' in manifest: - manifests = [ManifestPath( - type='ascii', - file=StringIO(manifest), - )] + temp_rc = configure_rc(rc, None, **kwargs) + context = create_test_context(temp_rc) + + if isinstance(manifest, str) and '|' in manifest: + if manifest_type and manifest_type != 'ascii': + ascii_file = _create_file_path_for_type(tmp_path, '_temp_ascii_manifest', 'ascii') + output_file = _create_file_path_for_type(tmp_path, '_temp_manifest', manifest_type) + with open(ascii_file, 'w') as f: + f.write(manifest) + copy_manifest( + context, + manifests=[ascii_file], + output=output_file, + output_type=manifest_type + ) + manifests = [output_file] + else: + manifests = [ManifestPath( + type='ascii', + file=StringIO(manifest), + )] elif manifest: + if isinstance(manifest, pathlib.Path): + manifest = str(manifest) + manifest_ = detect_manifest_from_path(rc, manifest) + if manifest_type and manifest_.type != manifest_type: + output_path = _create_file_path_for_type(tmp_path, '_temp_manifest', manifest_type) + copy_manifest( + context, + manifests=[manifest], + output=output_path, + output_type=manifest_type + ) + manifest = output_path manifests = [manifest] else: manifests = manifest - rc = configure_rc(rc, manifests, **kwargs) + manifest_type = manifest_type if manifest_type != 'internal_sql' else 'internal' + rc = configure_rc(rc, manifests, manifest_type='inline' if manifest_type != 'internal' else manifest_type, **kwargs) context = create_test_context(rc, request) store = load_store(context, verbose=False, ensure_config_dir=ensure_config_dir) - commands.load(context, store.manifest, load_internal=load_internal) + commands.load(context, store.manifest, load_internal=load_internal, full_load=full_load) commands.link(context, store.manifest) return context @@ -106,12 +154,14 @@ def bootstrap_manifest( *, request: FixtureRequest = None, load_internal: bool = True, + full_load: bool = True, **kwargs, ) -> TestContext: context = load_manifest_get_context( rc, manifest, request=request, load_internal=load_internal, + full_load=full_load, **kwargs, ) store: Store = context.get('store') diff --git a/spinta/testing/pytest.py b/spinta/testing/pytest.py index 17807b948..008aa8c2c 100644 --- a/spinta/testing/pytest.py +++ b/spinta/testing/pytest.py @@ -155,6 +155,12 @@ def pytest_addoption(parser): default=[], help="run tests only for particular model ['postgres', 'mongo', 'postgres/datasets']", ) + parser.addoption( + "--manifest_type", + action="append", + default=[], + help="run tests only for particular manifest ['internal_sql', 'csv', 'ascii']", + ) def pytest_configure(config): @@ -162,27 +168,44 @@ def pytest_configure(config): config.addinivalue_line( "markers", "models(*models): mark test to run multiple times with each model specified" ) + config.addinivalue_line( + "markers", "manifests(*manifests): mark test to run multiple times with each manifest type specified" + ) def pytest_generate_tests(metafunc): # Get model markers from test, if markers are set - leave test as is models = metafunc.definition.get_closest_marker('models') - if not models: - return - - # If there are markers, get them, together with model CLI options - models = set(models.args) - model_cli_options = set(metafunc.config.getoption('model')) - - # If model CLI options are not empty - # then get common markers from test and CLI options - if model_cli_options: - models = models.intersection(model_cli_options) - - # Parametrize our test with calculated models. - # If we pass to CLI model option, which does not have a test marker, - # then pytest will skip the test all together. - metafunc.parametrize('model', models) + if models: + # If there are markers, get them, together with model CLI options + models = set(models.args) + model_cli_options = set(metafunc.config.getoption('model')) + + # If model CLI options are not empty + # then get common markers from test and CLI options + if model_cli_options: + models = models.intersection(model_cli_options) + + # Parametrize our test with calculated models. + # If we pass to CLI model option, which does not have a test marker, + # then pytest will skip the test all together. + metafunc.parametrize('model', models) + + manifests = metafunc.definition.get_closest_marker('manifests') + if manifests: + # If there are markers, get them, together with manifest CLI options + manifests = set(manifests.args) + manifest_cli_options = set(metafunc.config.getoption('manifest_type')) + + # If model CLI options are not empty + # then get common markers from test and CLI options + if manifest_cli_options: + manifests = manifests.intersection(manifest_cli_options) + + # Parametrize our test with calculated manifests. + # If we pass to CLI model option, which does not have a test marker, + # then pytest will skip the test all together. + metafunc.parametrize('manifest_type', manifests) def _diff_line(line: str) -> str: diff --git a/spinta/testing/tabular.py b/spinta/testing/tabular.py index c531cbf77..0d8e11a51 100644 --- a/spinta/testing/tabular.py +++ b/spinta/testing/tabular.py @@ -3,6 +3,7 @@ from io import StringIO from typing import List +from spinta.components import Context from spinta.manifests.tabular.components import ManifestColumn from spinta.manifests.tabular.constants import DATASET from spinta.manifests.tabular.helpers import read_ascii_tabular_rows @@ -11,6 +12,7 @@ def create_tabular_manifest( + context: Context, path: pathlib.Path, manifest: str, ) -> None: @@ -19,7 +21,7 @@ def create_tabular_manifest( cols: List[ManifestColumn] = next(rows, []) if cols: rows = (torow(DATASET, dict(zip(cols, row))) for row in rows) - write_tabular_manifest(str(path), rows) + write_tabular_manifest(context, str(path), rows) def convert_ascii_manifest_to_csv(manifest: str) -> bytes: diff --git a/spinta/types/backref/link.py b/spinta/types/backref/link.py index 4ef611807..9e7889734 100644 --- a/spinta/types/backref/link.py +++ b/spinta/types/backref/link.py @@ -65,9 +65,9 @@ def _link_backref(context: Context, dtype: BackRef): # Self reference. dtype.model = dtype.prop.model else: - if backref_target_model not in dtype.prop.model.manifest.models: + if not commands.has_model(context, dtype.prop.model.manifest, backref_target_model): raise ModelReferenceNotFound(dtype, ref=backref_target_model) - dtype.model = dtype.prop.model.manifest.models[backref_target_model] + dtype.model = commands.get_model(context, dtype.prop.model.manifest, backref_target_model) given_refprop = dtype.refprop if dtype.refprop: dtype.explicit = True diff --git a/spinta/types/denorm/link.py b/spinta/types/denorm/link.py index 6f9fef807..756287512 100644 --- a/spinta/types/denorm/link.py +++ b/spinta/types/denorm/link.py @@ -10,6 +10,7 @@ def link(context: Context, dtype: Denorm) -> None: set_dtype_backend(dtype) dtype.rel_prop = _get_denorm_prop( + context, dtype.prop.name, dtype.prop, dtype.prop.model @@ -18,21 +19,22 @@ def link(context: Context, dtype: Denorm) -> None: # TODO: Add better support for denorm when nested (with object type, etc.) def _get_denorm_prop( + context: Context, name: str, prop: Property, model: Model, ) -> Property: - models = model.manifest.models + manifest = model.manifest name_parts = name.split('.', 1) name = name_parts[0] properties = prop.parent.dtype.model.properties if isinstance(prop.parent.dtype, Ref) else prop.parent.model.properties - model = models[prop.parent.dtype.model.name] if isinstance(prop.parent.dtype, Ref) else model + model = commands.get_model(context, manifest, prop.parent.dtype.model.name) if isinstance(prop.parent.dtype, Ref) else model if len(name_parts) > 1: ref_prop = properties[name] while isinstance(ref_prop.dtype, Array): ref_prop = ref_prop.dtype.items - model = models[ref_prop.dtype.model.name] if isinstance(ref_prop.dtype, Ref) else model + model = commands.get_model(context, manifest, ref_prop.dtype.model.name) if isinstance(ref_prop.dtype, Ref) else model if name not in properties or not isinstance(ref_prop.dtype, (Ref, Object)): if prop.model == model: raise NoRefPropertyForDenormProperty( @@ -46,7 +48,7 @@ def _get_denorm_prop( ref={'property': name, 'model': model.name}, ) else: - denorm_prop = _get_denorm_prop(name_parts[1], prop, model) + denorm_prop = _get_denorm_prop(context, name_parts[1], prop, model) else: if name not in properties: raise ReferencedPropertyNotFound( diff --git a/spinta/types/helpers.py b/spinta/types/helpers.py index 24d3bf105..400f04f90 100644 --- a/spinta/types/helpers.py +++ b/spinta/types/helpers.py @@ -48,9 +48,10 @@ def check_no_extra_keys(dtype: DataType, schema: Iterable, data: Iterable): def set_dtype_backend(dtype: DataType): if dtype.backend: backends = dtype.prop.model.manifest.store.backends - if dtype.backend not in backends: - raise BackendNotFound(dtype, name=dtype.backend) - dtype.backend = backends[dtype.backend] + if isinstance(dtype.backend, str): + if dtype.backend not in backends: + raise BackendNotFound(dtype, name=dtype.backend) + dtype.backend = backends[dtype.backend] else: dtype.backend = dtype.prop.model.backend diff --git a/spinta/types/model.py b/spinta/types/model.py index af1eb2efe..40f5fb29a 100644 --- a/spinta/types/model.py +++ b/spinta/types/model.py @@ -88,7 +88,6 @@ def load( else: model.keymap = manifest.keymap - manifest.add_model_endpoint(model) _load_namespace_from_model(context, manifest, model) load_access_param(model, data.get('access'), itertools.chain( [model.ns], @@ -105,6 +104,7 @@ def load( if model.base: base: dict = model.base model.base = get_node( + context, config, manifest, model.eid, @@ -133,6 +133,7 @@ def load( if model.external: external: dict = model.external model.external = get_node( + context, config, manifest, model.eid, @@ -153,6 +154,9 @@ def load( builder.update(model=model) builder.load_page() + if not model.name.startswith('_') and not model.basename[0].isupper(): + raise Exception(model.basename, "MODEL NAME NEEDS TO BE UPPER CASED") + return model @@ -172,7 +176,7 @@ def link(context: Context, model: Model): raise KeymapNotSet(model) # Link model backend. - if model.backend: + if model.backend and isinstance(model.backend, str): if model.backend in model.manifest.backends: model.backend = model.manifest.backends[model.backend] else: @@ -242,7 +246,7 @@ def _link_model_page(model: Model): @overload @commands.link.register(Context, Base) def link(context: Context, base: Base): - base.parent = base.model.manifest.models[base.parent] + base.parent = commands.get_model(context, base.model.manifest, base.parent) base.pk = [ base.parent.properties[pk] for pk in base.pk @@ -281,6 +285,7 @@ def load( if data['type'] == 'ref' and prop.level is not None and prop.level < 4: data['type'] = '_external_ref' prop.dtype = get_node( + context, config, manifest, prop.model.eid, @@ -398,6 +403,7 @@ def _load_property_external( config = context.get('config') external: Attribute = get_node( + context, config, manifest, prop.model.eid, diff --git a/spinta/types/namespace.py b/spinta/types/namespace.py index d135a6226..8f9fb94f1 100644 --- a/spinta/types/namespace.py +++ b/spinta/types/namespace.py @@ -64,7 +64,6 @@ def load_namespace_from_name( drop: bool = True, ) -> Namespace: ns: Optional[Namespace] = None - parent: Optional[Namespace] = None parts: List[str] = [] parts_ = [p for p in path.split('/') if p] if drop: @@ -72,7 +71,7 @@ def load_namespace_from_name( for part in [''] + parts_: parts.append(part) name = '/'.join(parts[1:]) - if name not in manifest.namespaces: + if not commands.has_namespace(context, manifest, name, loaded=True): ns = Namespace() data = { 'type': 'ns', @@ -82,23 +81,11 @@ def load_namespace_from_name( } commands.load(context, ns, data, manifest) ns.generated = True + commands.link(context, ns) else: - ns = manifest.namespaces[name] + ns = commands.get_namespace(context, manifest, name) pass - if parent: - if ns.name == parent.name: - raise RuntimeError(f"Self reference in {path!r}.") - - ns.parent = parent or manifest - - if part and part not in parent.names: - parent.names[part] = ns - else: - ns.parent = manifest - - parent = ns - return ns @@ -118,12 +105,24 @@ def load( ns.backend = None ns.names = {} ns.models = {} - manifest.namespaces[ns.name] = ns + commands.set_namespace(context, manifest, ns.name, ns) @commands.link.register(Context, Namespace) def link(context: Context, ns: Namespace): - pass + split = ns.name.split('/') + if len(split) > 1: + parent_ns = commands.get_namespace(context, ns.manifest, '/'.join(split[:-1])) + if parent_ns: + ns.parent = parent_ns + elif ns.name != '': + if commands.has_namespace(context, ns.manifest, ''): + ns.parent = commands.get_namespace(context, ns.manifest, '') + else: + ns.parent = ns.manifest + + if isinstance(ns.parent, Namespace): + ns.parent.names[ns.name] = ns @commands.check.register(Context, Namespace) @@ -139,7 +138,7 @@ def check(context: Context, ns: Namespace): @commands.authorize.register(Context, Action, Namespace) def authorize(context: Context, action: Action, ns: Namespace): - authorized(context, ns, action, throw=True) + commands.authorize(context, action, ns, ns.manifest) @commands.getall.register(Context, Namespace, Request) @@ -154,7 +153,7 @@ async def getall( ) -> Response: config: Config = context.get('config') if config.root and ns.is_root(): - ns = ns.manifest.namespaces[config.root] + ns = commands.get_namespace(context, ns.manifest, config.root) commands.authorize(context, action, ns) @@ -166,54 +165,7 @@ async def getall( ns=ns.name, action=action.value, ) - - if params.all and params.ns: - - for model in traverse_ns_models(context, ns, action, internal=True): - commands.authorize(context, action, model) - return _get_ns_content( - context, - request, - ns, - params, - action, - recursive=True, - ) - elif params.all: - accesslog = context.get('accesslog') - - prepare_data_for_response_kwargs = {} - for model in traverse_ns_models(context, ns, action, internal=True): - commands.authorize(context, action, model) - select_tree = get_select_tree(context, action, params.select) - prop_names = get_select_prop_names( - context, - model, - model.properties, - action, - select_tree, - ) - prepare_data_for_response_kwargs[model.model_type()] = { - 'select': select_tree, - 'prop_names': prop_names, - } - expr = urlparams_to_expr(params) - rows = getall(context, ns, action=action, query=expr) - rows = ( - commands.prepare_data_for_response( - context, - ns.manifest.models[row['_type']], - params.fmt, - row, - action=action, - **prepare_data_for_response_kwargs[row['_type']], - ) - for row in rows - ) - rows = log_response(context, rows) - return render(context, request, ns, params, rows, action=action) - else: - return _get_ns_content(context, request, ns, params, action) + return commands.getall(context, ns, request, ns.manifest, action=action, params=params) @commands.getall.register(Context, Namespace) @@ -237,12 +189,13 @@ def _query_data( resource: Optional[str] = None, **kwargs, ): - models = traverse_ns_models( + models = commands.traverse_ns_models( context, ns, + ns.manifest, action, - dataset_, - resource, + dataset_=dataset_, + resource=resource, internal=True, ) for model in models: @@ -253,35 +206,6 @@ def check_if_model_has_backend_and_source(model: Model): return not isinstance(model.backend, NoBackend) and (model.external and model.external.name) -def traverse_ns_models( - context: Context, - ns: Namespace, - action: Action, - dataset_: Optional[str] = None, - resource: Optional[str] = None, - *, - internal: bool = False, - source_check: bool = False -): - models = (ns.models or {}) - for model in models.values(): - if not (source_check and not check_if_model_has_backend_and_source(model)): - if _model_matches_params(context, model, action, dataset_, resource, internal): - yield model - for ns_ in ns.names.values(): - if not internal and ns_.name.startswith('_'): - continue - yield from traverse_ns_models( - context, - ns_, - action, - dataset_, - resource, - internal=internal, - source_check=source_check - ) - - def _model_matches_params( context: Context, model: Model, @@ -325,92 +249,6 @@ def _model_matches_params( return True -def _get_ns_content( - context: Context, - request: Request, - ns: Namespace, - params: UrlParams, - action: Action, - *, - recursive: bool = False, - dataset_: Optional[str] = None, - resource: Optional[str] = None, -) -> Response: - if recursive: - data = _get_ns_content_data_recursive(context, ns, action, dataset_, resource) - else: - data = _get_ns_content_data(context, ns, action, dataset_, resource) - - data = sorted(data, key=lambda x: (x.data['_type'] != 'ns', x.data['name'])) - - model = ns.manifest.models['_ns'] - select = params.select or ['name', 'title', 'description'] - select_tree = get_select_tree(context, action, select) - prop_names = get_select_prop_names( - context, - model, - model.properties, - action, - select_tree, - auth=False, - ) - rows = ( - commands.prepare_data_for_response( - context, - model, - params.fmt, - row.data, - action=action, - select=select_tree, - prop_names=prop_names, - ) - for row in data - ) - - rows = log_response(context, rows) - - return render(context, request, model, params, rows, action=action) - - -class _NodeAndData(NamedTuple): - node: Union[Namespace, Model] - data: Dict[str, Any] - - -def _get_ns_content_data_recursive( - context: Context, - ns: Namespace, - action: Action, - dataset_: Optional[str] = None, - resource: Optional[str] = None, -) -> Iterable[_NodeAndData]: - yield from _get_ns_content_data(context, ns, action, dataset_, resource) - for name in ns.names.values(): - yield from _get_ns_content_data_recursive(context, name, action, dataset_, resource) - - -def _get_ns_content_data( - context: Context, - ns: Namespace, - action: Action, - dataset_: Optional[str] = None, - resource: Optional[str] = None, -) -> Iterable[_NodeAndData]: - items: Iterable[Union[Namespace, Model]] = itertools.chain( - ns.names.values(), - ns.models.values(), - ) - - for item in items: - if _model_matches_params(context, item, action, dataset_, resource): - yield _NodeAndData(item, { - '_type': item.node_type(), - 'name': item.model_type(), - 'title': item.title, - 'description': item.description, - }) - - @commands.getone.register(Context, Request, Namespace) async def getone( context: Context, @@ -441,7 +279,7 @@ def in_namespace(node: Node, parent: Node) -> bool: # noqa @commands.wipe.register(Context, Namespace, type(None)) def wipe(context: Context, ns: Namespace, backend: type(None)): commands.authorize(context, Action.WIPE, ns) - models = traverse_ns_models(context, ns, Action.WIPE, internal=True) + models = commands.traverse_ns_models(context, ns, ns.manifest, Action.WIPE, internal=True) models = sort_models_by_refs(models) for model in models: if BackendFeatures.WRITE in model.backend.features: diff --git a/spinta/types/owner.py b/spinta/types/owner.py deleted file mode 100644 index fbe4abf8c..000000000 --- a/spinta/types/owner.py +++ /dev/null @@ -1,39 +0,0 @@ -from spinta.commands import check, load -from spinta.components import Context, MetaData -from spinta.manifests.components import Manifest -from spinta.nodes import load_node -from spinta import exceptions - - -class Owner(MetaData): - schema = { - 'title': {'type': 'string'}, - 'sector': {'type': 'string'}, - 'logo': {'type': 'path'}, - 'backend': {'type': 'backend', 'inherit': True, 'required': False}, - } - - def __init__(self): - super().__init__() - self.logo = None - self.sector = None - - -@load.register(Context, Owner, dict, Manifest) -def load( - context: Context, - owner: Owner, - data: dict, - manifest: Manifest, - *, - source: Manifest = None, -): - return load_node(context, owner, data) - - -@check.register(Context, Owner) -def check(context: Context, owner: Owner): - if owner.logo: - path = owner.manifest.path / 'media/owners' / owner.name / owner.logo - if not path.exists(): - raise exceptions.FileNotFound(str(path), owner) diff --git a/spinta/types/project.py b/spinta/types/project.py deleted file mode 100644 index e95b2de47..000000000 --- a/spinta/types/project.py +++ /dev/null @@ -1,84 +0,0 @@ -from spinta.commands import load, prepare, check -from spinta.components import Context, Node, MetaData -from spinta.manifests.components import Manifest -from spinta.nodes import load_node -from spinta import exceptions - - -class Project(MetaData): - schema = { - 'version': {'type': 'integer', 'required': True}, - 'date': {'type': 'date', 'required': True}, - 'objects': {'type': 'object', 'default': {}}, - 'impact': {'type': 'array', 'default': []}, - 'url': {'type': 'url'}, - 'source_code': {'type': 'url'}, - 'website': {'type': 'url'}, - 'owner': {'type': 'string'}, - 'dataset': {'type': 'string'}, - } - - -class Impact: - schema = { - 'year': {'type': 'integer', 'required': True}, - 'users': {'type': 'integer'}, - 'revenue': {'type': 'number'}, - 'employees': {'type': 'integer'}, - 'parent': {'type': 'project'}, - } - - -class Model(Node): - schema = { - 'properties': {'type': 'object', 'default': {}}, - 'dataset': {'type': 'string', 'inherit': True}, - 'target': {'type': 'string'}, - } - - -class Property(Node): - schema = { - 'enum': {'type': 'array'}, - 'dataset': {'type': 'string', 'inherit': True}, - 'target': {'type': 'string'}, - } - # TODO: inherit type from model if not provided, type is needed for data - # serialization. - - -@load.register(Context, Project, dict, Manifest) -def load( - context: Context, - project: Project, - data: dict, - manifest: Manifest, - *, - source: Manifest = None, -): - load_node(context, project, data) - project.impact = [ - { - 'year': None, - 'users': 0, - 'revenue': 0, - 'employees': 0, - **impact, - } for i, impact in enumerate(data.get('impact', [])) - ] - - -@prepare.register(Context, Project) -def prepare(context: Context, project: Project): - for model in project.objects.values(): - prepare(context, model) - - -@check.register(Context, Project) -def check(context: Context, project: Project): - if project.owner and project.owner not in project.manifest.objects['owner']: - raise exceptions.UnknownOwner(project) - - if project.dataset and project.dataset not in project.manifest.objects['dataset']: - # TODO add similar 'dataset' checks for model and property. - raise exceptions.UnknownProjectDataset(project) diff --git a/spinta/types/ref/link.py b/spinta/types/ref/link.py index a161df456..747fd0733 100644 --- a/spinta/types/ref/link.py +++ b/spinta/types/ref/link.py @@ -18,9 +18,9 @@ def link(context: Context, dtype: Ref) -> None: # Self reference. dtype.model = dtype.prop.model else: - if rmodel not in dtype.prop.model.manifest.models: + if not commands.has_model(context, dtype.prop.model.manifest, rmodel): raise ModelReferenceNotFound(dtype, ref=rmodel) - dtype.model = dtype.prop.model.manifest.models[rmodel] + dtype.model = commands.get_model(context, dtype.prop.model.manifest, rmodel) if dtype.refprops: refprops = [] diff --git a/spinta/types/store.py b/spinta/types/store.py index ca479652e..490a36812 100644 --- a/spinta/types/store.py +++ b/spinta/types/store.py @@ -78,7 +78,7 @@ def wait( store.manifest.backends.values(), ( resource.backend - for dataset in store.manifest.datasets.values() + for dataset in commands.get_datasets(context, store.manifest).values() for resource in dataset.resources.values() if resource.backend ) diff --git a/spinta/types/text/load.py b/spinta/types/text/load.py index 8c70a3737..49ed0c34b 100644 --- a/spinta/types/text/load.py +++ b/spinta/types/text/load.py @@ -17,6 +17,8 @@ def load(context: Context, dtype: Text, data: dict, manifest: Manifest) -> DataT prop.model = dtype.prop.model prop.list = dtype.prop.list commands.load(context, prop, params, manifest) + if prop.access is None: + prop.access = dtype.prop.access dtype.prop.model.flatprops[place] = prop props[name] = prop dtype.langs = props diff --git a/spinta/urlparams.py b/spinta/urlparams.py index d12e5725f..c8787692a 100644 --- a/spinta/urlparams.py +++ b/spinta/urlparams.py @@ -20,7 +20,7 @@ from spinta.utils import url as urlutil from spinta.components import UrlParams, Version from spinta.commands import is_object_id -from spinta import exceptions +from spinta import exceptions, commands from spinta import spyna from spinta.exceptions import ModelNotFound, InvalidPageParameterCount, InvalidPageKey from spinta.utils.config import asbool @@ -251,17 +251,12 @@ def _join_path_parts(*parts: str) -> str: def _find_model_name_index( - manifest: Manifest, parts: List[str], ) -> int: - keys = ( - set(manifest.namespaces) | - set(manifest.models) | - set(manifest.endpoints) - ) - for i, name in enumerate(itertools.accumulate(parts, _join_path_parts)): - if name not in keys: - return i + # By default, namespace, properties starts with lower case and Model starts with upper + for i, part in enumerate(parts): + if part[0].isupper(): + return i + 1 return len(parts) @@ -271,10 +266,10 @@ def _resolve_path(context: Context, params: UrlParams) -> None: params.path_parts = [] manifest = context.get('store').manifest - i = _find_model_name_index(manifest, params.path_parts) + i = _find_model_name_index(params.path_parts) parts = params.path_parts[i:] params.path = '/'.join(params.path_parts[:i]) - params.model = get_model_from_params(manifest, params) + params.model = get_model_from_params(context, manifest, params) if parts: # Resolve ID. @@ -309,27 +304,22 @@ def get_model_by_name(context: Context, manifest: Manifest, name: str) -> Node: def get_model_from_params( + context: Context, manifest: Manifest, params: UrlParams, ) -> Union[Namespace, Model]: name = params.path - - if name in manifest.endpoints: - # Allow users to specify a different URL endpoint to make URL look - # nicer, but that is optional, they can still use model.name. - name = manifest.endpoints[name] - if params.ns: - if name in manifest.namespaces: - return manifest.namespaces[name] + if commands.has_namespace(context, manifest, name): + return commands.get_namespace(context, manifest, name) else: raise ModelNotFound(manifest, model=name) - elif name in manifest.models: - return manifest.models[name] + elif commands.has_model(context, manifest, name): + return commands.get_model(context, manifest, name) - elif name in manifest.namespaces: - return manifest.namespaces[name] + elif commands.has_namespace(context, manifest, name): + return commands.get_namespace(context, manifest, name) else: raise ModelNotFound(model=name) diff --git a/tests/api/test_inspect.py b/tests/api/test_inspect.py index 690d0c43a..c8079397d 100644 --- a/tests/api/test_inspect.py +++ b/tests/api/test_inspect.py @@ -178,7 +178,7 @@ def test_inspect_manifest_resource_with_non_url_path( | | | code | string | | CODE | | open | Country code ''' - create_tabular_manifest(tmp_path / 'manifest.csv', table) + create_tabular_manifest(context, tmp_path / 'manifest.csv', table) app.authorize(["spinta_inspect"]) with open(tmp_path / 'manifest.csv', "rb") as f: form_data = { diff --git a/tests/backends/postgresql/commands/test_init.py b/tests/backends/postgresql/commands/test_init.py index 741af58a0..32b5b488a 100644 --- a/tests/backends/postgresql/commands/test_init.py +++ b/tests/backends/postgresql/commands/test_init.py @@ -20,7 +20,7 @@ def test_prepare(rc: RawConfig): | | | | name | string | | 3 | open | | | | country | ref | Country | 3 | open ''') - model = manifest.models['example/City'] + model = commands.get_model(context, manifest, 'example/City') backend = model.backend commands.prepare(context, backend, model) table = backend.get_table(model) @@ -48,7 +48,7 @@ def test_prepare_base_under_level(rc: RawConfig): | | | | test | string | | 3 | open ''') - model = manifest.models['example/base_under/NormalModel'] + model = commands.get_model(context, manifest, 'example/base_under/NormalModel') backend = model.backend commands.prepare(context, backend, model) table = backend.get_table(model) @@ -71,7 +71,7 @@ def test_prepare_base_over_level(rc: RawConfig): | | | | test | string | | 3 | open ''') - model = manifest.models['example/base_over/NormalModel'] + model = commands.get_model(context, manifest, 'example/base_over/NormalModel') backend = model.backend commands.prepare(context, backend, model) table = backend.get_table(model) @@ -95,7 +95,7 @@ def test_prepare_base_no_level(rc: RawConfig): | | | | test | string | | 3 | open ''') - model = manifest.models['example/base_no/NormalModel'] + model = commands.get_model(context, manifest, 'example/base_no/NormalModel') backend = model.backend commands.prepare(context, backend, model) table = backend.get_table(model) @@ -116,7 +116,7 @@ def test_prepare_model_ref_unique_constraint(rc: RawConfig): | | | | id | integer | | 3 | open | | | | name | string | | 3 | open ''') - model_single_unique = manifest.models['example/Continent'] + model_single_unique = commands.get_model(context, manifest, 'example/Continent') backend = model_single_unique.backend commands.prepare(context, backend, model_single_unique) table = backend.get_table(model_single_unique) @@ -124,7 +124,7 @@ def test_prepare_model_ref_unique_constraint(rc: RawConfig): [table.c['id']] == list(constraint.columns) for constraint in table.constraints if type(constraint).__name__ == 'UniqueConstraint') - model_multiple_unique = manifest.models['example/Country'] + model_multiple_unique = commands.get_model(context, manifest, 'example/Country') commands.prepare(context, backend, model_multiple_unique) table = backend.get_table(model_multiple_unique) assert any( diff --git a/tests/backends/postgresql/commands/test_migrations.py b/tests/backends/postgresql/commands/test_migrations.py index ee1aa49e0..8f56a854d 100644 --- a/tests/backends/postgresql/commands/test_migrations.py +++ b/tests/backends/postgresql/commands/test_migrations.py @@ -3,9 +3,11 @@ import pytest +from spinta.components import Context from spinta.core.config import RawConfig from spinta.manifests.tabular.helpers import striptable from spinta.testing.cli import SpintaCliRunner +from spinta.testing.context import create_test_context from spinta.testing.tabular import create_tabular_manifest from geoalchemy2.shape import to_shape @@ -43,10 +45,9 @@ def _prepare_migration_postgresql(dsn: URL) -> None: def _configure(rc, path, manifest): - override_manifest(path, manifest) url = make_url(rc.get('backends', 'default', 'dsn', required=True)) url = url.set(database=MIGRATION_DATABASE) - return rc.fork({ + rc = rc.fork({ 'manifests': { 'default': { 'type': 'tabular', @@ -63,11 +64,14 @@ def _configure(rc, path, manifest): }, }, }) + context = create_test_context(rc, name='pytest/cli') + override_manifest(context, path, manifest) + return context, rc -def override_manifest(path, manifest): +def override_manifest(context: Context, path, manifest): path = f'{path}/manifest.csv' - create_tabular_manifest(path, striptable(manifest)) + create_tabular_manifest(context, path, striptable(manifest)) def _clean_up_tables(meta: sa.MetaData, tables: list): @@ -127,11 +131,11 @@ def test_migrate_create_simple_datatype_model( initial_manifest = ''' d | r | b | m | property | type | ref | source | prepare ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -282,7 +286,7 @@ def test_migrate_add_simple_column( | | | Test | | | | | | someText | string ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -302,7 +306,7 @@ def test_migrate_add_simple_column( assert not {'someInteger'}.issubset(columns.keys()) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -365,7 +369,7 @@ def test_migrate_remove_simple_column( | | | | someText | string | | | | someInteger | integer ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -387,7 +391,7 @@ def test_migrate_remove_simple_column( assert isinstance(some_integer.type, sa.Integer) assert some_integer.nullable - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -455,7 +459,7 @@ def test_migrate_multiple_times_remove_simple_column( | | | | someText | string | | | | someInteger | integer ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -477,7 +481,7 @@ def test_migrate_multiple_times_remove_simple_column( assert isinstance(some_integer.type, sa.Integer) assert some_integer.nullable - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -525,7 +529,7 @@ def test_migrate_multiple_times_remove_simple_column( assert not {'someInteger'}.issubset(columns.keys()) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -548,7 +552,7 @@ def test_migrate_multiple_times_remove_simple_column( assert isinstance(some_integer.type, sa.Integer) assert some_integer.nullable - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -609,13 +613,13 @@ def test_migrate_model_ref_unique_constraint( initial_manifest = ''' d | r | b | m | property | type | ref ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref migrate/example | | | | | | | | | Test | | | someText @@ -723,7 +727,7 @@ def test_migrate_model_ref_unique_constraint( assert any(columns == ["someNumber"] for columns in constraint_columns) assert any(sorted(columns) == sorted(["someNumber", "someText"]) for columns in constraint_columns) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref migrate/example | | | | | | | | | Test | | | @@ -812,7 +816,7 @@ def test_migrate_add_unique_constraint( | | | Test | | | | | | someText | string ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -830,7 +834,7 @@ def test_migrate_add_unique_constraint( assert isinstance(some_text.type, sa.String) assert some_text.nullable - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -886,7 +890,7 @@ def test_migrate_remove_unique_constraint( | | | Test | | | | | | someText | string unique ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -907,7 +911,7 @@ def test_migrate_remove_unique_constraint( constraint_columns = _get_table_unique_constraint_columns(tables['migrate/example/Test']) assert any(columns == ["someText"] for columns in constraint_columns) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type migrate/example | | | | | | | | Test | | @@ -960,13 +964,13 @@ def test_migrate_create_models_with_ref( initial_manifest = ''' d | r | b | m | property | type | ref | level ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1163,7 +1167,7 @@ def test_migrate_remove_ref_column( | | | | someText | string | | | | | | someRef | ref | Test | 3 ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -1191,7 +1195,7 @@ def test_migrate_remove_ref_column( columns = tables['migrate/example/RefTwo'].columns assert {'someText', 'someRef.someText', 'someRef.someNumber'}.issubset(columns.keys()) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1246,7 +1250,7 @@ def test_migrate_remove_ref_column( constraint in columns ) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1312,7 +1316,7 @@ def test_migrate_adjust_ref_levels( | | | | someText | string | | | | | | someRef | ref | Test | 4 ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -1391,7 +1395,7 @@ def test_migrate_adjust_ref_levels( constraint in columns ) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1460,7 +1464,7 @@ def test_migrate_adjust_ref_levels( assert item["someRef.someText"] == insert_values[i]["someText"] assert item["someRef.someNumber"] == insert_values[i]["someNumber"] - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1544,13 +1548,13 @@ def test_migrate_create_models_with_base( initial_manifest = ''' d | r | b | m | property | type | ref | level ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Base | | | someText, someNumber | @@ -1673,13 +1677,13 @@ def test_migrate_remove_base_from_model( | | | Test | | | | | | | | someText | string | | ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Base | | | someText, someNumber | @@ -1734,13 +1738,13 @@ def test_migrate_create_models_with_file_type( initial_manifest = ''' d | r | b | m | property | type | ref | source ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | source migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1852,13 +1856,13 @@ def test_migrate_remove_file_type( | | | | flag | file | | | | | | new | file | | file() ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' ]) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | source migrate/example | | | | | | | | | | Test | | | someText, someNumber | @@ -1931,7 +1935,7 @@ def test_migrate_modify_geometry_type( | | | | someGeoLt | geometry(3346) | | | | | | someGeoWorld | geometry(4326) | | ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -1966,7 +1970,7 @@ def test_migrate_modify_geometry_type( assert float_equals(float(some_geo_world_values[0]), 15, epsilon=1e-2) assert float_equals(float(some_geo_world_values[1]), 15, epsilon=1e-2) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | source migrate/example | | | | | | | | | | Test | | | | @@ -2040,7 +2044,7 @@ def test_migrate_rename_model( | | | | someFile | file | | | | | | someRef | ref | Ref | ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -2058,7 +2062,7 @@ def test_migrate_rename_model( constraints = _get_table_foreign_key_constraint_columns(table) assert any(constraint["constraint_name"] == 'fk_migrate/example/Test_someRef._id' for constraint in constraints) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | source migrate/example | | | | | | | | | | NewRef | | | | @@ -2158,7 +2162,7 @@ def test_migrate_rename_property( | | | | someRef | ref | Ref | 3 | | | | someOther | ref | Ref | 4 ''' - rc = _configure(rc, tmp_path, initial_manifest) + context, rc = _configure(rc, tmp_path, initial_manifest) cli.invoke(rc, [ 'bootstrap', f'{tmp_path}/manifest.csv' @@ -2181,7 +2185,7 @@ def test_migrate_rename_property( assert any( constraint["constraint_name"] == 'fk_migrate/example/Test_someOther._id' for constraint in constraints) - override_manifest(tmp_path, ''' + override_manifest(context, tmp_path, ''' d | r | b | m | property | type | ref | level migrate/example | | | | | | | | | | Ref | | | newText | diff --git a/tests/backends/postgresql/test_lists.py b/tests/backends/postgresql/test_lists.py index a84ab5229..5386a23e4 100644 --- a/tests/backends/postgresql/test_lists.py +++ b/tests/backends/postgresql/test_lists.py @@ -7,7 +7,7 @@ def create_model(context, schema): manifest = context.get('store').manifest data = { 'type': 'model', - 'name': 'model', + 'name': 'Model', **schema, } model = Model() diff --git a/tests/backends/postgresql/test_query.py b/tests/backends/postgresql/test_query.py index 96897fcae..8fc37e527 100644 --- a/tests/backends/postgresql/test_query.py +++ b/tests/backends/postgresql/test_query.py @@ -32,7 +32,7 @@ def _build(rc: RawConfig, manifest: str, model_name: str, query: str, page_mappi backend.schema = sa.MetaData() backend.tables = {} commands.prepare(context, backend, manifest) - model = manifest.models[model_name] + model = commands.get_model(context, manifest, model_name) query = asttoexpr(spyna.parse(query)) if page_mapping: page = model.page diff --git a/tests/backends/postgresql/test_read.py b/tests/backends/postgresql/test_read.py index bc49908c1..9e74676ce 100644 --- a/tests/backends/postgresql/test_read.py +++ b/tests/backends/postgresql/test_read.py @@ -1,3 +1,4 @@ +from pathlib import Path from typing import Any from typing import Dict from typing import List @@ -15,6 +16,7 @@ from _pytest.fixtures import FixtureRequest from spinta.testing.utils import get_error_codes +import pytest def _prep_context(context: Context): @@ -47,7 +49,7 @@ def test_getall(rc: RawConfig): 'country.name': 'Lithuania', } ]) - model = manifest.models['example/City'] + model = commands.get_model(context, manifest, 'example/City') backend = model.backend query = asttoexpr(spyna.parse('select(_id, country.name)')) rows = commands.getall(context, model, backend, query=query) @@ -75,13 +77,27 @@ def test_getall(rc: RawConfig): ] -def test_getall_pagination_disabled(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_getall_pagination_disabled( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/getall/test | | | | | | | Test | | value | | | | | | value | integer | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/getall/test', ['insert', 'getall', 'search']) app.post('/example/getall/test/Test', json={'value': 0}) @@ -99,13 +115,27 @@ def test_getall_pagination_disabled(rc: RawConfig, postgresql: str, request: Fix assert len(json_response["_data"]) == 5 -def test_getall_pagination_enabled(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_getall_pagination_enabled( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/getall/test | | | | | | | Test | | value | | | | | | value | integer | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/getall/test', ['insert', 'getall', 'search']) app.post('/example/getall/test/Test', json={'value': 0}) @@ -123,13 +153,27 @@ def test_getall_pagination_enabled(rc: RawConfig, postgresql: str, request: Fixt assert len(json_response["_data"]) == 2 -def test_get_date(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_get_date( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/date/test | | | | | | | Test | | date | | | | | | date | date | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/date/test', ['insert', 'getall', 'search']) app.post('/example/date/test/Test', json={'date': '2020-01-01'}) @@ -139,13 +183,27 @@ def test_get_date(rc: RawConfig, postgresql: str, request: FixtureRequest): assert json_response['_data'][0]['date'] == '2020-01-01' -def test_get_datetime(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_get_datetime( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/datetime/test | | | | | | | Test | | date | | | | | | date | datetime | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/datetime/test', ['insert', 'getall', 'search']) app.post('/example/datetime/test/Test', json={'date': '2020-01-01T10:00:10'}) @@ -155,13 +213,27 @@ def test_get_datetime(rc: RawConfig, postgresql: str, request: FixtureRequest): assert json_response['_data'][0]['date'] == '2020-01-01T10:00:10' -def test_get_time(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_get_time( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/time/test | | | | | | | Test | | date | | | | | | date | time | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/time/test', ['insert', 'getall', 'search']) app.post('/example/time/test/Test', json={'date': '10:00:10'}) @@ -171,14 +243,28 @@ def test_get_time(rc: RawConfig, postgresql: str, request: FixtureRequest): assert json_response['_data'][0]['date'] == '10:00:10' -def test_get_paginate_with_none_simple(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_get_paginate_with_none_simple( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/page/null/simple | | | | | | | Test | | id | open | | | | | id | integer | | | | | | | name | string | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/page/null/simple', ['insert', 'getall', 'search']) app.post('/example/page/null/simple/Test', json={ @@ -239,14 +325,28 @@ def test_get_paginate_with_none_simple(rc: RawConfig, postgresql: str, request: ] -def test_get_paginate_with_none_multi_key(rc: RawConfig, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_get_paginate_with_none_multi_key( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, + request: FixtureRequest, +): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/page/null/multi | | | | | | | Test | | id, name | open | | | | | id | integer | | | | | | | name | string | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/page/null/multi', ['insert', 'getall', 'search']) app.post('/example/page/null/multi/Test', json={ @@ -325,12 +425,16 @@ def test_get_paginate_with_none_multi_key(rc: RawConfig, postgresql: str, reques ] +@pytest.mark.manifests('internal_sql', 'csv') def test_join_with_base( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | prepare | access datasets/basetest | | | | | | | | | | | | @@ -369,7 +473,13 @@ def test_join_with_base( | | | | country | ref | Country | | open | | | | testfk | ref | Test | | open | | | | | | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe', 'spinta_search', 'spinta_set_meta_fields']) @@ -454,12 +564,16 @@ def test_join_with_base( assert listdata(resp, 'name','population', 'koord', sort='name')[0] == ('Vilnius', 625349, 'POINT (54.68677 25.29067)') +@pytest.mark.manifests('internal_sql', 'csv') def test_invalid_inherit_check( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | prepare | access datasets/invalid/base | | | | | | | | | | | | @@ -481,7 +595,13 @@ def test_invalid_inherit_check( | | | | id | integer | | | open | | | | name | | | | open | | | | population | | | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe', 'spinta_search', 'spinta_set_meta_fields']) diff --git a/tests/backends/test_all.py b/tests/backends/test_all.py index 9da63e117..e34668929 100644 --- a/tests/backends/test_all.py +++ b/tests/backends/test_all.py @@ -3,8 +3,8 @@ @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_insert_get(model, app): app.authmodel(model, ['insert', 'getone']) @@ -55,8 +55,8 @@ def test_insert_get(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_update_get(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'getall']) @@ -158,8 +158,8 @@ def test_update_get(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_put_non_existant_resource(model, app): resp = app.get(f'/{model}/4e67-256f9a7388f88ccc502570f434f289e8-057553c2') @@ -173,20 +173,22 @@ def test_put_non_existant_resource(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_get_non_existant_subresource(model, context, app): app.authmodel(model, ['insert', 'getone']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'status': '42', }) assert resp.status_code == 201 id_ = resp.json()['_id'] resp = app.get(f'/{model}/{id_}/foo') + schema = '/'.join(model.split('/')[:-1]) + schema = f"{schema}/report.yml" assert resp.status_code == 404 # FIXME: Fix error message, here model and resource is found, but model # preprety is not found. @@ -197,7 +199,7 @@ def test_get_non_existant_subresource(model, context, app): 'message': "Property 'foo' not found.", 'context': { 'component': 'spinta.components.Model', - 'schema': f'{model}.yml', + 'schema': schema, 'manifest': 'default', 'model': model, 'property': 'foo', @@ -206,8 +208,8 @@ def test_get_non_existant_subresource(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_delete(model, app, tmp_path): # FIXME: `spinta_report_pdf_delete` gives access to: @@ -283,8 +285,8 @@ def test_delete(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_delete_with_list_value(model, app): app.authmodel(model, [ @@ -302,8 +304,8 @@ def test_delete_with_list_value(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_patch(model, app, context): app.authorize(['spinta_set_meta_fields']) @@ -404,8 +406,8 @@ def test_patch(model, app, context): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_escaping_chars(model, app): app.authmodel(model, ['insert', 'getone']) @@ -423,8 +425,8 @@ def test_escaping_chars(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_update_same_scalar(model, app): app.authmodel(model, ['insert', 'getone', 'update']) @@ -459,8 +461,8 @@ def test_update_same_scalar(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_update_same_obj(model, app): app.authmodel(model, ['insert', 'getone', 'update']) @@ -501,8 +503,8 @@ def test_update_same_obj(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_update_same_subresource(model, app): app.authmodel(model, ['insert', 'getone', 'update', 'subobj_update', 'subobj_get']) @@ -542,8 +544,8 @@ def test_update_same_subresource(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_update_same_array(model, app): app.authmodel(model, ['insert', 'getone', 'update']) diff --git a/tests/backends/test_fs.py b/tests/backends/test_fs.py index 5aa0ee73f..980913153 100644 --- a/tests/backends/test_fs.py +++ b/tests/backends/test_fs.py @@ -8,8 +8,8 @@ @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_crud(model, app, tmp_path): app.authmodel(model, [ @@ -104,8 +104,8 @@ def test_crud(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_add_existing_file(model, app, tmp_path): app.authmodel(model, ['insert', 'image_getone', 'image_patch']) @@ -141,8 +141,8 @@ def test_add_existing_file(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_add_missing_file(model, app, tmp_path): app.authmodel(model, ['insert', 'getone', 'image_patch']) @@ -171,8 +171,8 @@ def test_add_missing_file(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_create_hidden_image_on_insert(model, app, tmp_path): app.authmodel(model, ['insert', 'image_getone', 'image_patch']) @@ -198,8 +198,8 @@ def test_create_hidden_image_on_insert(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_add_missing_file_as_prop(model, app, tmp_path): app.authmodel(model, ['insert', 'getone', 'image_update']) @@ -231,8 +231,8 @@ def test_add_missing_file_as_prop(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_id_as_filename(model, app, tmp_path): app.authmodel(model, ['insert', 'getone', 'image_update', 'image_getone']) @@ -268,8 +268,8 @@ def test_id_as_filename(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_check_revision_for_file(model, app): app.authmodel(model, [ @@ -306,8 +306,8 @@ def test_check_revision_for_file(model, app): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_check_revision_for_file_ref(model, app, tmp_path): app.authmodel(model, ['insert', 'image_patch']) @@ -343,8 +343,8 @@ def test_check_revision_for_file_ref(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_check_extra_field(model, app, tmp_path): app.authmodel(model, ['insert', 'image_patch']) @@ -371,8 +371,8 @@ def test_check_extra_field(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_put_file_multiple_times(model, app): app.authmodel(model, ['insert', 'image_update', 'image_getone']) @@ -408,8 +408,8 @@ def test_put_file_multiple_times(model, app): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_file_get_headers(model, app): app.authmodel(model, ['insert', 'image_update', 'image_getone']) @@ -439,8 +439,8 @@ def test_file_get_headers(model, app): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_rename_non_existing_file(model, app): app.authmodel(model, ['getone', 'insert', 'image_update', 'image_patch', 'image_delete']) @@ -488,8 +488,8 @@ def test_rename_non_existing_file(model, app): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_put_file_no_content(context, model, app): app.authmodel(model, ['insert', 'image_update', 'image_getone']) @@ -517,8 +517,8 @@ def test_put_file_no_content(context, model, app): @pytest.mark.models( - # 'backends/mongo/photo', - 'backends/postgres/photo', + # 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_changelog(context, model, app): app.authmodel(model, [ @@ -561,8 +561,8 @@ def test_changelog(context, model, app): @pytest.mark.skip('NotImplemented') @pytest.mark.models( - # 'backends/mongo/photo', - 'backends/postgres/photo', + # 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_changelog_hidden_prop(context, model, app): app.authmodel(model, [ @@ -604,8 +604,8 @@ def test_changelog_hidden_prop(context, model, app): @pytest.mark.models( - 'backends/mongo/photo', - 'backends/postgres/photo', + 'backends/mongo/Photo', + 'backends/postgres/Photo', ) def test_wipe(tmp_path, model, app): app.authmodel(model, [ diff --git a/tests/backends/test_postgresql.py b/tests/backends/test_postgresql.py index c23f84ce8..38822eec3 100644 --- a/tests/backends/test_postgresql.py +++ b/tests/backends/test_postgresql.py @@ -1,3 +1,5 @@ +from pathlib import Path + import pytest from spinta.components import Model, Property @@ -58,47 +60,47 @@ def test_get_pg_sequence_name(name: str, result: str): def test_changes(app): - app.authmodel('country', ['insert', 'update', 'changes']) - data = app.post('/country', json={'_type': 'country', 'code': 'lt', 'title': "Lithuania"}).json() - app.put(f'/country/{data["_id"]}', json={'_type': 'country', '_id': data['_id'], 'title': "Lietuva"}) - app.put(f'/country/{data["_id"]}', json={'type': 'country', '_id': data['_id'], 'code': 'lv', 'title': "Latvia"}) - app.get(f'/country/{data["_id"]}/:changes').json() == {} + app.authmodel('Country', ['insert', 'update', 'changes']) + data = app.post('/Country', json={'_type': 'Country', 'code': 'lt', 'title': "Lithuania"}).json() + app.put(f'/Country/{data["_id"]}', json={'_type': 'Country', '_id': data['_id'], 'title': "Lietuva"}) + app.put(f'/Country/{data["_id"]}', json={'type': 'Country', '_id': data['_id'], 'code': 'lv', 'title': "Latvia"}) + app.get(f'/Country/{data["_id"]}/:changes').json() == {} def test_delete(context, app): - app.authmodel('country', ['insert', 'getall', 'delete']) + app.authmodel('Country', ['insert', 'getall', 'delete']) resp = app.post('/', json={ '_data': [ - {'_op': 'insert', '_type': 'country', 'code': 'fi', 'title': 'Finland'}, - {'_op': 'insert', '_type': 'country', 'code': 'lt', 'title': 'Lithuania'}, + {'_op': 'insert', '_type': 'Country', 'code': 'fi', 'title': 'Finland'}, + {'_op': 'insert', '_type': 'Country', 'code': 'lt', 'title': 'Lithuania'}, ], }) ids = [x['_id'] for x in resp.json()['_data']] revs = [x['_revision'] for x in resp.json()['_data']] - resp = app.get('/country').json() + resp = app.get('/Country').json() data = [x['_id'] for x in resp['_data']] assert ids[0] in data assert ids[1] in data # XXX: DELETE method should not include a request body. - resp = app.request('DELETE', f'/country/{ids[0]}', json={ + resp = app.request('DELETE', f'/Country/{ids[0]}', json={ '_revision': revs[0], }) assert resp.status_code == 204 # multiple deletes should just return HTTP/404 - resp = app.delete(f'/country/{ids[0]}') + resp = app.delete(f'/Country/{ids[0]}') assert resp.status_code == 404 assert get_error_codes(resp.json()) == ['ItemDoesNotExist'] assert get_error_context(resp.json(), 'ItemDoesNotExist', ['manifest', 'model', 'id']) == { 'manifest': 'default', - 'model': 'country', + 'model': 'Country', 'id': ids[0], } - resp = app.get('/country').json() + resp = app.get('/Country').json() data = [x['_id'] for x in resp['_data']] assert ids[0] not in data assert ids[1] in data @@ -114,13 +116,13 @@ def test_patch(app): 'spinta_org_patch', ]) - country_data = app.post('/country', json={ - '_type': 'country', + country_data = app.post('/Country', json={ + '_type': 'Country', 'code': 'lt', 'title': 'Lithuania', }).json() - org_data = app.post('/org', json={ - '_type': 'org', + org_data = app.post('/Org', json={ + '_type': 'Org', 'title': 'My Org', 'govid': '0042', 'country': { @@ -129,7 +131,7 @@ def test_patch(app): }).json() id_ = org_data['_id'] - resp = app.patch(f'/org/{org_data["_id"]}', json={ + resp = app.patch(f'/Org/{org_data["_id"]}', json={ '_revision': org_data['_revision'], 'title': 'foo org', }) @@ -139,7 +141,7 @@ def test_patch(app): assert org_data['_revision'] != revision # test that revision mismatch is checked - resp = app.patch(f'/org/{org_data["_id"]}', json={ + resp = app.patch(f'/Org/{org_data["_id"]}', json={ '_revision': 'r3v1510n', 'title': 'foo org', }) @@ -148,25 +150,25 @@ def test_patch(app): assert get_error_context(resp.json(), "ConflictingValue", ["given", "expected", "model"]) == { 'given': 'r3v1510n', 'expected': revision, - 'model': 'org', + 'model': 'Org', } # test that type mismatch is checked - resp = app.patch(f'/org/{org_data["_id"]}', json={ - '_type': 'country', + resp = app.patch(f'/Org/{org_data["_id"]}', json={ + '_type': 'Country', '_revision': org_data['_revision'], 'title': 'foo org', }) assert resp.status_code == 409 assert get_error_codes(resp.json()) == ['ConflictingValue'] assert get_error_context(resp.json(), 'ConflictingValue', ['given', 'expected', 'model']) == { - 'given': 'country', - 'expected': 'org', - 'model': 'org', + 'given': 'Country', + 'expected': 'Org', + 'model': 'Org', } # test that id mismatch is checked - resp = app.patch(f'/org/{org_data["_id"]}', json={ + resp = app.patch(f'/Org/{org_data["_id"]}', json={ '_id': '0007ddec-092b-44b5-9651-76884e6081b4', '_revision': revision, 'title': 'foo org', @@ -175,7 +177,7 @@ def test_patch(app): data = resp.json() assert data['_revision'] != revision assert data == { - '_type': 'org', + '_type': 'Org', '_id': '0007ddec-092b-44b5-9651-76884e6081b4', '_revision': data['_revision'], } @@ -183,9 +185,9 @@ def test_patch(app): revision = data['_revision'] # patch using same values as already stored in database - resp = app.patch(f'/org/{id_}', json={ + resp = app.patch(f'/Org/{id_}', json={ '_id': id_, - '_type': 'org', + '_type': 'Org', '_revision': revision, 'title': 'foo org', }) @@ -193,24 +195,34 @@ def test_patch(app): resp_data = resp.json() assert resp_data['_id'] == id_ - assert resp_data['_type'] == 'org' + assert resp_data['_type'] == 'Org' # title have not changed, so should not be included in result assert 'title' not in resp_data # revision must be the same, since nothing has changed assert resp_data['_revision'] == revision +@pytest.mark.manifests('internal_sql', 'ascii') def test_exceptions_unique_constraint_single_column( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/unique/single | | | | | | | Country | | name | | | | | | name | string | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/unique/single', ['insert']) @@ -237,18 +249,28 @@ def test_exceptions_unique_constraint_single_column( } +@pytest.mark.manifests('internal_sql', 'ascii') def test_exceptions_unique_constraint_multiple_columns( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/unique/multiple | | | | | | | Country | | name, id | | | | | | name | string | | open | | | | | id | integer | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/unique/multiple', ['insert']) diff --git a/tests/cli/test_copy.py b/tests/cli/test_copy.py index 6596202d4..002a7e058 100644 --- a/tests/cli/test_copy.py +++ b/tests/cli/test_copy.py @@ -1,14 +1,16 @@ from pathlib import Path +from spinta.components import Context from spinta.core.config import RawConfig from spinta.testing.cli import SpintaCliRunner from spinta.manifests.tabular.helpers import striptable +from spinta.testing.context import create_test_context from spinta.testing.tabular import create_tabular_manifest from spinta.testing.manifest import load_manifest -def test_copy(rc, cli: SpintaCliRunner, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy(context: Context, rc, cli: SpintaCliRunner, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | data | sql | | | | @@ -54,8 +56,8 @@ def test_copy(rc, cli: SpintaCliRunner, tmp_path): ''' -def test_copy_enum_0(rc, cli: SpintaCliRunner, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_enum_0(context: Context, rc, cli: SpintaCliRunner, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | data | sql | | | | @@ -88,8 +90,8 @@ def test_copy_enum_0(rc, cli: SpintaCliRunner, tmp_path): ''' -def test_copy_global_enum(rc, cli: SpintaCliRunner, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_global_enum(context: Context, rc, cli: SpintaCliRunner, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | enum | direction | l | 0 | @@ -122,26 +124,26 @@ def test_copy_global_enum(rc, cli: SpintaCliRunner, tmp_path): ''' -def test_copy_with_filters_and_externals(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_with_filters_and_externals(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | data | sql | | | | | | | | | - | | | country | | code | salis | code='lt' | + | | | Country | | code | salis | code='lt' | | | | | code | string | | kodas | | private | | | | name | string | | pavadinimas | | open | | | | driving | string | | vairavimas | | open | | | | | enum | | l | 'left' | private | | | | | | | r | 'right' | open | | | | | - | | | city | | name | miestas | | + | | | City | | name | miestas | | | | | | name | string | | pavadinimas | | open - | | | | country | ref | country | salis | | open + | | | | country | ref | Country | salis | | open | | | | | - | | | capital | | name | miestas | | + | | | Capital | | name | miestas | | | | | | name | string | | pavadinimas | | - | | | | country | ref | country | salis | | + | | | | country | ref | Country | salis | | ''')) cli.invoke(rc, [ @@ -157,19 +159,19 @@ def test_copy_with_filters_and_externals(rc, cli, tmp_path): datasets/gov/example | | | | | | | data | sql | | | | | | | | | | | - | | | country | | | salis | code='lt' | | + | | | Country | | | salis | code='lt' | | | | | | name | string | | pavadinimas | | | open | | | | driving | string | | vairavimas | | | open | enum | | r | 'right' | | open | | | | | | - | | | city | | name | miestas | | | + | | | City | | name | miestas | | | | | | | name | string | | pavadinimas | | | open - | | | | country | ref | country | salis | | | open + | | | | country | ref | Country | salis | | | open ''' -def test_copy_and_format_names(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_and_format_names(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | level | access | title datasets/gov/example | | | | | | | Example dataset | data | sql | | | | | | @@ -214,8 +216,8 @@ def test_copy_and_format_names(rc, cli, tmp_path): ''' -def test_copy_and_format_names_for_ref(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_and_format_names_for_ref(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | prepare datasets/gov/example | | | | data | sql | | @@ -258,8 +260,8 @@ def test_copy_and_format_names_for_ref(rc, cli, tmp_path): ''' -def test_copy_and_format_names_with_formulas(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_and_format_names_with_formulas(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | prepare datasets/gov/example | | | data | sql | @@ -286,7 +288,7 @@ def test_copy_and_format_names_with_formulas(rc, cli, tmp_path): ''' -def test_copy_to_stdout(rc, cli, tmp_path): +def test_copy_to_stdout(context: Context, rc, cli, tmp_path): manifest = striptable(''' d | r | b | m | property | type datasets/gov/example | @@ -295,7 +297,7 @@ def test_copy_to_stdout(rc, cli, tmp_path): | | | City | | | | | name | string ''') - create_tabular_manifest(tmp_path / 'manifest.csv', manifest) + create_tabular_manifest(context, tmp_path / 'manifest.csv', manifest) result = cli.invoke(rc, [ 'copy', @@ -306,8 +308,8 @@ def test_copy_to_stdout(rc, cli, tmp_path): assert result.stdout.strip() == manifest -def test_copy_order_by_access(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_order_by_access(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | data | sql | | | | @@ -364,8 +366,8 @@ def test_copy_order_by_access(rc, cli, tmp_path): ''' -def test_copy_rename_duplicates(rc, cli, tmp_path): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_copy_rename_duplicates(context: Context, rc, cli, tmp_path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type datasets/gov/example | | data | sql @@ -404,8 +406,8 @@ def test_copy_rename_duplicates(rc, cli, tmp_path): ''' -def test_enum_ref(rc: RawConfig, cli: SpintaCliRunner, tmp_path: Path): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' +def test_enum_ref(context: Context, rc: RawConfig, cli: SpintaCliRunner, tmp_path: Path): + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | prepare | access | title | enum | sex | | 1 | | Male | | | | 2 | | Female diff --git a/tests/cli/test_helpers.py b/tests/cli/test_helpers.py index 9ec3585db..1e99ff603 100644 --- a/tests/cli/test_helpers.py +++ b/tests/cli/test_helpers.py @@ -11,7 +11,8 @@ def test_configure(tmp_path: Path, rc: RawConfig): - create_tabular_manifest(tmp_path / 'm1.csv', striptable(''' + context: Context = create_test_context(rc) + create_tabular_manifest(context, tmp_path / 'm1.csv', striptable(''' d | r | b | m | property | type | source datasets/1 | | | data | sql | @@ -20,7 +21,7 @@ def test_configure(tmp_path: Path, rc: RawConfig): | | | | name | string | PAVADINIMAS ''')) - create_tabular_manifest(tmp_path / 'm2.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'm2.csv', striptable(''' d | r | b | m | property | type | source datasets/2 | | | data | sql | @@ -28,8 +29,6 @@ def test_configure(tmp_path: Path, rc: RawConfig): | | | Country | | SALIS | | | | name | string | PAVADINIMAS ''')) - - context: Context = create_test_context(rc) context = configure_context(context, [ str(tmp_path / 'm1.csv'), str(tmp_path / 'm2.csv'), diff --git a/tests/cli/test_init.py b/tests/cli/test_init.py index ef59d4165..bc1901efe 100644 --- a/tests/cli/test_init.py +++ b/tests/cli/test_init.py @@ -1,12 +1,12 @@ from spinta.manifests.tabular.helpers import render_tabular_manifest from spinta.testing.cli import SpintaCliRunner from spinta.manifests.tabular.helpers import striptable -from spinta.testing.manifest import load_manifest +from spinta.testing.manifest import load_manifest, load_manifest_and_context def test_show(rc, cli: SpintaCliRunner, tmp_path): cli.invoke(rc, ['init', tmp_path / 'manifest.csv']) - manifest = load_manifest(rc, tmp_path / 'manifest.csv') - assert render_tabular_manifest(manifest) == striptable(''' + context, manifest = load_manifest_and_context(rc, tmp_path / 'manifest.csv') + assert render_tabular_manifest(context, manifest) == striptable(''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description ''') diff --git a/tests/cli/test_push.py b/tests/cli/test_push.py index 418fec68a..c2e07f2fe 100644 --- a/tests/cli/test_push.py +++ b/tests/cli/test_push.py @@ -125,6 +125,7 @@ def errordb(): def test_push_with_progress_bar( + context, postgresql, rc, cli: SpintaCliRunner, @@ -133,7 +134,7 @@ def test_push_with_progress_bar( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | | @@ -166,6 +167,7 @@ def test_push_with_progress_bar( def test_push_without_progress_bar( + context, postgresql, rc, cli: SpintaCliRunner, @@ -174,7 +176,7 @@ def test_push_without_progress_bar( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | | @@ -205,6 +207,7 @@ def test_push_without_progress_bar( def test_push_error_exit_code( + context, postgresql, rc, cli: SpintaCliRunner, @@ -213,7 +216,7 @@ def test_push_error_exit_code( errordb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | | @@ -242,6 +245,7 @@ def test_push_error_exit_code( def test_push_error_exit_code_with_bad_resource( + context, postgresql, rc, cli: SpintaCliRunner, @@ -249,7 +253,7 @@ def test_push_error_exit_code_with_bad_resource( tmp_path, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(f''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(f''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | sqlite:///{tmp_path}/bad.db | @@ -294,6 +298,7 @@ def test_push_error_exit_code_with_bad_resource( def test_push_ref_with_level_3( + context, postgresql, rc, cli: SpintaCliRunner, @@ -302,7 +307,7 @@ def test_push_ref_with_level_3( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access level3dataset | | | | | | db | sql | | | | @@ -345,6 +350,7 @@ def test_push_ref_with_level_3( def test_push_ref_with_level_4( + context, postgresql, rc, cli: SpintaCliRunner, @@ -353,7 +359,7 @@ def test_push_ref_with_level_4( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access level4dataset | | | | | | db | sql | | | | @@ -396,6 +402,7 @@ def test_push_ref_with_level_4( def test_push_with_resource_check( + context, postgresql, rc, cli: SpintaCliRunner, @@ -404,16 +411,16 @@ def test_push_with_resource_check( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | access datasets/gov/exampleRes | | | | | data | sql | | | - | | | countryRes | | code | salis | + | | | CountryRes | | code | salis | | | | | code | string | | kodas | open | | | | name | string | | pavadinimas | open | | | | | | datasets/gov/exampleNoRes | | | | - | | | countryNoRes | | | | + | | | CountryNoRes | | | | | | | | code | string | | | open | | | | name | string | | | open ''')) @@ -445,16 +452,17 @@ def test_push_with_resource_check( ]) assert result.exit_code == 0 - remote.app.authmodel('datasets/gov/exampleRes/countryRes', ['getall']) - resp_res = remote.app.get('/datasets/gov/exampleRes/countryRes') + remote.app.authmodel('datasets/gov/exampleRes/CountryRes', ['getall']) + resp_res = remote.app.get('/datasets/gov/exampleRes/CountryRes') assert len(listdata(resp_res)) == 3 - remote.app.authmodel('datasets/gov/exampleNoRes/countryNoRes', ['getall']) - resp_no_res = remote.app.get('/datasets/gov/exampleNoRes/countryNoRes') + remote.app.authmodel('datasets/gov/exampleNoRes/CountryNoRes', ['getall']) + resp_no_res = remote.app.get('/datasets/gov/exampleNoRes/CountryNoRes') assert len(listdata(resp_no_res)) == 0 def test_push_ref_with_level_no_source( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -477,7 +485,7 @@ def test_push_ref_with_level_no_source( | | | | code | string | | | 4 | open | | | | name | string | | | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) app = create_client(rc, tmp_path, geodb) app.authmodel('leveldataset', ['getall']) @@ -509,6 +517,7 @@ def test_push_ref_with_level_no_source( def test_push_ref_with_level_no_source_status_code_400_check( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -532,7 +541,7 @@ def test_push_ref_with_level_no_source_status_code_400_check( | | | | name | string | | | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) app = create_client(rc, tmp_path, geodb) app.authmodel('leveldataset', ['getall']) @@ -562,6 +571,7 @@ def test_push_ref_with_level_no_source_status_code_400_check( def test_push_pagination_incremental( + context, postgresql, rc, cli: SpintaCliRunner, @@ -570,7 +580,7 @@ def test_push_pagination_incremental( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access paginated | | | | | | db | sql | | | | @@ -615,6 +625,7 @@ def test_push_pagination_incremental( def test_push_pagination_without_incremental( + context, postgresql, rc, cli: SpintaCliRunner, @@ -623,7 +634,7 @@ def test_push_pagination_without_incremental( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access paginated/without | | | | | | db | sql | | | | @@ -667,6 +678,7 @@ def test_push_pagination_without_incremental( def test_push_pagination_incremental_with_page_valid( + context, postgresql, rc, cli: SpintaCliRunner, @@ -675,7 +687,7 @@ def test_push_pagination_incremental_with_page_valid( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access paginated/valid | | | | | | db | sql | | | | @@ -725,6 +737,7 @@ def test_push_pagination_incremental_with_page_valid( def test_push_pagination_incremental_with_page_invalid( + context, postgresql, rc, cli: SpintaCliRunner, @@ -733,7 +746,7 @@ def test_push_pagination_incremental_with_page_invalid( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access paginated/invalid | | | | | | db | sql | | | | @@ -767,6 +780,7 @@ def test_push_pagination_incremental_with_page_invalid( def test_push_with_base( + context, postgresql, rc, cli: SpintaCliRunner, @@ -775,7 +789,7 @@ def test_push_with_base( request, base_geodb ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access level4basedataset | | | | | | db | sql | | | | @@ -827,6 +841,7 @@ def test_push_with_base( def test_push_with_base_different_ref( + context, postgresql, rc, cli: SpintaCliRunner, @@ -835,7 +850,7 @@ def test_push_with_base_different_ref( request, base_geodb ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access level4basedatasetref | | | | | | db | sql | | | | @@ -887,6 +902,7 @@ def test_push_with_base_different_ref( def test_push_with_base_level_3( + context, postgresql, rc, cli: SpintaCliRunner, @@ -895,7 +911,7 @@ def test_push_with_base_level_3( request, base_geodb ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | base | m | property | type | ref | source | level | access level3basedataset | | | | | | db | | | sql | | | | @@ -948,6 +964,7 @@ def test_push_with_base_level_3( def test_push_sync( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -970,7 +987,7 @@ def test_push_sync( | | | | code | integer | | | 4 | open | | | | name | string | | | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) # Configure local server with SQL backend localrc = create_rc(rc, tmp_path, geodb) @@ -1023,6 +1040,7 @@ def test_push_sync( @pytest.mark.skip("Private now sends warning that model has been skipped syncing rather throwing exception") def test_push_sync_to_private_error( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -1046,7 +1064,7 @@ def test_push_sync_to_private_error( | | | | code | integer | | | 4 | private | | | | name | string | | | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) # Configure local server with SQL backend localrc = create_rc(rc, tmp_path, geodb) @@ -1068,6 +1086,7 @@ def test_push_sync_to_private_error( def test_push_sync_private_no_error( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -1090,7 +1109,7 @@ def test_push_sync_private_no_error( | | | | code | integer | | | 4 | private | | | | name | string | | | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) # Configure local server with SQL backend localrc = create_rc(rc, tmp_path, geodb) @@ -1112,6 +1131,7 @@ def test_push_sync_private_no_error( def test_push_with_text( + context, postgresql, rc, cli: SpintaCliRunner, @@ -1120,7 +1140,7 @@ def test_push_with_text( request, text_geodb ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access textnormal | | | | | | db | sql | | | | @@ -1158,6 +1178,7 @@ def test_push_with_text( def test_push_with_text_unknown( + context, postgresql, rc, cli: SpintaCliRunner, @@ -1166,7 +1187,7 @@ def test_push_with_text_unknown( request, text_geodb ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access textunknown | | | | | | db | sql | | | | @@ -1203,6 +1224,7 @@ def test_push_with_text_unknown( def test_push_postgresql( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -1239,7 +1261,7 @@ def test_push_postgresql( | | | | id | integer | | id | 4 | open | | | | name | string | | name | 2 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) # Configure local server with SQL backend tmp = Sqlite(db) @@ -1271,6 +1293,7 @@ def test_push_postgresql( def test_push_with_nulls( + context, postgresql, rc, cli: SpintaCliRunner, @@ -1279,7 +1302,7 @@ def test_push_with_nulls( geodb, request ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | level | access nullpush | | | | | | db | sql | | | | diff --git a/tests/cli/test_show.py b/tests/cli/test_show.py index f5fb1222f..fda05fa9c 100644 --- a/tests/cli/test_show.py +++ b/tests/cli/test_show.py @@ -3,7 +3,7 @@ from spinta.testing.tabular import create_tabular_manifest -def test_show(rc, cli: SpintaCliRunner, tmp_path): +def test_show(context, rc, cli: SpintaCliRunner, tmp_path): manifest = striptable(''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | datasets/gov/example | | | | | | protected | | | @@ -22,7 +22,7 @@ def test_show(rc, cli: SpintaCliRunner, tmp_path): | | | | | country | ref | Country | salis | | | protected | | | ''') - create_tabular_manifest(tmp_path / 'manifest.csv', manifest) + create_tabular_manifest(context, tmp_path / 'manifest.csv', manifest) result = cli.invoke(rc, ['show', tmp_path / 'manifest.csv']) diff --git a/tests/commands/load/test_array.py b/tests/commands/load/test_array.py index 4c0359239..43f036d40 100644 --- a/tests/commands/load/test_array.py +++ b/tests/commands/load/test_array.py @@ -10,7 +10,7 @@ def create_model(context, schema): manifest = context.get('store').manifest data = { 'type': 'model', - 'name': 'model', + 'name': 'Model', **schema, } model = Model() diff --git a/tests/commands/test_build_data_patch_for_write.py b/tests/commands/test_build_data_patch_for_write.py index 6fb428c54..47bb85be6 100644 --- a/tests/commands/test_build_data_patch_for_write.py +++ b/tests/commands/test_build_data_patch_for_write.py @@ -9,7 +9,7 @@ def create_model(context, schema): manifest = context.get('store').manifest data = { 'type': 'model', - 'name': 'model', + 'name': 'Model', **schema, } model = Model() diff --git a/tests/commands/test_build_full_response.py b/tests/commands/test_build_full_response.py index d152eb5ed..16e14900c 100644 --- a/tests/commands/test_build_full_response.py +++ b/tests/commands/test_build_full_response.py @@ -7,7 +7,7 @@ def create_model(context, schema): manifest = context.get('store').manifest data = { 'type': 'model', - 'name': 'model', + 'name': 'Model', **schema, } model = Model() diff --git a/tests/datasets/sql/test_query.py b/tests/datasets/sql/test_query.py index 9a89c3319..9a5269237 100644 --- a/tests/datasets/sql/test_query.py +++ b/tests/datasets/sql/test_query.py @@ -5,8 +5,9 @@ from sqlalchemy.sql import Select from sqlalchemy.sql.type_api import TypeEngine +from spinta import commands from spinta.auth import AdminToken -from spinta.components import Model, Mode +from spinta.components import Model, Mode, Context from spinta.core.config import RawConfig from spinta.datasets.backends.sql.commands.query import SqlQueryBuilder from spinta.datasets.backends.sql.components import Sql @@ -51,9 +52,9 @@ def _get_model_db_name(model: Model) -> str: return model.get_name_without_ns().upper() -def _meta_from_manifest(manifest: Manifest) -> sa.MetaData: +def _meta_from_manifest(context: Context, manifest: Manifest) -> sa.MetaData: meta = sa.MetaData() - for model in manifest.models.values(): + for model in commands.get_models(context, manifest).values(): columns = [ sa.Column(prop.external.name, _get_sql_type(prop.dtype)) for name, prop in model.properties.items() @@ -70,8 +71,8 @@ def _meta_from_manifest(manifest: Manifest) -> sa.MetaData: def _build(rc: RawConfig, manifest: str, model_name: str, page_mapping: dict = None) -> str: context, manifest = load_manifest_and_context(rc, manifest, mode=Mode.external) context.set('auth.token', AdminToken()) - model = manifest.models[model_name] - meta = _meta_from_manifest(manifest) + model = commands.get_model(context, manifest, model_name) + meta = _meta_from_manifest(context, manifest) backend = Sql() backend.schema = meta query = model.external.prepare diff --git a/tests/datasets/sql/test_read.py b/tests/datasets/sql/test_read.py index 02993a1be..b56d0c4ed 100644 --- a/tests/datasets/sql/test_read.py +++ b/tests/datasets/sql/test_read.py @@ -1,3 +1,4 @@ +from spinta import commands from spinta.core.config import RawConfig from spinta.datasets.backends.sql.commands.read import _get_row_value from spinta.exceptions import TooShortPageSize @@ -64,13 +65,13 @@ def test__get_row_value_null(rc: RawConfig): | | | | | | | 2 | 2 | ''') row = ["Vilnius", None] - model = manifest.models['example/City'] + model = commands.get_model(context, manifest, 'example/City') sel = Selected(1, model.properties['rating']) assert _get_row_value(context, row, sel) is None -def test_getall_paginate_null_check_value(rc, tmp_path, geodb_null_check): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_null_check_value(context, rc, tmp_path, geodb_null_check): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate | | | | | | | data | | sql | | | @@ -88,8 +89,8 @@ def test_getall_paginate_null_check_value(rc, tmp_path, geodb_null_check): ] -def test_getall_paginate_with_nulls_page_too_small(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls_page_too_small(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate | | | | | | | data | | sql | | | @@ -109,8 +110,8 @@ def test_getall_paginate_with_nulls_page_too_small(rc, tmp_path, geodb_with_null assert isinstance(exceptions[0], TooShortPageSize) -def test_getall_paginate_with_nulls(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate/null0 | | | | | | | data | | sql | | | @@ -139,8 +140,8 @@ def test_getall_paginate_with_nulls(rc, tmp_path, geodb_with_nulls): ] -def test_getall_paginate_with_nulls_multi_key(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls_multi_key(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate/null1 | | | | | | | data | | sql | | | @@ -169,8 +170,8 @@ def test_getall_paginate_with_nulls_multi_key(rc, tmp_path, geodb_with_nulls): ] -def test_getall_paginate_with_nulls_all_keys(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls_all_keys(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate/null1 | | | | | | | data | | sql | | | @@ -199,8 +200,8 @@ def test_getall_paginate_with_nulls_all_keys(rc, tmp_path, geodb_with_nulls): ] -def test_getall_paginate_with_nulls_and_sort(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls_and_sort(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate/null2 | | | | | | | data | | sql | | | @@ -229,8 +230,8 @@ def test_getall_paginate_with_nulls_and_sort(rc, tmp_path, geodb_with_nulls): ] -def test_getall_paginate_with_nulls_unique(rc, tmp_path, geodb_with_nulls): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall_paginate_with_nulls_unique(context, rc, tmp_path, geodb_with_nulls): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | external/paginate/null3 | | | | | | | data | | sql | | | diff --git a/tests/datasets/sql/test_ufunc.py b/tests/datasets/sql/test_ufunc.py index b479ee3f7..8451a49a3 100644 --- a/tests/datasets/sql/test_ufunc.py +++ b/tests/datasets/sql/test_ufunc.py @@ -4,6 +4,7 @@ import sqlalchemy as sa +from spinta import commands from spinta.core.config import RawConfig from spinta.datasets.backends.sql.ufuncs.components import SqlResultBuilder from spinta.testing.manifest import load_manifest_and_context @@ -26,7 +27,7 @@ def test_cast_integer(rc: RawConfig, value): | | | Data | | | | | | | | value | integer | | | ''') - dtype = manifest.models['example/Data'].properties['value'].dtype + dtype = commands.get_model(context, manifest, 'example/Data').properties['value'].dtype env = SqlResultBuilder(context) env.call('cast', dtype, value) @@ -42,7 +43,7 @@ def test_cast_integer_error(rc: RawConfig, value): | | | Data | | | | | | | | value | integer | | | ''') - dtype = manifest.models['example/Data'].properties['value'].dtype + dtype = commands.get_model(context, manifest, 'example/Data').properties['value'].dtype env = SqlResultBuilder(context) with pytest.raises(UnableToCast) as e: env.call('cast', dtype, value) @@ -63,7 +64,7 @@ def test_point(rc: RawConfig): context.set('auth.token', AdminToken()) model_name = 'example/Data' - model = manifest.models[model_name] + model = commands.get_model(context, manifest, model_name) env = SqlQueryBuilder(context) env.update(model=model) diff --git a/tests/datasets/test_geojson.py b/tests/datasets/test_geojson.py index f69d37ae3..17372e3d7 100644 --- a/tests/datasets/test_geojson.py +++ b/tests/datasets/test_geojson.py @@ -14,7 +14,7 @@ def test_geojson_resource(rc: RawConfig): | | | | name | string | | NAME | open ''' context, manifest = load_manifest_and_context(rc, table, mode=Mode.external) - backend = manifest.models['example/City'].backend + backend = commands.get_model(context, manifest, 'example/City').backend assert backend.type == 'geojson' assert manifest == table diff --git a/tests/datasets/test_html.py b/tests/datasets/test_html.py index d5890085f..c13b4ace4 100644 --- a/tests/datasets/test_html.py +++ b/tests/datasets/test_html.py @@ -14,7 +14,7 @@ def test_html(rc: RawConfig): | | | | name | string | | td | open ''' context, manifest = load_manifest_and_context(rc, table, mode=Mode.external) - backend = manifest.models['example/City'].backend + backend = commands.get_model(context, manifest, 'example/City').backend assert backend.type == 'html' assert manifest == table diff --git a/tests/datasets/test_sql.py b/tests/datasets/test_sql.py index dec49c646..2fc0d7a50 100644 --- a/tests/datasets/test_sql.py +++ b/tests/datasets/test_sql.py @@ -114,27 +114,27 @@ def geodb_denorm(): yield db -def test_filter(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code='lt' | | code | | | | Country | + | | | | Country | salis | code='lt' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('lt', 'Lietuva'), ] -def test_filter_join(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_join(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | sql | | | | | | | Data | @@ -162,11 +162,12 @@ def test_filter_join(rc, tmp_path, geodb): def test_filter_join_nested( + context, rc: RawConfig, tmp_path: pathlib.Path, sqlite: Sqlite, ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access example/join/nested | | | | | | data | sql | | | | @@ -216,27 +217,27 @@ def test_filter_join_nested( ] -def test_filter_join_array_value(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_join_array_value(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | | | | | | | | | | | | | - | | | | city | miestas | country.code=['lt','lv'] | | name | | | | City | + | | | | City | miestas | country.code=['lt','lv'] | | name | | | | City | | | | | | name | pavadinimas | | string | | 3 | open | | Name | - | | | | | country | salis | | ref | country | 4 | open | | Country | + | | | | | country | salis | | ref | Country | 4 | open | | Country | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') codes = dict(listdata(resp, '_id', 'code')) - resp = app.get('/datasets/gov/example/city?sort(name)') + resp = app.get('/datasets/gov/example/City?sort(name)') data = listdata(resp, 'country._id', 'name', sort='name') data = [(codes.get(country), city) for country, city in data] assert data == [ @@ -245,27 +246,27 @@ def test_filter_join_array_value(rc, tmp_path, geodb): ] -def test_filter_join_ne_array_value(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_join_ne_array_value(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | | | | | | | | | | | | | - | | | | city | miestas | country.code!=['lt','lv'] | | name | | | | City | + | | | | City | miestas | country.code!=['lt','lv'] | | name | | | | City | | | | | | name | pavadinimas | | string | | 3 | open | | Name | - | | | | | country | salis | | ref | country | 4 | open | | Country | + | | | | | country | salis | | ref | Country | 4 | open | | Country | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') codes = dict(listdata(resp, '_id', 'code')) - resp = app.get('/datasets/gov/example/city?sort(name)') + resp = app.get('/datasets/gov/example/City?sort(name)') data = listdata(resp, 'country._id', 'name', sort='name') data = [(codes.get(country), city) for country, city in data] assert data == [ @@ -273,9 +274,8 @@ def test_filter_join_ne_array_value(rc, tmp_path, geodb): ] -@pytest.mark.skip('todo') -def test_filter_multi_column_pk(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_multi_column_pk(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | keymap | | | | | | | | Example | | | data | | | sql | | | | | Data | @@ -302,55 +302,55 @@ def test_filter_multi_column_pk(rc, tmp_path, geodb): ] -def test_getall(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_getall(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | | | | | | | | | | | | | - | | | | city | miestas | | | name | | | | City | + | | | | City | miestas | | | name | | | | City | | | | | | name | pavadinimas | | string | | 3 | open | | Name | - | | | | | country | salis | | ref | country | 4 | open | | Country | + | | | | | country | salis | | ref | Country | 4 | open | | Country | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?sort(code)') + resp = app.get('/datasets/gov/example/Country?sort(code)') codes = dict(listdata(resp, '_id', 'code')) assert listdata(resp, 'code', 'name', '_type') == [ - ('ee', 'Estija', 'datasets/gov/example/country'), - ('lt', 'Lietuva', 'datasets/gov/example/country'), - ('lv', 'Latvija', 'datasets/gov/example/country'), + ('ee', 'Estija', 'datasets/gov/example/Country'), + ('lt', 'Lietuva', 'datasets/gov/example/Country'), + ('lv', 'Latvija', 'datasets/gov/example/Country'), ] - resp = app.get('/datasets/gov/example/city?sort(name)') + resp = app.get('/datasets/gov/example/City?sort(name)') data = listdata(resp, 'country._id', 'name', '_type', sort='name') data = [(codes.get(country), city, _type) for country, city, _type in data] assert data == [ - ('lv', 'Ryga', 'datasets/gov/example/city'), - ('ee', 'Talinas', 'datasets/gov/example/city'), - ('lt', 'Vilnius', 'datasets/gov/example/city'), + ('lv', 'Ryga', 'datasets/gov/example/City'), + ('ee', 'Talinas', 'datasets/gov/example/City'), + ('lt', 'Vilnius', 'datasets/gov/example/City'), ] -def test_select(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_select(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?select(code,name)') + resp = app.get('/datasets/gov/example/Country?select(code,name)') assert listdata(resp, 'code', 'name') == [ ('ee', 'Estija'), ('lt', 'Lietuva'), @@ -359,20 +359,20 @@ def test_select(rc, tmp_path, geodb): @pytest.mark.skip('TODO') -def test_select_len(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_select_len(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?select(code,len(name))') + resp = app.get('/datasets/gov/example/Country?select(code,len(name))') assert listdata(resp, 'code', 'len(name)') == [ ('ee', 6), ('lt', 7), @@ -380,153 +380,153 @@ def test_select_len(rc, tmp_path, geodb): ] -def test_filter_len(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_len(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | | | code | | | | Country | + | | | | Country | salis | | | code | | | | Country | | | | | | code | kodas | | string | | 3 | open | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?select(code,name)&len(name)=7&sort(code)') + resp = app.get('/datasets/gov/example/Country?select(code,name)&len(name)=7&sort(code)') assert listdata(resp, 'code', 'name') == [ ('lt', 'Lietuva'), ('lv', 'Latvija'), ] -def test_private_property(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_private_property(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | private | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp) == [ 'Latvija', 'Lietuva', ] -def test_all_private_properties(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_all_private_properties(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | private | | Code | | | | | | name | pavadinimas | | string | | 3 | private | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert error(resp, status=401) == 'AuthorizedClientsOnly' -def test_default_access(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_default_access(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | | | Code | | | | | | name | pavadinimas | | string | | 3 | | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert error(resp, status=401) == 'AuthorizedClientsOnly' -def test_model_open_access(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_model_open_access(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | open | | Country | + | | | | Country | salis | code!='ee' | | code | | open | | Country | | | | | | code | kodas | | string | | 3 | | | Code | | | | | | name | pavadinimas | | string | | 3 | | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp) == [ ('lt', 'Lietuva'), ('lv', 'Latvija'), ] -def test_property_public_access(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_property_public_access(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | public | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp) == [ 'Latvija', 'Lietuva', ] - resp = app.get('/datasets/gov/example/country', headers={'Accept': 'text/html'}) + resp = app.get('/datasets/gov/example/Country', headers={'Accept': 'text/html'}) assert listdata(resp) == [ 'Latvija', 'Lietuva', ] -def test_select_protected_property(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_select_protected_property(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | public | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?select(code,name)') + resp = app.get('/datasets/gov/example/Country?select(code,name)') assert error(resp) == 'PropertyNotFound' - resp = app.get('/datasets/gov/example/country?select(code,name)', headers={'Accept': 'text/html'}) + resp = app.get('/datasets/gov/example/Country?select(code,name)', headers={'Accept': 'text/html'}) assert error(resp) == 'PropertyNotFound' -def test_ns_getall(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_ns_getall(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | prepare | type | ref | level | access | uri | title | description | datasets/gov/example | | | | | | | | Example | | | data | | | sql | | | | | Data | | | | | | | | | | | | | - | | | | country | salis | code!='ee' | | code | | | | Country | + | | | | Country | salis | code!='ee' | | code | | | | Country | | | | | | code | kodas | | string | | 3 | public | | Code | | | | | | name | pavadinimas | | string | | 3 | open | | Name | ''')) @@ -535,17 +535,17 @@ def test_ns_getall(rc, tmp_path, geodb): resp = app.get('/datasets/gov/example') assert listdata(resp, 'name', 'title') == [ - ('datasets/gov/example/country', 'Country'), + ('datasets/gov/example/Country', 'Country'), ] resp = app.get('/datasets/gov/example', headers={'Accept': 'text/html'}) assert listdata(resp, 'name', 'title') == [ - ('📄 country', 'Country'), + ('📄 Country', 'Country'), ] -def test_push(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_push(context, postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | | @@ -626,8 +626,8 @@ def test_push(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, ] -def test_push_dry_run(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_push_dry_run(context, postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example | | | | | data | sql | | | @@ -664,20 +664,20 @@ def test_push_dry_run(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, assert listdata(resp, 'code', 'name') == [] -def test_no_primary_key(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_no_primary_key(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example | | | | | data | | sql | | | | | | | | - | | | country | salis | | | + | | | Country | salis | | | | | | | code | kodas | string | | open | | | | name | pavadinimas | string | | open ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') codes = dict(listdata(resp, '_id', 'code')) data = listdata(resp, '_id', 'code', 'name', sort='code') data = [(codes.get(_id), code, name) for _id, code, name in data] @@ -688,24 +688,25 @@ def test_no_primary_key(rc, tmp_path, geodb): ] -def test_count(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_count(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example | | | | | data | | sql | | | | | | | | - | | | country | salis | | | + | | | Country | salis | | | | | | | code | kodas | string | | open | | | | name | pavadinimas | string | | open ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country?select(count())') + resp = app.get('/datasets/gov/example/Country?select(count())') assert listdata(resp) == [3] def test_push_chunks( + context, postgresql, rc, cli: SpintaCliRunner, @@ -714,12 +715,12 @@ def test_push_chunks( geodb, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example | | | | | data | | sql | | | | | | | | - | | | country | salis | | code | + | | | Country | salis | | code | | | | | code | kodas | string | | open | | | | name | pavadinimas | string | | open ''')) @@ -740,8 +741,8 @@ def test_push_chunks( '--chunk-size=1', ]) - remote.app.authmodel('datasets/gov/example/country', ['getall']) - resp = remote.app.get('/datasets/gov/example/country') + remote.app.authmodel('datasets/gov/example/Country', ['getall']) + resp = remote.app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('ee', 'Estija'), ('lt', 'Lietuva'), @@ -749,13 +750,13 @@ def test_push_chunks( ] -def test_push_state(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_push_state(context, postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example | | | | | data | | sql | | | | | | | | - | | | country | salis | | code | + | | | Country | salis | | code | | | | | code | kodas | string | | open | | | | name | pavadinimas | string | | open ''')) @@ -779,8 +780,8 @@ def test_push_state(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, g '--state', tmp_path / 'state.db', ]) - remote.app.authmodel('datasets/gov/example/country', ['getall']) - resp = remote.app.get('/datasets/gov/example/country') + remote.app.authmodel('datasets/gov/example/Country', ['getall']) + resp = remote.app.get('/datasets/gov/example/Country') assert len(listdata(resp)) == 1 cli.invoke(localrc, [ @@ -792,16 +793,16 @@ def test_push_state(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, g '--state', tmp_path / 'state.db', ]) - resp = remote.app.get('/datasets/gov/example/country') + resp = remote.app.get('/datasets/gov/example/Country') assert len(listdata(resp)) == 2 -def test_prepared_property(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_prepared_property(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | data | sql | | | | - | | | country | | code | salis | | open + | | | Country | | code | salis | | open | | | | code | string | | kodas | | | | | | name | string | | pavadinimas | | | | | | continent | string | | | 'EU' | @@ -809,7 +810,7 @@ def test_prepared_property(rc, tmp_path, geodb): app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'continent', 'code', 'name') == [ ('EU', 'ee', 'Estija'), ('EU', 'lt', 'Lietuva'), @@ -817,8 +818,8 @@ def test_prepared_property(rc, tmp_path, geodb): ] -def test_composite_keys(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_composite_keys(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/ds | | | | | | rs | sql | | | | @@ -897,8 +898,8 @@ def test_composite_keys(rc, tmp_path, sqlite): ] -def test_composite_ref_keys(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_composite_ref_keys(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/ds | | | | | | rs | sql | | | | @@ -988,8 +989,8 @@ def test_composite_ref_keys(rc, tmp_path, sqlite): ] -def test_composite_non_pk_keys(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_composite_non_pk_keys(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/ds | | | | | | rs | sql | | | | @@ -1067,8 +1068,8 @@ def test_composite_non_pk_keys(rc, tmp_path, sqlite): ] -def test_composite_non_pk_keys_with_filter(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_composite_non_pk_keys_with_filter(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/ds | | | | | | rs | sql | | | | @@ -1144,8 +1145,8 @@ def test_composite_non_pk_keys_with_filter(rc, tmp_path, sqlite): ] -def test_access_private_primary_key(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_access_private_primary_key(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | access datasets/ds | | | | | rs | sql | | | @@ -1215,8 +1216,8 @@ def test_access_private_primary_key(rc, tmp_path, sqlite): ] -def test_enum(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1249,8 +1250,8 @@ def test_enum(rc, tmp_path, sqlite): ] -def test_enum_ref(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum_ref(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access | enum | side | l | 'left' | open | | | r | 'right' | open @@ -1284,8 +1285,8 @@ def test_enum_ref(rc, tmp_path, sqlite): ] -def test_enum_no_prepare(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum_no_prepare(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1318,8 +1319,8 @@ def test_enum_no_prepare(rc, tmp_path, sqlite): ] -def test_enum_empty_source(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum_empty_source(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1353,8 +1354,8 @@ def test_enum_empty_source(rc, tmp_path, sqlite): ] -def test_enum_ref_empty_source(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum_ref_empty_source(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access | enum | side | l | | open | | | r | | open @@ -1388,8 +1389,8 @@ def test_enum_ref_empty_source(rc, tmp_path, sqlite): ] -def test_enum_empty_integer_source(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_enum_empty_integer_source(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1422,8 +1423,8 @@ def test_enum_empty_integer_source(rc, tmp_path, sqlite): ] -def test_filter_by_enum_access(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_enum_access(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1455,8 +1456,8 @@ def test_filter_by_enum_access(rc, tmp_path, sqlite): ] -def test_filter_by_ref_enum_access(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_ref_enum_access(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access | enum | side | 0 | 'l' | private | | | 1 | 'r' | open @@ -1488,8 +1489,8 @@ def test_filter_by_ref_enum_access(rc, tmp_path, sqlite): ] -def test_filter_by_enum(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_enum(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1521,8 +1522,8 @@ def test_filter_by_enum(rc, tmp_path, sqlite): ] -def test_filter_by_ref_enum(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_ref_enum(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access | enum | side | 0 | 'l' | private | | | 1 | 'r' | open @@ -1554,8 +1555,8 @@ def test_filter_by_ref_enum(rc, tmp_path, sqlite): ] -def test_filter_by_enum_multi_value(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_enum_multi_value(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1588,8 +1589,8 @@ def test_filter_by_enum_multi_value(rc, tmp_path, sqlite): ] -def test_filter_by_enum_list_value(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_filter_by_enum_list_value(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1622,8 +1623,8 @@ def test_filter_by_enum_list_value(rc, tmp_path, sqlite): ] -def test_implicit_filter(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_implicit_filter(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1670,8 +1671,8 @@ def test_implicit_filter(rc, tmp_path, sqlite): ] -def test_implicit_filter_no_external_source(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_implicit_filter_no_external_source(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1713,8 +1714,8 @@ def test_implicit_filter_no_external_source(rc, tmp_path, sqlite): ] -def test_implicit_filter_two_refs(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_implicit_filter_two_refs(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | type | ref | source | prepare | access example/standards | | | | | | sql | sql | | sqlite:// | | @@ -1773,8 +1774,8 @@ def test_implicit_filter_two_refs(rc, tmp_path, sqlite): ] -def test_implicit_filter_by_enum(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_implicit_filter_by_enum(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1823,8 +1824,8 @@ def test_implicit_filter_by_enum(rc, tmp_path, sqlite): ] -def test_implicit_filter_by_enum_empty_access(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_implicit_filter_by_enum_empty_access(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | enum | Side | 0 | 'l' | @@ -1876,8 +1877,8 @@ def test_implicit_filter_by_enum_empty_access(rc, tmp_path, sqlite): ] -def test_file(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' +def test_file(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -1919,6 +1920,7 @@ def test_file(rc, tmp_path, sqlite): def test_push_file( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -1927,7 +1929,7 @@ def test_push_file( sqlite: Sqlite, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | m | property | type | ref | source | prepare | access datasets/gov/push/file | | | | | | resource | sql | sql | | | @@ -1985,8 +1987,8 @@ def test_push_file( assert resp.content == b'DATA' -def test_image(rc, tmp_path, sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' +def test_image(context, rc, tmp_path, sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | m | property | type | ref | source | prepare | access datasets/gov/example | | | | | | resource | sql | | | | @@ -2028,6 +2030,7 @@ def test_image(rc, tmp_path, sqlite): def test_image_file( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2036,7 +2039,7 @@ def test_image_file( sqlite: Sqlite, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | m | property | type | ref | source | prepare | access datasets/gov/push/file | | | | | | resource | sql | sql | | | @@ -2095,6 +2098,7 @@ def test_image_file( def test_push_null_foreign_key( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2103,7 +2107,7 @@ def test_push_null_foreign_key( sqlite: Sqlite, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | access example/null/fk | | | | | resource | sql | sql | | @@ -2199,6 +2203,7 @@ def test_push_null_foreign_key( def test_push_self_ref( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2207,7 +2212,7 @@ def test_push_self_ref( sqlite: Sqlite, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | access example/self/ref | | | | | resource | sql | sql | | @@ -2267,6 +2272,7 @@ def test_push_self_ref( def _prep_error_handling( + context, tmp_path: pathlib.Path, sqlite: Sqlite, rc: RawConfig, @@ -2275,7 +2281,7 @@ def _prep_error_handling( response: Tuple[int, Dict[str, str], str] = None, exception: Exception = None, ) -> RawConfig: - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | access example/errors | | | | | resource | sql | sql | | @@ -2328,6 +2334,7 @@ def handler(request: PreparedRequest): def test_error_handling_server_error( + context, rc: RawConfig, cli: SpintaCliRunner, responses: RequestsMock, @@ -2335,7 +2342,7 @@ def test_error_handling_server_error( sqlite: Sqlite, caplog: LogCaptureFixture, ): - rc = _prep_error_handling(tmp_path, sqlite, rc, responses, response=( + rc = _prep_error_handling(context, tmp_path, sqlite, rc, responses, response=( 400, {'content-type': 'application/json'}, '{"errors":[{"type": "system", "message": "ERROR"}]}', @@ -2358,6 +2365,7 @@ def test_error_handling_server_error( def test_error_handling_io_error( + context, rc: RawConfig, cli: SpintaCliRunner, responses: RequestsMock, @@ -2366,6 +2374,7 @@ def test_error_handling_io_error( caplog: LogCaptureFixture, ): rc = _prep_error_handling( + context, tmp_path, sqlite, rc, responses, exception=IOError('I/O error.'), ) @@ -2386,8 +2395,8 @@ def test_error_handling_io_error( assert 'Error: I/O error.' in caplog.text -def test_sql_views(rc: RawConfig, tmp_path: pathlib.Path, sqlite: Sqlite): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' +def test_sql_views(context, rc: RawConfig, tmp_path: pathlib.Path, sqlite: Sqlite): + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | access example/views | | | | | resource | sql | sql | | @@ -2420,6 +2429,7 @@ def test_sql_views(rc: RawConfig, tmp_path: pathlib.Path, sqlite: Sqlite): @pytest.mark.skip('TODO') def test_params( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2427,7 +2437,7 @@ def test_params( tmp_path, sqlite: Sqlite, ): - create_tabular_manifest(tmp_path / 'manifest.csv', ''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | ref | source | prepare example/self/ref/param | | | | | resource | sql | sql | | @@ -2468,6 +2478,7 @@ def test_params( def test_cast_string( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2476,7 +2487,7 @@ def test_cast_string( sqlite: Sqlite, ): dataset = 'example/func/cast/string' - create_tabular_manifest(tmp_path / 'manifest.csv', f''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', f''' d | r | b | m | property | type | ref | source | prepare {dataset} | | | | | resource | sql | sql | | @@ -2500,8 +2511,8 @@ def test_cast_string( @pytest.mark.skip('todo') -def test_type_text_push(postgresql, rc, cli: SpintaCliRunner, responses, tmpdir, geodb, request): - create_tabular_manifest(tmpdir / 'manifest.csv', striptable(''' +def test_type_text_push(context, postgresql, rc, cli: SpintaCliRunner, responses, tmpdir, geodb, request): + create_tabular_manifest(context, tmpdir / 'manifest.csv', striptable(''' d | r | b | m | property| type | ref | source | access datasets/gov/example/text_push | | | | | data | sql | | | @@ -2539,6 +2550,7 @@ def test_type_text_push(postgresql, rc, cli: SpintaCliRunner, responses, tmpdir, def test_text_type_push_chunks( + context, postgresql, rc, cli: SpintaCliRunner, @@ -2547,12 +2559,12 @@ def test_text_type_push_chunks( geodb, request, ): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example/text_chunks | | | | | data | | sql | | | | | | | | - | | | country | salis | | code | + | | | Country | salis | | code | | | | | code | kodas | string | | open | | | | name@lt | pavadinimas | string | | open | | | | name@en | pavadinimas | string | | open @@ -2582,8 +2594,8 @@ def test_text_type_push_chunks( '--chunk-size=1', ]) - remote.app.authmodel('datasets/gov/example/text_chunks/country', ['getall']) - resp = remote.app.get('/datasets/gov/example/text_chunks/country') + remote.app.authmodel('datasets/gov/example/text_chunks/Country', ['getall']) + resp = remote.app.get('/datasets/gov/example/text_chunks/Country') assert listdata(resp, 'code', 'name') == [ ('ee', 'Estija'), ('lt', 'Lietuva'), @@ -2591,13 +2603,13 @@ def test_text_type_push_chunks( ] -def test_text_type_push_state(postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_text_type_push_state(context, postgresql, rc, cli: SpintaCliRunner, responses, tmp_path, geodb, request): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | b | m | property | source | type | ref | access datasets/gov/example/text | | | | | data | | sql | | | | | | | | - | | | country | salis | | code | + | | | Country | salis | | code | | | | | code | kodas | string | | open | | | | name@lt | pavadinimas | string | | open ''')) @@ -2621,8 +2633,8 @@ def test_text_type_push_state(postgresql, rc, cli: SpintaCliRunner, responses, t '--state', tmp_path / 'state.db', ]) - remote.app.authmodel('/datasets/gov/example/text/country', ['getall']) - resp = remote.app.get('/datasets/gov/example/text/country') + remote.app.authmodel('/datasets/gov/example/text/Country', ['getall']) + resp = remote.app.get('/datasets/gov/example/text/Country') assert len(listdata(resp)) == 1 cli.invoke(localrc, [ @@ -2634,11 +2646,12 @@ def test_text_type_push_state(postgresql, rc, cli: SpintaCliRunner, responses, t '--state', tmp_path / 'state.db', ]) - resp = remote.app.get('/datasets/gov/example/text/country') + resp = remote.app.get('/datasets/gov/example/text/Country') assert len(listdata(resp)) == 2 def test_cast_integer( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2647,7 +2660,7 @@ def test_cast_integer( sqlite: Sqlite, ): dataset = 'example/func/cast/integer' - create_tabular_manifest(tmp_path / 'manifest.csv', f''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', f''' d | r | b | m | property | type | ref | source | prepare {dataset} | | | | | resource | sql | sql | | @@ -2671,6 +2684,7 @@ def test_cast_integer( def test_cast_integer_error( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2679,7 +2693,7 @@ def test_cast_integer_error( sqlite: Sqlite, ): dataset = 'example/func/cast/integer/error' - create_tabular_manifest(tmp_path / 'manifest.csv', f''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', f''' d | r | b | m | property | type | ref | source | prepare {dataset} | | | | | resource | sql | sql | | @@ -2703,6 +2717,7 @@ def test_cast_integer_error( def test_point( + context, postgresql, rc: RawConfig, cli: SpintaCliRunner, @@ -2711,7 +2726,7 @@ def test_point( sqlite: Sqlite, ): dataset = 'example/func/point' - create_tabular_manifest(tmp_path / 'manifest.csv', f''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', f''' d | r | b | m | property | type | ref | source | prepare | access {dataset} | | | | | | resource | sql | sql | | | @@ -2743,20 +2758,20 @@ def test_point( assert listdata(resp) == [(1, 'POINT (4.5 2.5)')] -def test_swap_single(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_swap_single(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | datasets/gov/example | | | | | | | data | | sql | | | | | | | | | | | - | | | | country | salis | | code | open | + | | | | Country | salis | | code | open | | | | | | code | kodas | string | | | swap('lt', 'LT') | | | | | name | pavadinimas | string | | | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('LT', 'Lietuva'), ('ee', 'Estija'), @@ -2764,20 +2779,20 @@ def test_swap_single(rc, tmp_path, geodb): ] -def test_swap_multi_with_dot(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_swap_multi_with_dot(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | datasets/gov/example | | | | | | | data | | sql | | | | | | | | | | | - | | | | country | salis | | code | open | + | | | | Country | salis | | code | open | | | | | | code | kodas | string | | | swap('lt', 'LT').swap('lv', 'LV') | | | | | name | pavadinimas | string | | | ''')) app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('LT', 'Lietuva'), ('LV', 'Latvija'), @@ -2785,13 +2800,13 @@ def test_swap_multi_with_dot(rc, tmp_path, geodb): ] -def test_swap_multi_with_multi_lines(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_swap_multi_with_multi_lines(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | datasets/gov/example | | | | | | | data | | sql | | | | | | | | | | | - | | | | country | salis | | code | open | + | | | | Country | salis | | code | open | | | | | | code | kodas | string | | | swap('lt', 'LT') | | | | | | lv | | | | swap('LV') | | | | | name | pavadinimas | string | | | @@ -2799,7 +2814,7 @@ def test_swap_multi_with_multi_lines(rc, tmp_path, geodb): app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('LT', 'Lietuva'), ('LV', 'Latvija'), @@ -2807,13 +2822,13 @@ def test_swap_multi_with_multi_lines(rc, tmp_path, geodb): ] -def test_swap_multi_with_multi_lines_all_to_same(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_swap_multi_with_multi_lines_all_to_same(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | datasets/gov/example | | | | | | | data | | sql | | | | | | | | | | | - | | | | country | salis | | code | open | + | | | | Country | salis | | code | open | | | | | | code | kodas | string | | | swap('lt', 'CODE') | | | | | | lv | | | | swap('CODE') | | | | | | | | | | swap('ee', 'CODE') @@ -2822,7 +2837,7 @@ def test_swap_multi_with_multi_lines_all_to_same(rc, tmp_path, geodb): app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/country') + resp = app.get('/datasets/gov/example/Country') assert listdata(resp, 'code', 'name') == [ ('CODE', 'Estija'), ('CODE', 'Latvija'), @@ -2830,13 +2845,13 @@ def test_swap_multi_with_multi_lines_all_to_same(rc, tmp_path, geodb): ] -def test_swap_multi_escape_source(rc, tmp_path, geodb): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_swap_multi_escape_source(context, rc, tmp_path, geodb): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' id | d | r | b | m | property | source | type | ref | access | prepare | datasets/gov/example | | | | | | | data | | sql | | | | | | | | | | | - | | | | test | test | | id | open | + | | | | Test | test | | id | open | | | | | | id | id | integer | | | | | | | | text | text | string | | | swap("\\"TEST\\"", "NORMAL SWAPPED PREPARE") | | | | | | 'TEST' | | | | swap("TESTAS") @@ -2848,7 +2863,7 @@ def test_swap_multi_escape_source(rc, tmp_path, geodb): app = create_client(rc, tmp_path, geodb) - resp = app.get('/datasets/gov/example/test') + resp = app.get('/datasets/gov/example/Test') assert listdata(resp, 'old', 'text') == [ ("'TEST'", 'TESTAS'), ("test 'TEST'", "TEST 'test'"), @@ -2858,8 +2873,8 @@ def test_swap_multi_escape_source(rc, tmp_path, geodb): ] -def test_advanced_denorm(rc, tmp_path, geodb_denorm): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_advanced_denorm(context, rc, tmp_path, geodb_denorm): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access datasets/denorm | | | | | | rs | sql | | | | @@ -2951,8 +2966,8 @@ def test_advanced_denorm(rc, tmp_path, geodb_denorm): }] -def test_advanced_denorm_lvl_3(rc, tmp_path, geodb_denorm): - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' +def test_advanced_denorm_lvl_3(context, rc, tmp_path, geodb_denorm): + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' d | r | m | property | type | ref | source | prepare | access | level datasets/denorm/lvl3 | | | | | | | rs | sql | | | | | diff --git a/tests/dtypes/test_array.py b/tests/dtypes/test_array.py index f8331908f..d8b364cb5 100644 --- a/tests/dtypes/test_array.py +++ b/tests/dtypes/test_array.py @@ -1,10 +1,13 @@ from spinta.testing.client import create_test_client from spinta.testing.manifest import bootstrap_manifest from _pytest.fixtures import FixtureRequest +import pytest -def test_getall_level4(rc, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_getall_level4(manifest_type, rc, tmp_path, postgresql: str, request: FixtureRequest): + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/lvl4 | | | | | | | Country | | | | @@ -12,7 +15,13 @@ def test_getall_level4(rc, postgresql: str, request: FixtureRequest): | | | | cities[] | ref | City | open | 4 | | | City | | name | | | | | | name | string | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields', 'spinta_getone']) app.authmodel('example/lvl4/City', ['insert']) @@ -54,8 +63,11 @@ def test_getall_level4(rc, postgresql: str, request: FixtureRequest): ] -def test_getall_level3(rc, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_getall_level3(manifest_type, rc, tmp_path, postgresql: str, request: FixtureRequest): + context = bootstrap_manifest( + rc, + ''' d | r | b | m | property | type | ref | access | level example/lvl3 | | | | | | | Country | | | | @@ -63,7 +75,13 @@ def test_getall_level3(rc, postgresql: str, request: FixtureRequest): | | | | cities[] | ref | City | open | 3 | | | City | | name | | | | | | name | string | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields', 'spinta_getone']) app.authmodel('example/lvl3/City', ['insert']) @@ -105,14 +123,23 @@ def test_getall_level3(rc, postgresql: str, request: FixtureRequest): ] -def test_getall_simple_type(rc, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_getall_simple_type(manifest_type, rc, tmp_path, postgresql: str, request: FixtureRequest): + context = bootstrap_manifest( + rc, + ''' d | r | b | m | property | type | ref | access | level example/simple | | | | | | | Country | | | | | | | | name | string | | open | | | | | cities[] | string | | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields', 'spinta_getone']) @@ -134,8 +161,11 @@ def test_getall_simple_type(rc, postgresql: str, request: FixtureRequest): ] -def test_array_shortcut_inherit_access_open(rc, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_array_shortcut_inherit_access_open(manifest_type, rc, tmp_path, postgresql: str, request: FixtureRequest): + context = bootstrap_manifest( + rc, + ''' d | r | b | m | property | type | ref | access example/dtypes/array/open | | | | | | @@ -145,7 +175,13 @@ def test_array_shortcut_inherit_access_open(rc, postgresql: str, request: Fixtur | | | Country | | | | | | | name | string | | open | | | | languages[] | ref | Language | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) app.authmodel('example/dtypes/array/open', ['insert', 'getone', 'getall']) @@ -179,8 +215,11 @@ def test_array_shortcut_inherit_access_open(rc, postgresql: str, request: Fixtur ] -def test_array_shortcut_inherit_access_private(rc, postgresql: str, request: FixtureRequest): - context = bootstrap_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_array_shortcut_inherit_access_private(manifest_type, rc, tmp_path, postgresql: str, request: FixtureRequest): + context = bootstrap_manifest( + rc, + ''' d | r | b | m | property | type | ref | access example/dtypes/array/private | | | | | | @@ -190,7 +229,13 @@ def test_array_shortcut_inherit_access_private(rc, postgresql: str, request: Fix | | | Country | | | | | | | name | string | | open | | | | languages[] | ref | Language | private - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) app.authmodel('example/dtypes/array/private', ['insert', 'getone', 'getall']) diff --git a/tests/dtypes/test_backref.py b/tests/dtypes/test_backref.py index 74f22da63..31ba1f7ec 100644 --- a/tests/dtypes/test_backref.py +++ b/tests/dtypes/test_backref.py @@ -1,3 +1,5 @@ +import pathlib + import pytest from pytest import FixtureRequest @@ -11,12 +13,17 @@ from spinta.testing.utils import error +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_one_to_one_level_4( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, + ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/oto/level4 | | | | | | | | @@ -29,7 +36,13 @@ def test_backref_one_to_one_level_4( | | | | id | integer | | open | | | | | name | string | | open | | | | | leader | ref | Leader | open | 4 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -79,12 +92,16 @@ def test_backref_one_to_one_level_4( } +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_one_to_one_level_3( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/oto/level3 | | | | | | | | @@ -97,7 +114,13 @@ def test_backref_one_to_one_level_3( | | | | id | integer | | open | | | | | name | string | | open | | | | | leader | ref | Leader | open | 3 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -152,12 +175,16 @@ def test_backref_one_to_one_level_3( } +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_many_to_one_level_4( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/mto/level4 | | | | | | | | @@ -170,7 +197,13 @@ def test_backref_many_to_one_level_4( | | | | id | integer | | open | | | | | name | string | | open | | | | | language | ref | Language | open | 4 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -241,12 +274,16 @@ def test_backref_many_to_one_level_4( assert result_json[2]['country'] == [] +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_many_to_one_level_3( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/mto/level3 | | | | | | | | @@ -259,7 +296,13 @@ def test_backref_many_to_one_level_3( | | | | id | integer | | open | | | | | name | string | | open | | | | | language | ref | Language | open | 3 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -336,12 +379,16 @@ def test_backref_many_to_one_level_3( assert result_json[2]['country'] == [] +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_many_to_many_level_4( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/mtm/level4 | | | | | | | | @@ -354,7 +401,13 @@ def test_backref_many_to_many_level_4( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | 4 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -438,12 +491,16 @@ def test_backref_many_to_many_level_4( assert result_json[3]['country'] == [] +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_many_to_many_level_3( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/mtm/level3 | | | | | | | | @@ -456,7 +513,13 @@ def test_backref_many_to_many_level_3( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | 3 - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -548,12 +611,16 @@ def test_backref_many_to_many_level_3( assert result_json[3]['country'] == [] +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_x_to_many_expand( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/xtm/expand | | | | | | | | @@ -566,7 +633,13 @@ def test_backref_x_to_many_expand( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -650,12 +723,16 @@ def test_backref_x_to_many_expand( ]) +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_modify_backref_insert( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/modify/insert | | | | | | | | @@ -668,7 +745,13 @@ def test_backref_error_modify_backref_insert( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -687,12 +770,16 @@ def test_backref_error_modify_backref_insert( } +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_modify_backref_put( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/modify/put | | | | | | | | @@ -705,7 +792,13 @@ def test_backref_error_modify_backref_put( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -731,12 +824,16 @@ def test_backref_error_modify_backref_put( } +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_modify_backref_patch( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/modify/patch | | | | | | | | @@ -749,7 +846,13 @@ def test_backref_error_modify_backref_patch( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -775,12 +878,16 @@ def test_backref_error_modify_backref_patch( } +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_multiple_same( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/multiple/same | | | | | | | | @@ -795,7 +902,13 @@ def test_backref_multiple_same( | | | | name | string | | open | | | | | language_primary | ref | Language | open | | | | | language_secondary | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -874,12 +987,16 @@ def test_backref_multiple_same( ]) +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_multiple_all_types( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/multiple/all | | | | | | | | @@ -896,7 +1013,13 @@ def test_backref_multiple_all_types( | | | | language_0 | ref | Language | open | | | | | language_1 | ref | Language | open | | | | | language_array[] | ref | Language | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) @@ -1077,12 +1200,16 @@ def test_backref_multiple_all_types( assert result_json[3]['country_array'] == [] +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_no_ref( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, postgresql: str, ): with pytest.raises(NoBackRefReferencesFound): - bootstrap_manifest(rc, ''' + bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/no_ref | | | | | | | | @@ -1094,15 +1221,24 @@ def test_backref_error_no_ref( | | | Country | | id | | | | | | id | integer | | open | | | | | name | string | | open | - ''', backend=postgresql) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_invalid_ref( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, - postgresql: str + postgresql: str, ): with pytest.raises(NoReferencesFound): - bootstrap_manifest(rc, ''' + bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/no_ref | | | | | | | | @@ -1115,15 +1251,24 @@ def test_backref_error_invalid_ref( | | | | id | integer | | open | | | | | name | string | | open | | | | | language | ref | Language | open | - ''', backend=postgresql) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_multiple_ref( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, - postgresql: str + postgresql: str, ): with pytest.raises(MultipleBackRefReferencesFound): - bootstrap_manifest(rc, ''' + bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/error/no_ref | | | | | | | | @@ -1137,15 +1282,24 @@ def test_backref_error_multiple_ref( | | | | name | string | | open | | | | | language | ref | Language | open | | | | | language0 | ref | Language | open | - ''', backend=postgresql) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) +@pytest.mark.manifests('internal_sql', 'csv') def test_backref_error_one_to_many( + manifest_type: str, + tmp_path: pathlib.Path, rc: RawConfig, - postgresql: str + postgresql: str, ): with pytest.raises(OneToManyBackRefNotSupported): - bootstrap_manifest(rc, ''' + bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | level example/dtypes/backref/otm/level4 | | | | | | | | @@ -1158,4 +1312,9 @@ def test_backref_error_one_to_many( | | | | id | integer | | open | | | | | name | string | | open | | | | | language[] | ref | Language | open | 4 - ''', backend=postgresql) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) diff --git a/tests/dtypes/test_binary.py b/tests/dtypes/test_binary.py index b40a7793a..2a7283bfe 100644 --- a/tests/dtypes/test_binary.py +++ b/tests/dtypes/test_binary.py @@ -4,7 +4,7 @@ @pytest.mark.models( - 'backends/postgres/dtypes/binary', + 'backends/postgres/dtypes/Binary', ) def test_insert(model, app): data = base64.b64encode(b'data').decode('ascii') @@ -15,7 +15,7 @@ def test_insert(model, app): @pytest.mark.models( - 'backends/postgres/dtypes/binary', + 'backends/postgres/dtypes/Binary', ) def test_upsert(model, app): data = base64.b64encode(b'data').decode('ascii') @@ -47,7 +47,7 @@ def test_upsert(model, app): @pytest.mark.models( - 'datasets/dtypes/binary', + 'datasets/dtypes/Binary', ) def test_getone(model, app): data = base64.b64encode(b'data').decode('ascii') @@ -63,7 +63,7 @@ def test_getone(model, app): @pytest.mark.models( - 'datasets/dtypes/binary', + 'datasets/dtypes/Binary', ) def test_getall(model, app): data = base64.b64encode(b'data').decode('ascii') diff --git a/tests/dtypes/test_external_ref.py b/tests/dtypes/test_external_ref.py index 3524a45a9..7edc9461c 100644 --- a/tests/dtypes/test_external_ref.py +++ b/tests/dtypes/test_external_ref.py @@ -9,34 +9,39 @@ from spinta.testing.manifest import load_manifest from spinta.testing.tabular import create_tabular_manifest from spinta.testing.utils import get_error_codes +import pytest -def test_load(tmp_path: Path, rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_load(manifest_type, tmp_path: Path, rc: RawConfig): table = ''' - d | r | b | m | property | type | ref | source | level | access - dataset/1 | | | | | - | external | sql | | sqlite:// | | - | | | | | - | | | Country | | code | | | - | | | | code | string | | | | open - | | | | name | string | | | | open - dataset/2 | | | | | - | | | | | - | | | City | | | | | - | | | | name | string | | | | open - | | | | country | ref | /dataset/1/Country | | 3 | open + d | r | b | m | property | type | ref | source | level | access + dataset/one | | | | | + | external | sql | | sqlite:// | | + | | | | | + | | | Country | | code | | | + | | | | code | string | | | | open + | | | | name | string | | | | open + dataset/two | | | | | + | | | | | + | | | City | | | | | + | | | | name | string | | | | open + | | | | country | ref | /dataset/one/Country | | 3 | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', table) - manifest = load_manifest(rc, tmp_path / 'manifest.csv') + manifest = load_manifest(rc, table, manifest_type=manifest_type, tmp_path=tmp_path) assert manifest == table +@pytest.mark.manifests('internal_sql', 'csv') def test_external_ref( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | source | level | access datasets/externalref | | | | | | external | sql | | sqlite:// | | @@ -47,7 +52,13 @@ def test_external_ref( | | | City | | | | | | | | | name | string | | | | open | | | | country | ref | /datasets/externalref/Country | | 3 | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('datasets/internal/City', [ @@ -78,12 +89,16 @@ def test_external_ref( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_external_ref_without_primary_key( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | source | level | access datasets/externalref | | | | | | external | sql | | sqlite:// | | @@ -94,7 +109,13 @@ def test_external_ref_without_primary_key( | | | City | | | | | | | | | name | string | | | | open | | | | country | ref | /datasets/externalref/Country | | 3 | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('datasets/internal/pk/City', [ @@ -126,12 +147,16 @@ def test_external_ref_without_primary_key( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_external_ref_with_explicit_key( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | source | level | access datasets/external/ref | | | | | | external | sql | | sqlite:// | | @@ -142,7 +167,13 @@ def test_external_ref_with_explicit_key( | | | City | | | | | | | | | name | string | | | | open | | | | country | ref | /datasets/external/ref/Country[id]| | 3 | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('datasets/explicit/ref/City', [ diff --git a/tests/dtypes/test_file.py b/tests/dtypes/test_file.py index c2b42703f..3ecc6e9d8 100644 --- a/tests/dtypes/test_file.py +++ b/tests/dtypes/test_file.py @@ -27,7 +27,7 @@ def _create_file( @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_insert(model, app): app.authmodel(model, ['insert', 'getone']) @@ -57,7 +57,7 @@ def test_insert(model, app): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_update(model, app, tmp_path): app.authmodel(model, ['insert', 'update', 'getone']) @@ -78,7 +78,7 @@ def test_update(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file', + '_type': 'backends/postgres/dtypes/File', '_id': data['_id'], '_revision': data['_revision'], 'file': { @@ -96,7 +96,7 @@ def test_update(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_patch(model, app, tmp_path): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -116,7 +116,7 @@ def test_patch(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file', + '_type': 'backends/postgres/dtypes/File', '_id': data['_id'], '_revision': data['_revision'], 'file': { @@ -133,7 +133,7 @@ def test_patch(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_update(model, app, tmp_path): app.authmodel(model, ['insert', 'update', 'getone']) @@ -151,7 +151,7 @@ def test_subresource_update(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file.file', + '_type': 'backends/postgres/dtypes/File.file', '_revision': data['_revision'], '_id': 'data.txt', '_content_type': 'text/plain', @@ -166,7 +166,7 @@ def test_subresource_update(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_patch(model, app, tmp_path): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -184,7 +184,7 @@ def test_subresource_patch(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file.file', + '_type': 'backends/postgres/dtypes/File.file', '_revision': data['_revision'], '_id': 'data_new.txt', '_content_type': 'application/pdf', @@ -200,7 +200,7 @@ def test_subresource_patch(model, app, tmp_path): # update ref of file by PUT /model/id/file:ref @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_update_ref(model, app, tmp_path): app.authmodel(model, ['insert', 'update', 'getone']) @@ -218,7 +218,7 @@ def test_subresource_update_ref(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file.file', + '_type': 'backends/postgres/dtypes/File.file', '_revision': data['_revision'], '_id': 'data.rst', '_content_type': 'text/x-rst', @@ -234,7 +234,7 @@ def test_subresource_update_ref(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_patch_ref(model, app, tmp_path): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -251,7 +251,7 @@ def test_subresource_patch_ref(model, app, tmp_path): data = resp.json() assert resp.status_code == 200, data assert data == { - '_type': 'backends/postgres/dtypes/file.file', + '_type': 'backends/postgres/dtypes/File.file', '_revision': data['_revision'], '_id': 'data.rst', } @@ -266,7 +266,7 @@ def test_subresource_patch_ref(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_get_ref(model, app, tmp_path): app.authmodel(model, ['insert', 'getone']) @@ -288,7 +288,7 @@ def test_subresource_get_ref(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_subresource_delete(model, app, tmp_path): app.authmodel(model, ['insert', 'delete', 'getone']) @@ -325,7 +325,7 @@ def test_subresource_delete(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_select(model, app, tmp_path): app.authmodel(model, ['insert', 'search']) @@ -345,7 +345,7 @@ def test_select(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_select_content(model, app, tmp_path): app.authmodel(model, ['insert', 'search']) @@ -366,7 +366,7 @@ def test_select_content(model, app, tmp_path): @pytest.mark.models( # TODO 'backends/mongo/dtypes/postgres/file', - 'backends/postgres/dtypes/file', + 'backends/postgres/dtypes/File', ) def test_select_all(model, app, tmp_path): app.authmodel(model, ['insert', 'search']) @@ -386,8 +386,8 @@ def test_select_all(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/dtypes/fs/file', - 'backends/postgres/dtypes/fs/file', + 'backends/mongo/dtypes/fs/File', + 'backends/postgres/dtypes/fs/File', ) def test_insert_fs_file(model, app): app.authmodel(model, ['insert', 'getone']) @@ -416,8 +416,8 @@ def test_insert_fs_file(model, app): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '/tmp/etc/passwd', @@ -436,8 +436,8 @@ def test_insert_fs_file_with_injection(model, filename, app, tmp_path): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) def test_upload_fs_file(model, app): app.authmodel(model, ['insert', 'update', 'getone']) @@ -475,8 +475,8 @@ def test_upload_fs_file(model, app): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '/tmp/etc/passwd', @@ -509,8 +509,8 @@ def test_upload_fs_file_with_path_injection(model, app, filename, tmp_path): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) def test_patch_fs_file(model, app): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -548,8 +548,8 @@ def test_patch_fs_file(model, app): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '/tmp/etc/passwd', @@ -582,8 +582,8 @@ def test_patch_fs_file_with_path_injection(model, app, filename, tmp_path): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '../../passwd', @@ -613,8 +613,8 @@ def test_path_injection_put(model, filename, app, tmp_path): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '../../passwd', @@ -644,8 +644,8 @@ def test_path_injection_patch(model, filename, app, tmp_path): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '../../passwd', @@ -671,8 +671,8 @@ def test_path_injection_update_file_ref(model, filename, app): @pytest.mark.models( - 'backends/postgres/dtypes/fs/file', - 'backends/mongo/dtypes/fs/file', + 'backends/postgres/dtypes/fs/File', + 'backends/mongo/dtypes/fs/File', ) @pytest.mark.parametrize('filename', [ '../../passwd', diff --git a/tests/dtypes/test_geometry.py b/tests/dtypes/test_geometry.py index c44275aa0..2f9e4d66f 100644 --- a/tests/dtypes/test_geometry.py +++ b/tests/dtypes/test_geometry.py @@ -21,24 +21,33 @@ from spinta.testing.data import listdata from spinta.testing.manifest import bootstrap_manifest, load_manifest_and_context from spinta.testing.request import render_data -from spinta.testing.tabular import create_tabular_manifest from spinta.testing.manifest import load_manifest from spinta.testing.manifest import load_manifest_get_context from spinta.types.geometry.constants import WGS84, LKS94 +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref backends/postgres/dtypes/geometry | | | | | City | | | | | | name | string | | | | | coordinates | geometry | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('backends/postgres/dtypes/geometry/City', [ @@ -63,18 +72,28 @@ def test_geometry( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_params_point( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref backends/postgres/dtypes/geometry/point | | | | | City | | | | | | name | string | | | | | coordinates | geometry(point) | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/point' model: str = f'{ns}/City' @@ -108,18 +127,28 @@ def test_geometry_params_point( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_params_srid( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref backends/postgres/dtypes/geometry/srid | | | | | City | | | | | | name | string | | | | | coordinates | geometry(3346) | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/srid' model: str = f'{ns}/City' @@ -143,14 +172,20 @@ def test_geometry_params_srid( assert resp.status_code == 201 -def test_geometry_html(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_geometry_html( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | description example | | | | | | City | | | | | | | name | string | | | | | | coordinates | geometry(4326) | | WGS - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) result = render_data( context, manifest, 'example/City', @@ -189,6 +224,7 @@ def test_geometry_html(rc: RawConfig): ) +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.parametrize('wkt, srid, link', [ ('POINT (6061789 582964)', LKS94, osm_url), ('POINT (54.68569111173754 25.286688302053335)', WGS84, osm_url), @@ -196,7 +232,10 @@ def test_geometry_html(rc: RawConfig): ('POINT (25.273658402751387 54.662851967609136)', None, None), ]) def test_geometry_coordinate_transformation( + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, wkt: str, srid: Optional[int], link: Optional[str], @@ -212,9 +251,9 @@ def test_geometry_coordinate_transformation( | | | City | | | | | | | name | string | | | | | | coordinates | {dtype} | | - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) - model = manifest.models['example/City'] + model = commands.get_model(context, manifest, 'example/City') prop = model.properties['coordinates'] value = shapely.wkt.loads(wkt) @@ -233,13 +272,17 @@ def test_geometry_coordinate_transformation( assert result.link == link +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.parametrize('wkt, display', [ ('POINT (25.282 54.681)', 'POINT (25.282 54.681)'), ('POLYGON ((25.28 54.68, 25.29 54.69, 25.38 54.64, 25.28 54.68))', 'POLYGON'), ('LINESTRING (25.28 54.68, 25.29 54.69)', 'LINESTRING'), ]) def test_geometry_wkt_value_shortening( + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, wkt: str, display: str, ): @@ -249,8 +292,8 @@ def test_geometry_wkt_value_shortening( | | | City | | | | | | | name | string | | | | | | coordinates | geometry(4326) | | WGS - ''') - model = manifest.models['example/City'] + ''', manifest_type=manifest_type, tmp_path=tmp_path) + model = commands.get_model(context, manifest, 'example/City') prop = model.properties['coordinates'] value = shapely.wkt.loads(wkt) @@ -268,7 +311,13 @@ def test_geometry_wkt_value_shortening( assert result.value == display -def test_loading(tmp_path: Path, rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_loading( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + postgresql: str, +): table = ''' d | r | b | m | property | type | ref | access datasets/gov/example | | | open @@ -277,17 +326,20 @@ def test_loading(tmp_path: Path, rc: RawConfig): | | | | name | string | | open | | | | country | geometry(point, 3346) | | open ''' - create_tabular_manifest(tmp_path / 'manifest.csv', table) - manifest = load_manifest(rc, tmp_path / 'manifest.csv') + manifest = load_manifest(rc, table, manifest_type=manifest_type, tmp_path=tmp_path) assert manifest == table +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_params_with_srid_without_srid( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | access backends/postgres/dtypes/geometry/srid | | | | | Point | | @@ -296,7 +348,13 @@ def test_geometry_params_with_srid_without_srid( | | | | point | geometry(3346) | open | | | PointWGS84 | | | | | | point | geometry(4326) | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/srid' model: str = f'{ns}/PointLKS94' @@ -335,6 +393,7 @@ def test_geometry_params_with_srid_without_srid( assert resp.status_code == 201 +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.parametrize('path', [ # LKS94 (3346) -> WGS84 (4326) Bell tower of Vilnius Cathedral '3346/6061789/582964', @@ -344,14 +403,15 @@ def test_geometry_params_with_srid_without_srid( '4326/54.68569/25.28668', ]) def test_srid_service( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, - request: FixtureRequest, path: str, ): context = load_manifest_get_context(rc, ''' d | r | b | m | property | type | ref - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) app = create_test_client(context) @@ -369,18 +429,28 @@ def test_srid_service( assert y[:8] == '25.28668' +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_delete( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | access backends/postgres/dtypes/geometry/error | | | | | Point | | | | | | point | geometry | open | | | | number | integer | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/error' model: str = f'{ns}/Point' @@ -400,18 +470,28 @@ def test_geometry_delete( assert resp.status_code == 204 +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_insert_without_geometry( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | access backends/postgres/dtypes/geometry/error | | | | | Point | | | | | | point | geometry | open | | | | number | integer | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/error' model: str = f'{ns}/Point' @@ -427,18 +507,28 @@ def test_geometry_insert_without_geometry( assert resp.status_code == 201 +@pytest.mark.manifests('internal_sql', 'csv') def test_geometry_update_without_geometry( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | access backends/postgres/dtypes/geometry/error | | | | | Point | | | | | | point | geometry | open | | | | number | integer | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) ns: str = 'backends/postgres/dtypes/geometry/error' model: str = f'{ns}/Point' diff --git a/tests/dtypes/test_integer.py b/tests/dtypes/test_integer.py index 9e586d7a8..83f5613d0 100644 --- a/tests/dtypes/test_integer.py +++ b/tests/dtypes/test_integer.py @@ -1,3 +1,4 @@ +from pathlib import Path from typing import Optional import pytest @@ -11,6 +12,7 @@ from spinta.manifests.components import Manifest +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.parametrize('value', [ None, 0, @@ -19,17 +21,28 @@ 1000, -1000, ]) -def test_integer(rc: RawConfig, value: Optional[int]): - context = bootstrap_manifest(rc, ''' +def test_integer( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + value: Optional[int]): + context = bootstrap_manifest( + rc, ''' d | m | property | type datasets/gov/example | | City | | | population | integer - ''', backend='memory') + ''', + backend='memory', + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) + store: Store = context.get('store') manifest: Manifest = store.manifest backend: Memory = manifest.backend - model = manifest.models['datasets/gov/example/City'] + model = commands.get_model(context, manifest, 'datasets/gov/example/City') payload = { '_op': 'insert', 'population': value, diff --git a/tests/dtypes/test_object.py b/tests/dtypes/test_object.py index 7ba82870e..2b36f2f81 100644 --- a/tests/dtypes/test_object.py +++ b/tests/dtypes/test_object.py @@ -5,10 +5,10 @@ @pytest.mark.models( - 'backends/mongo/dtypes/object', - 'backends/postgres/dtypes/object', - 'backends/mongo/dtypes/array/object', - 'backends/postgres/dtypes/array/object', + 'backends/mongo/dtypes/Object', + 'backends/postgres/dtypes/Object', + 'backends/mongo/dtypes/array/Object', + 'backends/postgres/dtypes/array/Object', ) def test_update_empty(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'search']) @@ -23,8 +23,8 @@ def test_update_empty(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/object', - 'backends/postgres/dtypes/object', + 'backends/mongo/dtypes/Object', + 'backends/postgres/dtypes/Object', ) def test_patch_empty(model, app): app.authmodel(model, ['insert', 'patch', 'getone', 'search']) @@ -39,8 +39,8 @@ def test_patch_empty(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/array/object', - 'backends/postgres/dtypes/array/object', + 'backends/mongo/dtypes/array/Object', + 'backends/postgres/dtypes/array/Object', ) def test_patch_empty_array(model, app): app.authmodel(model, ['insert', 'patch', 'getone', 'search']) diff --git a/tests/dtypes/test_required.py b/tests/dtypes/test_required.py index d3ffc9e64..d321228de 100644 --- a/tests/dtypes/test_required.py +++ b/tests/dtypes/test_required.py @@ -1,4 +1,5 @@ import uuid +from pathlib import Path from pytest import FixtureRequest @@ -7,21 +8,32 @@ from spinta.testing.data import listdata from spinta.testing.manifest import bootstrap_manifest from spinta.testing.utils import get_error_codes, get_error_context +import pytest +@pytest.mark.manifests('internal_sql', 'csv') def test_insert_with_required_property( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): dataset = 'type/required/insert' - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref {dataset} | | | | | City | | | | | | name | string required | | | | | description | string | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel(f'{dataset}/City', [ @@ -56,19 +68,29 @@ def test_insert_with_required_property( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_update_with_required_property( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): dataset = 'type/required/update' - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref {dataset} | | | | | City | | | | | | name | string required | | | | | description | string | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel(f'{dataset}/City', [ @@ -112,19 +134,29 @@ def test_update_with_required_property( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_patch_with_required_property( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): dataset = 'type/required/patch' - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref {dataset} | | | | | City | | | | | | name | string required | | | | | description | string | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel(f'{dataset}/City', [ @@ -169,19 +201,29 @@ def test_patch_with_required_property( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_upsert_insert_with_required_property( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): dataset = 'type/required/upsert' - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref {dataset} | | | | | City | | | | | | name | string required | | | | | description | string | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel(f'{dataset}/City', [ @@ -221,19 +263,29 @@ def test_upsert_insert_with_required_property( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_upsert_patch_with_required_property( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): dataset = 'type/required/upsert/patch' - context = bootstrap_manifest(rc, f''' + context = bootstrap_manifest( + rc, f''' d | r | b | m | property | type | ref {dataset} | | | | | City | | | | | | name | string required | | | | | description | string | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel(f'{dataset}/City', [ diff --git a/tests/dtypes/test_string.py b/tests/dtypes/test_string.py index e129428e4..af2d8eba3 100644 --- a/tests/dtypes/test_string.py +++ b/tests/dtypes/test_string.py @@ -5,12 +5,12 @@ @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_insert(model, app): app.authmodel(model, ['insert', 'getone', 'search']) @@ -21,12 +21,12 @@ def test_insert(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_upsert_insert(model, app): app.authmodel(model, ['upsert', 'getone', 'search']) @@ -38,12 +38,12 @@ def test_upsert_insert(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_upsert_patch(model, app): app.authmodel(model, ['insert', 'upsert', 'getone', 'search']) @@ -58,12 +58,12 @@ def test_upsert_patch(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_update(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'search']) @@ -77,10 +77,10 @@ def test_update(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', ) def test_update_missing(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'search']) @@ -94,8 +94,8 @@ def test_update_missing(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_update_missing_in_array(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'search']) @@ -109,12 +109,12 @@ def test_update_missing_in_array(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_update_same(model, app): app.authmodel(model, ['insert', 'update', 'getone', 'search']) @@ -127,12 +127,12 @@ def test_update_same(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_patch(model, app): app.authmodel(model, ['insert', 'patch', 'getone', 'search']) @@ -146,12 +146,12 @@ def test_patch(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_patch_missing(model, app): app.authmodel(model, ['insert', 'patch', 'getone', 'search']) @@ -164,12 +164,12 @@ def test_patch_missing(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', - 'backends/mongo/dtypes/object/string', - 'backends/postgres/dtypes/object/string', - 'backends/mongo/dtypes/array/string', - 'backends/postgres/dtypes/array/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', + 'backends/mongo/dtypes/object/String', + 'backends/postgres/dtypes/object/String', + 'backends/mongo/dtypes/array/String', + 'backends/postgres/dtypes/array/String', ) def test_patch_same(model, app): app.authmodel(model, ['insert', 'patch', 'getone', 'search']) @@ -182,8 +182,8 @@ def test_patch_same(model, app): @pytest.mark.models( - 'backends/mongo/dtypes/string', - 'backends/postgres/dtypes/string', + 'backends/mongo/dtypes/String', + 'backends/postgres/dtypes/String', ) def test_delete(model, app): app.authmodel(model, ['insert', 'delete', 'getone', 'search']) diff --git a/tests/dtypes/test_text.py b/tests/dtypes/test_text.py index 89b9f5a86..cd810c8c9 100644 --- a/tests/dtypes/test_text.py +++ b/tests/dtypes/test_text.py @@ -1,3 +1,5 @@ +from pathlib import Path + from pytest import FixtureRequest import pytest from spinta.core.config import RawConfig @@ -12,18 +14,28 @@ from spinta.testing.utils import error +@pytest.mark.manifests('internal_sql', 'csv') def test_text( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type backends/postgres/dtypes/text | | | | Country | | | | | name@lt | string | | | | name@en | string - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('backends/postgres/dtypes/text/Country', [ @@ -48,18 +60,28 @@ def test_text( listdata(resp, full=True) +@pytest.mark.manifests('internal_sql', 'csv') def test_text_patch( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type backends/postgres/dtypes/text | | | | Country | | | | | name@lt | string | | | | name@en | string - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('backends/postgres/dtypes/text/Country', [ @@ -98,14 +120,23 @@ def test_text_patch( assert listdata(resp, full=True) == [{'name.lt': "Latvija", 'name.en': "Latvia"}] -def test_html(rc: RawConfig): - context, manifest = load_manifest_and_context(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_html( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + context, manifest = load_manifest_and_context( + rc, ''' d | r | b | m | property | type | access example | | | | | Country | | | | | | name@lt | string | open | | | | name@en | string | open - ''') + ''', + tmp_path=tmp_path, + manifest_type=manifest_type + ) result = render_data( context, manifest, 'example/Country', @@ -131,18 +162,28 @@ def test_html(rc: RawConfig): } +@pytest.mark.manifests('internal_sql', 'csv') def test_text_change_log( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type backends/postgres/dtypes/text | | | | Country | | | | | name@lt | string | | | | name@en | string - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'backends/postgres/dtypes/text/Country' app = create_test_client(context) app.authmodel('backends/postgres/dtypes/text/Country', [ @@ -178,12 +219,16 @@ def test_text_change_log( assert resp_changes.json()['_data'][1]['_op'] == 'delete' +@pytest.mark.manifests('internal_sql', 'csv') def test_text_select_by_prop( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | types/text | | | | | | | | | | | | | | | | | | | @@ -193,7 +238,13 @@ def test_text_select_by_prop( | | | | Country1 | | | | | | | | | | | | | | name@lt | string | | | | 3 | open | | | | | | | | name@en | string | | | | 3 | open | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'types/text/Country1' app = create_test_client(context) app.authmodel('types/text/Country1', [ @@ -219,18 +270,28 @@ def test_text_select_by_prop( assert sort_by_prop.status_code == 200 +@pytest.mark.manifests('internal_sql', 'csv') def test_text_post_wrong_property_with_text( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type backends/postgres/dtypes/text | | | | Country | | | | | name@lt | string | | | | name@en | string - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'backends/postgres/dtypes/text/Country' app = create_test_client(context) app.authmodel('backends/postgres/dtypes/text/Country', [ @@ -251,18 +312,28 @@ def test_text_post_wrong_property_with_text( assert resp.status_code != 200 +@pytest.mark.manifests('internal_sql', 'csv') def test_text_accept_language( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | types/text/accept | | | | | | | | | | | | | Country | | | | | | | | | | | | | | name@lt | string | | | | 3 | open | | | | | | | | name@en | string | | | | 3 | open | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'types/text/accept/Country' app = create_test_client(context) app.authmodel('types/text/accept/Country', [ @@ -300,18 +371,28 @@ def test_text_accept_language( }] +@pytest.mark.manifests('internal_sql', 'csv') def test_text_content_language( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | types/text/content | | | | | | | | | | | | | Country | | | | | | | | | | | | | | name@lt | string | | | | 3 | open | | | | | | | | name@en | string | | | | 3 | open | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'types/text/content/Country' app = create_test_client(context) app.authmodel('types/text/content/Country', [ @@ -340,19 +421,29 @@ def test_text_content_language( }] +@pytest.mark.manifests('internal_sql', 'csv') def test_text_unknown_language( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | types/text/content | | | | | | | | | | | | | Country | | | | | | | | | | | | | | name | text | | | | 3 | open | | | | | | | | name@lt | string | | | | | open | | | | | | | | name@en | string | | | | | open | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'types/text/content/Country' app = create_test_client(context) app.authmodel('types/text/content/Country', [ @@ -381,19 +472,29 @@ def test_text_unknown_language( }] +@pytest.mark.manifests('internal_sql', 'csv') def test_text_unknown_language_invalid( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | types/text/content | | | | | | | | | | | | | Country | | | | | | | | | | | | | | name | text | | | | 4 | open | | | | | | | | name@lt | string | | | | | open | | | | | | | | name@en | string | | | | | open | | | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) model = 'types/text/content/Country' app = create_test_client(context) app.authmodel('types/text/content/Country', [ diff --git a/tests/dtypes/test_time.py b/tests/dtypes/test_time.py index 414b0ae4d..0c2249dae 100644 --- a/tests/dtypes/test_time.py +++ b/tests/dtypes/test_time.py @@ -1,23 +1,36 @@ +from pathlib import Path + from pytest import FixtureRequest from spinta.core.config import RawConfig from spinta.testing.client import create_test_client from spinta.testing.data import listdata from spinta.testing.manifest import bootstrap_manifest +import pytest +@pytest.mark.manifests('internal_sql', 'csv') def test_time( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref backends/postgres/dtypes/time | | | | | City | | | | | | name | string | | | | | time | time | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('backends/postgres/dtypes/time/City', [ diff --git a/tests/dtypes/test_two_arrays.py b/tests/dtypes/test_two_arrays.py index 8ce11918e..9e9119560 100644 --- a/tests/dtypes/test_two_arrays.py +++ b/tests/dtypes/test_two_arrays.py @@ -2,8 +2,8 @@ @pytest.mark.models( - 'backends/mongo/dtypes/array/two_arrays', - 'backends/postgres/dtypes/array/two_arrays', + 'backends/mongo/dtypes/array/Two_arrays', + 'backends/postgres/dtypes/array/Two_arrays', ) def test_update_with_two_array(model, app): app.authmodel(model, ['insert', 'update', 'search']) diff --git a/tests/dtypes/test_uri.py b/tests/dtypes/test_uri.py index 244f65a03..9e7a08659 100644 --- a/tests/dtypes/test_uri.py +++ b/tests/dtypes/test_uri.py @@ -1,23 +1,36 @@ +from pathlib import Path + from pytest import FixtureRequest from spinta.core.config import RawConfig from spinta.testing.client import create_test_client from spinta.testing.data import listdata from spinta.testing.manifest import bootstrap_manifest +import pytest +@pytest.mark.manifests('internal_sql', 'csv') def test_uri( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref backends/postgres/dtypes/uri | | | | | City | | | | | | name | string | | | | | website | uri | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('backends/postgres/dtypes/uri/City', [ diff --git a/tests/formats/test_ascii.py b/tests/formats/test_ascii.py index cf6187e70..ae7aaeb44 100644 --- a/tests/formats/test_ascii.py +++ b/tests/formats/test_ascii.py @@ -1,10 +1,12 @@ import base64 import datetime import hashlib +from pathlib import Path import pytest from _pytest.fixtures import FixtureRequest +from spinta import commands from spinta.auth import AdminToken from spinta.components import Action from spinta.components import UrlParams @@ -76,8 +78,12 @@ def test_export_ascii(app, mocker): assert data == changes +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.asyncio -async def test_export_multiple_types(rc: RawConfig): +async def test_export_multiple_types( + manifest_type: str, + tmp_path: Path, + rc: RawConfig): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | @@ -88,7 +94,7 @@ async def test_export_multiple_types(rc: RawConfig): | | | C | | value | | | | | value | integer | | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) rows = [ {'_type': 'example/A', 'value': 1}, {'_type': 'example/A', 'value': 2}, @@ -103,7 +109,7 @@ async def test_export_multiple_types(rc: RawConfig): context.set('auth.token', AdminToken()) config = context.get('config') exporter = config.exporters['ascii'] - ns = manifest.namespaces[''] + ns = commands.get_namespace(context, manifest, '') params = UrlParams() assert ''.join(exporter(context, ns, Action.GETALL, params, rows)) == ( '\n' @@ -163,12 +169,16 @@ def test_export_ascii_params(app, mocker): ) +@pytest.mark.manifests('internal_sql', 'csv') def test_ascii_ref_dtype( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/ascii/ref | | | | | | Country | | name | @@ -176,7 +186,13 @@ def test_ascii_ref_dtype( | | | City | | name | | | | | name | string | | open | | | | country | ref | Country | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/ascii/ref', ['insert', 'search']) @@ -195,18 +211,28 @@ def test_ascii_ref_dtype( ) +@pytest.mark.manifests('internal_sql', 'csv') def test_ascii_file_dtype( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/ascii/file | | | | | | Country | | name | | | | | name | string | | open | | | | flag | file | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/ascii/file', ['insert', 'search']) @@ -228,8 +254,11 @@ def test_ascii_file_dtype( ) +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.asyncio async def test_ascii_getone( + manifest_type: str, + tmp_path: Path, rc: RawConfig, ): context, manifest = load_manifest_and_context(rc, ''' @@ -237,7 +266,7 @@ async def test_ascii_getone( example | | | | | | City | | name | | | | | name | string | | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) _id = '19e4f199-93c5-40e5-b04e-a575e81ac373' result = render_data( @@ -263,18 +292,28 @@ async def test_ascii_getone( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_ascii_params( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/ascii/params | | | | | | Country | | name | | | | | name | string | | open | | | | capital | string | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/ascii/params', ['insert', 'search']) @@ -307,18 +346,28 @@ def test_ascii_params( ) +@pytest.mark.manifests('internal_sql', 'csv') def test_ascii_multiline( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/ascii/params | | | | | | Country | | name | | | | | name | string | | open | | | | capital | string | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/ascii/params', ['insert', 'search']) diff --git a/tests/formats/test_csv.py b/tests/formats/test_csv.py index a7c496330..0af059347 100644 --- a/tests/formats/test_csv.py +++ b/tests/formats/test_csv.py @@ -1,5 +1,6 @@ import base64 - +from pathlib import Path +import pytest from _pytest.fixtures import FixtureRequest from spinta.core.config import RawConfig @@ -12,7 +13,7 @@ def test_export_csv(app): app.authorize(['spinta_set_meta_fields']) - app.authmodel('datasets/csv/country', [ + app.authmodel('datasets/csv/Country', [ 'insert', 'patch', 'getall', @@ -20,16 +21,16 @@ def test_export_csv(app): 'changes', ]) - resp = app.post('/datasets/csv/country', json={'_data': [ + resp = app.post('/datasets/csv/Country', json={'_data': [ { '_op': 'insert', - '_type': 'datasets/csv/country', + '_type': 'datasets/csv/Country', 'code': 'lt', 'title': 'Lithuania', }, { '_op': 'insert', - '_type': 'datasets/csv/country', + '_type': 'datasets/csv/Country', 'code': 'lv', 'title': 'LATVIA', }, @@ -37,23 +38,23 @@ def test_export_csv(app): assert resp.status_code == 200, resp.json() data = resp.json()['_data'] lv = data[1] - resp = app.patch(f'/datasets/csv/country/{lv["_id"]}/', json={ + resp = app.patch(f'/datasets/csv/Country/{lv["_id"]}/', json={ '_revision': lv['_revision'], 'title': 'Latvia', }) assert resp.status_code == 200, resp.json() assert app.get( - '/datasets/csv/country/:format/csv?select(code,title)&sort(+code)' + '/datasets/csv/Country/:format/csv?select(code,title)&sort(+code)' ).text == ( 'code,title\r\n' 'lt,Lithuania\r\n' 'lv,Latvia\r\n' ) - resp = app.get('/datasets/csv/country/:changes/:format/csv') + resp = app.get('/datasets/csv/Country/:changes/:format/csv') assert resp.status_code == 200 - assert resp.headers['content-disposition'] == 'attachment; filename="country.csv"' + assert resp.headers['content-disposition'] == 'attachment; filename="Country.csv"' header, *lines = resp.text.splitlines() header = header.split(',') assert header == [ @@ -83,15 +84,15 @@ def test_export_csv(app): def test_csv_limit(app: TestClient): - app.authmodel('country', ['insert', 'search', ]) - resp = app.post('/country', json={'_data': [ - {'_op': 'insert', '_type': 'country', 'code': 'lt', 'title': 'Lithuania'}, - {'_op': 'insert', '_type': 'country', 'code': 'lv', 'title': 'Latvia'}, - {'_op': 'insert', '_type': 'country', 'code': 'ee', 'title': 'Estonia'}, + app.authmodel('Country', ['insert', 'search', ]) + resp = app.post('/Country', json={'_data': [ + {'_op': 'insert', '_type': 'Country', 'code': 'lt', 'title': 'Lithuania'}, + {'_op': 'insert', '_type': 'Country', 'code': 'lv', 'title': 'Latvia'}, + {'_op': 'insert', '_type': 'Country', 'code': 'ee', 'title': 'Estonia'}, ]}) assert resp.status_code == 200, resp.json() - resp = app.get('/country/:format/csv?select(code,title)&sort(code)&limit(1)') + resp = app.get('/Country/:format/csv?select(code,title)&sort(code)&limit(1)') assert parse_csv(resp) == [ ['code', 'title'], ['ee', 'Estonia'], @@ -128,18 +129,28 @@ def test_csv_ref_dtype( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_csv_file_dtype( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/csv/file | | | | | | Country | | name | | | | | name | string | | open | | | | flag | file | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/csv/file', ['insert', 'search']) @@ -159,12 +170,16 @@ def test_csv_file_dtype( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_csv_empty_ref( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/csv/ref | | | | | | Country | | name | @@ -172,7 +187,13 @@ def test_csv_empty_ref( | | | City | | name | | | | | name | string | | open | | | | country | ref | Country | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) app.authmodel('example/csv/ref', ['insert', 'search']) @@ -188,12 +209,16 @@ def test_csv_empty_ref( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_csv_mixed_ref( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example/csv/ref | | | | | | Country | | name | @@ -201,7 +226,13 @@ def test_csv_mixed_ref( | | | City | | name | | | | | name | string | | open | | | | country | ref | Country | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) app.authmodel('example/csv/ref', ['insert', 'search']) diff --git a/tests/formats/test_helpers.py b/tests/formats/test_helpers.py index fd7464b4e..3b94db9bd 100644 --- a/tests/formats/test_helpers.py +++ b/tests/formats/test_helpers.py @@ -1,3 +1,4 @@ +from pathlib import Path from typing import List import pytest @@ -13,6 +14,7 @@ from spinta.core.config import RawConfig +@pytest.mark.manifests('internal_sql', 'csv') @pytest.mark.parametrize('query, header', [ ('', ['_type', '_id', '_revision', '_page', 'name', 'country._id']), ('count()', ['count()']), @@ -21,7 +23,12 @@ ('select(_id, country._id)', ['_id', 'country._id']), ('select(_id, country._id, country.name)', ['_id', 'country._id', 'country.name']), ]) -def test_get_model_tabular_header(rc: RawConfig, query: str, header: List[str]): +def test_get_model_tabular_header( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, + query: str, + header: List[str]): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | @@ -30,9 +37,9 @@ def test_get_model_tabular_header(rc: RawConfig, query: str, header: List[str]): | | | City | | | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) context.set('auth.token', AdminToken()) - model = manifest.models['example/City'] + model = commands.get_model(context, manifest, 'example/City') request = make_get_request(model.name, query) params = commands.prepare(context, UrlParams(), Version(), request) action = Action.SEARCH if query else Action.GETALL diff --git a/tests/formats/test_html.py b/tests/formats/test_html.py index b462e34af..a6b93609c 100644 --- a/tests/formats/test_html.py +++ b/tests/formats/test_html.py @@ -1,5 +1,7 @@ import base64 import hashlib +from pathlib import Path + from lxml import html from typing import Any from typing import Dict @@ -92,10 +94,10 @@ def test_select_with_joins(app): def test_limit_in_links(app): - app.authmodel('country', ['search', ]) - resp = app.get('/country/:format/html?limit(1)') + app.authmodel('Country', ['search', ]) + resp = app.get('/Country/:format/html?limit(1)') assert resp.context['formats'][0] == ( - 'CSV', '/country/:format/csv?limit(1)' + 'CSV', '/Country/:format/csv?limit(1)' ) @@ -110,11 +112,11 @@ def _get_current_loc(context: Context, path: str): params = commands.prepare(context, UrlParams(), Version(), request) if isinstance(params.model, Namespace): store: Store = context.get('store') - model = store.manifest.models['_ns'] + model = commands.get_model(context, store.manifest, '_ns') else: model = params.model config: Config = context.get('config') - return get_current_location(config, model, params) + return get_current_location(context, config, model, params) @pytest.fixture(scope='module') @@ -246,18 +248,27 @@ def test_current_location_with_root( def _prep_file_type( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ) -> Tuple[TestClient, str]: - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | access example/html/file | | | resource | | | | | Country | | | | | | name | string | open | | | | flag | image | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/html/file', [ 'insert', @@ -300,12 +311,15 @@ def _table_with_header( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_file_type_list( - postgresql: str, + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, request: FixtureRequest, ): - app, _id = _prep_file_type(rc, postgresql, request) + app, _id = _prep_file_type(manifest_type, tmp_path, rc, postgresql, request) resp = app.get('/example/html/file/Country', headers={ 'Accept': 'text/html', }) @@ -336,12 +350,15 @@ def _row(row: List[Tuple[str, Cell]]) -> List[Tuple[str, Dict[str, Any]]]: ] +@pytest.mark.manifests('internal_sql', 'csv') def test_file_type_details( - postgresql: str, + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, request: FixtureRequest, ): - app, _id = _prep_file_type(rc, postgresql, request) + app, _id = _prep_file_type(manifest_type, tmp_path, rc, postgresql, request) resp = app.get(f'/example/html/file/Country/{_id}', headers={ 'Accept': 'text/html', }) @@ -357,12 +374,15 @@ def test_file_type_details( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_file_type_changes( - postgresql: str, + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, request: FixtureRequest, ): - app, _id = _prep_file_type(rc, postgresql, request) + app, _id = _prep_file_type(manifest_type, tmp_path, rc, postgresql, request) resp = app.get('/example/html/file/Country/:changes', headers={ 'Accept': 'text/html', }) @@ -380,12 +400,15 @@ def test_file_type_changes( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_file_type_changes_single_object( - postgresql: str, + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, request: FixtureRequest, ): - app, _id = _prep_file_type(rc, postgresql, request) + app, _id = _prep_file_type(manifest_type, tmp_path, rc, postgresql, request) resp = app.get(f'/example/html/file/Country/{_id}/:changes', headers={ 'Accept': 'text/html', }) @@ -403,12 +426,15 @@ def test_file_type_changes_single_object( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_file_type_no_pk( - postgresql: str, + manifest_type: str, + tmp_path: Path, rc: RawConfig, + postgresql: str, request: FixtureRequest, ): - app, _id = _prep_file_type(rc, postgresql, request) + app, _id = _prep_file_type(manifest_type, tmp_path, rc, postgresql, request) resp = app.get('/example/html/file/Country?select(name, flag)', headers={ 'Accept': 'text/html', }) @@ -442,7 +468,12 @@ def test_limit_iter(limit, exhausted, result): assert it.exhausted is exhausted -def test_prepare_ref_for_response(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_prepare_ref_for_response( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | @@ -451,11 +482,11 @@ def test_prepare_ref_for_response(rc: RawConfig): | | | City | | name | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) fmt = Html() value = {'_id': 'c634dbd8-416f-457d-8bda-5a6c35bbd5d6'} cell = Cell('c634dbd8', link='/example/Country/c634dbd8-416f-457d-8bda-5a6c35bbd5d6') - dtype = manifest.models['example/City'].properties['country'].dtype + dtype = commands.get_model(context, manifest, 'example/City').properties['country'].dtype result = commands.prepare_dtype_for_response( context, fmt, @@ -472,7 +503,12 @@ def test_prepare_ref_for_response(rc: RawConfig): assert result['_id'].link == cell.link -def test_prepare_ref_for_response_empty(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_prepare_ref_for_response_empty( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | @@ -481,11 +517,11 @@ def test_prepare_ref_for_response_empty(rc: RawConfig): | | | City | | name | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) fmt = Html() value = None cell = Cell('', link=None, color=Color.null) - dtype = manifest.models['example/City'].properties['country'].dtype + dtype = commands.get_model(context, manifest, 'example/City').properties['country'].dtype result = commands.prepare_dtype_for_response( context, fmt, @@ -501,13 +537,18 @@ def test_prepare_ref_for_response_empty(rc: RawConfig): assert result.link == cell.link -def test_select_id(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_select_id( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | | | | City | | name | | | | | name | string | | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) result = render_data( context, manifest, 'example/City', @@ -527,7 +568,12 @@ def test_select_id(rc: RawConfig): } -def test_select_join(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_select_join( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | @@ -536,7 +582,7 @@ def test_select_join(rc: RawConfig): | | | City | | | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) result = render_data( context, manifest, 'example/City', @@ -558,8 +604,14 @@ def test_select_join(rc: RawConfig): } -def test_select_join_multiple_props(rc: RawConfig): - context, manifest = load_manifest_and_context(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_select_join_multiple_props( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + context, manifest = load_manifest_and_context( + rc, ''' d | r | b | m | property | type | ref | access example | | | | | | Country | | | @@ -567,7 +619,7 @@ def test_select_join_multiple_props(rc: RawConfig): | | | City | | | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) result = render_data( context, manifest, 'example/City', @@ -597,14 +649,19 @@ def test_select_join_multiple_props(rc: RawConfig): } -def test_recursive_refs(rc: RawConfig): +@pytest.mark.manifests('internal_sql', 'csv') +def test_recursive_refs( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access example | | | | | | Category | | | | | | | name | string | | open | | | | parent | ref | Category | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) result = render_data( context, manifest, 'example/Category/262f6c72-4284-4d26-b9b0-e282bfe46a46', @@ -646,19 +703,29 @@ def test_recursive_refs(rc: RawConfig): } +@pytest.mark.manifests('internal_sql', 'csv') def test_show_single_object( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ) -> Tuple[TestClient, str]: - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access example | | | | | | City | | id | | | | | id | integer | | open | | | | name | string | | open - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) app.authmodel('example/City', [ diff --git a/tests/formats/test_rdf.py b/tests/formats/test_rdf.py index 484590c5e..955efd360 100644 --- a/tests/formats/test_rdf.py +++ b/tests/formats/test_rdf.py @@ -1,4 +1,5 @@ import base64 +from pathlib import Path from _pytest.fixtures import FixtureRequest @@ -6,14 +7,19 @@ from spinta.testing.client import create_test_client from spinta.testing.data import pushdata, encode_page_values_manually from spinta.testing.manifest import bootstrap_manifest +import pytest +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_get_all_without_uri( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -23,7 +29,13 @@ def test_rdf_get_all_without_uri( | | | City | | name | | | | | | name | string | | open | | | | | country | ref | Country | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -58,12 +70,16 @@ def test_rdf_get_all_without_uri( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_get_all_with_uri( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -75,7 +91,13 @@ def test_rdf_get_all_with_uri( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | dct:country - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -112,12 +134,16 @@ def test_rdf_get_all_with_uri( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_get_one( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -129,7 +155,13 @@ def test_rdf_get_one( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | dct:country - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getone']) @@ -155,12 +187,16 @@ def test_rdf_get_one( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_with_file( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/file | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -170,7 +206,13 @@ def test_rdf_with_file( | | | Country | | name | | dcat:country | | | | name | string | | open | dct:name | | | | flag | file | | open | dct:flag - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf/file', ['insert', 'getone']) @@ -200,12 +242,16 @@ def test_rdf_with_file( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_get_with_uri_model_rename( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/rename | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -215,7 +261,13 @@ def test_rdf_get_with_uri_model_rename( | | | Country | | name | | dcat:country | | | | name | string | | open | dct:name | | | | country | uri | | open | dcat:country - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -241,12 +293,16 @@ def test_rdf_get_with_uri_model_rename( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_get_with_uri_ref_rename( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/rename | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -259,7 +315,13 @@ def test_rdf_get_with_uri_ref_rename( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | dct:country - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -289,12 +351,16 @@ def test_rdf_get_with_uri_ref_rename( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_empty_ref( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/ref | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -306,7 +372,13 @@ def test_rdf_empty_ref( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -330,12 +402,16 @@ def test_rdf_empty_ref( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_mixed_ref( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/ref/multi | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -347,7 +423,13 @@ def test_rdf_mixed_ref( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -384,12 +466,16 @@ def test_rdf_mixed_ref( f'\n' +@pytest.mark.manifests('internal_sql', 'csv') def test_rdf_namespace_all( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | access | uri example/rdf/ref/simple | | | | | | | | | prefix | rdf | | http://www.rdf.com @@ -412,7 +498,13 @@ def test_rdf_namespace_all( | | | City | | name | | dcat:city | | | | name | string | | open | dct:name | | | | country | ref | Country | open | - ''', backend=postgresql, request=request) + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + request=request, + full_load=True + ) app = create_test_client(context) app.authmodel('example/rdf', ['insert', 'getall']) @@ -439,7 +531,6 @@ def test_rdf_namespace_all( }) res = app.get("/example/rdf/ref/:all/:format/rdf").text - print(res) assert res == f'\n'\ f' str: @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_pdf_ref_update_accesslog(model, app, context, tmp_path): app.authmodel(model, ['insert', 'update', 'getone', 'pdf_getone', 'pdf_update', 'pdf_delete']) @@ -907,11 +907,11 @@ def test_pdf_ref_update_accesslog(model, app, context, tmp_path): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_batch_write(model, app, context, tmp_path): - ns = model[:-len('/report')] + ns = model[:-len('/Report')] app.authmodel(ns, ['insert']) @@ -955,11 +955,11 @@ def test_batch_write(model, app, context, tmp_path): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_stream_write(model, app, context, tmp_path): - ns = model[:-len('/report')] + ns = model[:-len('/Report')] app.authmodel(ns, ['insert']) @@ -1000,11 +1000,11 @@ def test_stream_write(model, app, context, tmp_path): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_ns_read(model, app, context, tmp_path): - ns = model[:-len('/report')] + ns = model[:-len('/Report')] app.authmodel(ns, ['getall']) @@ -1012,8 +1012,8 @@ def test_ns_read(model, app, context, tmp_path): assert resp.status_code == 200, resp.json() objects = { - 'backends/mongo/report': 20, - 'backends/postgres/report': 21, + 'backends/mongo/Report': 20, + 'backends/postgres/Report': 21, } accesslog = context.get('accesslog.stream') @@ -1044,11 +1044,11 @@ def test_ns_read(model, app, context, tmp_path): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_ns_read_csv(model, app, context, tmp_path): - ns = model[:-len('/report')] + ns = model[:-len('/Report')] app.authmodel(ns, ['getall']) @@ -1056,8 +1056,8 @@ def test_ns_read_csv(model, app, context, tmp_path): assert resp.status_code == 200 objects = { - 'backends/mongo/report': 20, - 'backends/postgres/report': 21, + 'backends/mongo/Report': 20, + 'backends/postgres/Report': 21, } accesslog = context.get('accesslog.stream') diff --git a/tests/test_api.py b/tests/test_api.py index ea8f1d112..84d7cc8fb 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -117,7 +117,7 @@ def test_app(app): ], } assert next(d for d in data['data'] if d['title'] == 'Country') == { - 'name': '📄 country', + 'name': '📄 Country', 'title': 'Country', 'description': '', } @@ -125,13 +125,13 @@ def test_app(app): html = get_html_tree(resp) rows = html.cssselect('table.table tr td:nth-child(1)') rows = [row.text_content().strip() for row in rows] - assert '📄 country' in rows + assert '📄 Country' in rows assert '📁 datasets/' in rows def test_directory(app): app.authorize(['spinta_datasets_getall']) - resp = app.get('/datasets/xlsx/rinkimai/:ns', headers={'accept': 'text/html'}) + resp = app.get('/datasets/xlsx/Rinkimai/:ns', headers={'accept': 'text/html'}) assert resp.status_code == 200 resp.context.pop('request') @@ -140,45 +140,45 @@ def test_directory(app): ('🏠', '/'), ('datasets', '/datasets'), ('xlsx', '/datasets/xlsx'), - ('rinkimai', None), + ('Rinkimai', None), ], 'empty': False, 'header': ['name', 'title', 'description'], 'data': [ [ - {'value': '📄 apygarda', 'link': '/datasets/xlsx/rinkimai/apygarda'}, + {'value': '📄 Apygarda', 'link': '/datasets/xlsx/Rinkimai/Apygarda'}, {'value': '', 'color': '#f5f5f5'}, {'value': '', 'color': '#f5f5f5'}, ], [ - {'value': '📄 apylinke', 'link': '/datasets/xlsx/rinkimai/apylinke'}, + {'value': '📄 Apylinke', 'link': '/datasets/xlsx/Rinkimai/Apylinke'}, {'value': '', 'color': '#f5f5f5'}, {'value': '', 'color': '#f5f5f5'}, ], [ - {'value': '📄 kandidatas', 'link': '/datasets/xlsx/rinkimai/kandidatas'}, + {'value': '📄 Kandidatas', 'link': '/datasets/xlsx/Rinkimai/Kandidatas'}, {'value': '', 'color': '#f5f5f5'}, {'value': '', 'color': '#f5f5f5'}, ], [ - {'value': '📄 turas', 'link': '/datasets/xlsx/rinkimai/turas'}, + {'value': '📄 Turas', 'link': '/datasets/xlsx/Rinkimai/Turas'}, {'value': '', 'color': '#f5f5f5'}, {'value': '', 'color': '#f5f5f5'}, ] ], 'formats': [ - ('CSV', '/datasets/xlsx/rinkimai/:ns/:format/csv'), - ('JSON', '/datasets/xlsx/rinkimai/:ns/:format/json'), - ('JSONL', '/datasets/xlsx/rinkimai/:ns/:format/jsonl'), - ('ASCII', '/datasets/xlsx/rinkimai/:ns/:format/ascii'), - ('RDF', '/datasets/xlsx/rinkimai/:ns/:format/rdf'), + ('CSV', '/datasets/xlsx/Rinkimai/:ns/:format/csv'), + ('JSON', '/datasets/xlsx/Rinkimai/:ns/:format/json'), + ('JSONL', '/datasets/xlsx/Rinkimai/:ns/:format/jsonl'), + ('ASCII', '/datasets/xlsx/Rinkimai/:ns/:format/ascii'), + ('RDF', '/datasets/xlsx/Rinkimai/:ns/:format/rdf'), ], } @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_model(model, context, app): app.authmodel(model, ['insert', 'getall']) @@ -197,7 +197,7 @@ def test_model(model, context, app): resp.context.pop('request') - backend = model[len('backends/'):len(model) - len('/report')] + backend = model[len('backends/'):len(model) - len('/Report')] assert _cleaned_context(resp, data=False) == { 'location': [ ('🏠', '/'), @@ -250,8 +250,8 @@ def test_model(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_model_get(model, app): app.authmodel(model, ['insert', 'getone']) @@ -328,15 +328,15 @@ def test_model_get(model, app): def test_dataset(app): - app.authmodel('datasets/json/rinkimai', ['insert', 'getall']) - resp = app.post('/datasets/json/rinkimai', json={ + app.authmodel('datasets/json/Rinkimai', ['insert', 'getall']) + resp = app.post('/datasets/json/Rinkimai', json={ 'id': '1', 'pavadinimas': 'Rinkimai 1', }) data = resp.json() assert resp.status_code == 201, data - resp = app.get('/datasets/json/rinkimai', headers={'accept': 'text/html'}) + resp = app.get('/datasets/json/Rinkimai', headers={'accept': 'text/html'}) assert resp.status_code == 200, resp.json() pk = data['_id'] @@ -346,39 +346,39 @@ def test_dataset(app): ('🏠', '/'), ('datasets', '/datasets'), ('json', '/datasets/json'), - ('rinkimai', None), - ('Changes', '/datasets/json/rinkimai/:changes/-10'), + ('Rinkimai', None), + ('Changes', '/datasets/json/Rinkimai/:changes/-10'), ], 'header': ['_id', 'id', 'pavadinimas'], 'empty': False, 'data': [ [ - {'link': f'/datasets/json/rinkimai/{pk}', 'value': pk[:8]}, + {'link': f'/datasets/json/Rinkimai/{pk}', 'value': pk[:8]}, {'value': '1'}, {'value': 'Rinkimai 1'}, ], ], 'formats': [ - ('CSV', '/datasets/json/rinkimai/:format/csv'), - ('JSON', '/datasets/json/rinkimai/:format/json'), - ('JSONL', '/datasets/json/rinkimai/:format/jsonl'), - ('ASCII', '/datasets/json/rinkimai/:format/ascii'), - ('RDF', '/datasets/json/rinkimai/:format/rdf'), + ('CSV', '/datasets/json/Rinkimai/:format/csv'), + ('JSON', '/datasets/json/Rinkimai/:format/json'), + ('JSONL', '/datasets/json/Rinkimai/:format/jsonl'), + ('ASCII', '/datasets/json/Rinkimai/:format/ascii'), + ('RDF', '/datasets/json/Rinkimai/:format/rdf'), ], } def test_dataset_with_show(context, app): - app.authmodel('/datasets/json/rinkimai', ['insert', 'search']) + app.authmodel('/datasets/json/Rinkimai', ['insert', 'search']) - resp = app.post('/datasets/json/rinkimai', json={ + resp = app.post('/datasets/json/Rinkimai', json={ 'id': '1', 'pavadinimas': 'Rinkimai 1', }) data = resp.json() assert resp.status_code == 201, data - resp = app.get('/datasets/json/rinkimai?select(pavadinimas)', headers={ + resp = app.get('/datasets/json/Rinkimai?select(pavadinimas)', headers={ 'accept': 'text/html', }) assert resp.status_code == 200 @@ -391,35 +391,35 @@ def test_dataset_with_show(context, app): def test_dataset_url_without_resource(context, app): - app.authmodel('datasets/json/rinkimai', ['insert', 'getall']) - resp = app.post('/datasets/json/rinkimai', json={ + app.authmodel('datasets/json/Rinkimai', ['insert', 'getall']) + resp = app.post('/datasets/json/Rinkimai', json={ 'id': '1', 'pavadinimas': 'Rinkimai 1', }) data = resp.json() pk = data['_id'] assert resp.status_code == 201, data - resp = app.get('/datasets/json/rinkimai', headers={'accept': 'text/html'}) + resp = app.get('/datasets/json/Rinkimai', headers={'accept': 'text/html'}) assert resp.status_code == 200 context = _cleaned_context(resp, data=False) assert context['header'] == ['_id', 'id', 'pavadinimas'] assert context['data'] == [[ - {'link': f'/datasets/json/rinkimai/{pk}', 'value': pk[:8]}, + {'link': f'/datasets/json/Rinkimai/{pk}', 'value': pk[:8]}, {'value': '1'}, {'value': 'Rinkimai 1'}, ]] def test_nested_dataset(app): - app.authmodel('datasets/nested/dataset/name/model', ['insert', 'getall']) - resp = app.post('/datasets/nested/dataset/name/model', json={ + app.authmodel('datasets/nested/dataset/name/Model', ['insert', 'getall']) + resp = app.post('/datasets/nested/dataset/name/Model', json={ 'name': 'Nested One', }) data = resp.json() assert resp.status_code == 201, data pk = data['_id'] - resp = app.get('/datasets/nested/dataset/name/model', headers={'accept': 'text/html'}) + resp = app.get('/datasets/nested/dataset/name/Model', headers={'accept': 'text/html'}) assert resp.status_code == 200 resp.context.pop('request') @@ -430,30 +430,30 @@ def test_nested_dataset(app): ('nested', '/datasets/nested'), ('dataset', '/datasets/nested/dataset'), ('name', '/datasets/nested/dataset/name'), - ('model', None), - ('Changes', '/datasets/nested/dataset/name/model/:changes/-10'), + ('Model', None), + ('Changes', '/datasets/nested/dataset/name/Model/:changes/-10'), ], 'header': ['_id', 'name'], 'empty': False, 'data': [ [ - {'link': f'/datasets/nested/dataset/name/model/{pk}', 'value': pk[:8]}, + {'link': f'/datasets/nested/dataset/name/Model/{pk}', 'value': pk[:8]}, {'value': 'Nested One'}, ], ], 'formats': [ - ('CSV', '/datasets/nested/dataset/name/model/:format/csv'), - ('JSON', '/datasets/nested/dataset/name/model/:format/json'), - ('JSONL', '/datasets/nested/dataset/name/model/:format/jsonl'), - ('ASCII', '/datasets/nested/dataset/name/model/:format/ascii'), - ('RDF', '/datasets/nested/dataset/name/model/:format/rdf'), + ('CSV', '/datasets/nested/dataset/name/Model/:format/csv'), + ('JSON', '/datasets/nested/dataset/name/Model/:format/json'), + ('JSONL', '/datasets/nested/dataset/name/Model/:format/jsonl'), + ('ASCII', '/datasets/nested/dataset/name/Model/:format/ascii'), + ('RDF', '/datasets/nested/dataset/name/Model/:format/rdf'), ], } def test_dataset_key(app): - app.authmodel('datasets/json/rinkimai', ['insert', 'getone']) - resp = app.post('/datasets/json/rinkimai', json={ + app.authmodel('datasets/json/Rinkimai', ['insert', 'getone']) + resp = app.post('/datasets/json/Rinkimai', json={ 'id': '1', 'pavadinimas': 'Rinkimai 1', }) @@ -462,7 +462,7 @@ def test_dataset_key(app): pk = data['_id'] rev = data['_revision'] - resp = app.get(f'/datasets/json/rinkimai/{pk}', headers={'accept': 'text/html'}) + resp = app.get(f'/datasets/json/Rinkimai/{pk}', headers={'accept': 'text/html'}) assert resp.status_code == 200 resp.context.pop('request') @@ -471,16 +471,16 @@ def test_dataset_key(app): ('🏠', '/'), ('datasets', '/datasets'), ('json', '/datasets/json'), - ('rinkimai', '/datasets/json/rinkimai'), + ('Rinkimai', '/datasets/json/Rinkimai'), (short_id(pk), None), - ('Changes', f'/datasets/json/rinkimai/{pk}/:changes/-10'), + ('Changes', f'/datasets/json/Rinkimai/{pk}/:changes/-10'), ], 'formats': [ - ('CSV', f'/datasets/json/rinkimai/{pk}/:format/csv'), - ('JSON', f'/datasets/json/rinkimai/{pk}/:format/json'), - ('JSONL', f'/datasets/json/rinkimai/{pk}/:format/jsonl'), - ('ASCII', f'/datasets/json/rinkimai/{pk}/:format/ascii'), - ('RDF', f'/datasets/json/rinkimai/{pk}/:format/rdf'), + ('CSV', f'/datasets/json/Rinkimai/{pk}/:format/csv'), + ('JSON', f'/datasets/json/Rinkimai/{pk}/:format/json'), + ('JSONL', f'/datasets/json/Rinkimai/{pk}/:format/jsonl'), + ('ASCII', f'/datasets/json/Rinkimai/{pk}/:format/ascii'), + ('RDF', f'/datasets/json/Rinkimai/{pk}/:format/rdf'), ], 'header': [ '_type', @@ -491,8 +491,8 @@ def test_dataset_key(app): ], 'empty': False, 'data': [[ - {'value': 'datasets/json/rinkimai'}, - {'value': short_id(pk), 'link': f'/datasets/json/rinkimai/{pk}'}, + {'value': 'datasets/json/Rinkimai'}, + {'value': short_id(pk), 'link': f'/datasets/json/Rinkimai/{pk}'}, {'value': rev}, {'value': '1'}, {'value': 'Rinkimai 1'}, @@ -501,9 +501,9 @@ def test_dataset_key(app): def test_changes_single_object(app: TestClient, mocker): - app.authmodel('datasets/json/rinkimai', ['insert', 'patch', 'changes']) + app.authmodel('datasets/json/Rinkimai', ['insert', 'patch', 'changes']) - model = 'datasets/json/rinkimai' + model = 'datasets/json/Rinkimai' obj = send(app, model, 'insert', { 'id': '1', @@ -514,7 +514,7 @@ def test_changes_single_object(app: TestClient, mocker): 'pavadinimas': 'Rinkimai 2', }) - resp = app.get(f'/datasets/json/rinkimai/{obj.id}/:changes/-10', headers={ + resp = app.get(f'/datasets/json/Rinkimai/{obj.id}/:changes/-10', headers={ 'accept': 'text/html', }) assert resp.status_code == 200 @@ -525,8 +525,8 @@ def test_changes_single_object(app: TestClient, mocker): ('🏠', '/'), ('datasets', '/datasets'), ('json', '/datasets/json'), - ('rinkimai', '/datasets/json/rinkimai'), - (obj1.sid, f'/datasets/json/rinkimai/{obj.id}'), + ('Rinkimai', '/datasets/json/Rinkimai'), + (obj1.sid, f'/datasets/json/Rinkimai/{obj.id}'), ('Changes', None), ], 'formats': [ @@ -562,7 +562,7 @@ def test_changes_single_object(app: TestClient, mocker): {'value': 2}, {'value': change[1]['_created']}, {'value': 'patch'}, - {'value': obj.sid, 'link': f'/datasets/json/rinkimai/{obj.id}'}, + {'value': obj.sid, 'link': f'/datasets/json/Rinkimai/{obj.id}'}, {'value': change[1]['_txn']}, {'value': obj1.rev}, {'value': '', 'color': '#f5f5f5'}, @@ -573,9 +573,9 @@ def test_changes_single_object(app: TestClient, mocker): def test_changes_object_list(app, mocker): - app.authmodel('datasets/json/rinkimai', ['insert', 'patch', 'changes']) + app.authmodel('datasets/json/Rinkimai', ['insert', 'patch', 'changes']) - model = 'datasets/json/rinkimai' + model = 'datasets/json/Rinkimai' obj = send(app, model, 'insert', { 'id': '1', @@ -586,7 +586,7 @@ def test_changes_object_list(app, mocker): 'pavadinimas': 'Rinkimai 2', }) - resp = app.get('/datasets/json/rinkimai/:changes/-10', headers={'accept': 'text/html'}) + resp = app.get('/datasets/json/Rinkimai/:changes/-10', headers={'accept': 'text/html'}) assert resp.status_code == 200 change = _cleaned_context(resp)['data'] @@ -595,7 +595,7 @@ def test_changes_object_list(app, mocker): ('🏠', '/'), ('datasets', '/datasets'), ('json', '/datasets/json'), - ('rinkimai', f'/{model}'), + ('Rinkimai', f'/{model}'), ('Changes', None), ], 'formats': [ @@ -642,19 +642,19 @@ def test_changes_object_list(app, mocker): def test_count(app): - app.authmodel('/datasets/json/rinkimai', ['upsert', 'search']) + app.authmodel('/datasets/json/Rinkimai', ['upsert', 'search']) - resp = app.post('/datasets/json/rinkimai', json={'_data': [ + resp = app.post('/datasets/json/Rinkimai', json={'_data': [ { '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="1"', 'id': '1', 'pavadinimas': 'Rinkimai 1', }, { '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="2"', 'id': '2', 'pavadinimas': 'Rinkimai 2', @@ -663,7 +663,7 @@ def test_count(app): # FIXME: Status code on multiple objects must be 207. assert resp.status_code == 200, resp.json() - resp = app.get('/datasets/json/rinkimai?count()', headers={'accept': 'text/html'}) + resp = app.get('/datasets/json/Rinkimai?count()', headers={'accept': 'text/html'}) assert resp.status_code == 200 context = _cleaned_context(resp) @@ -671,8 +671,8 @@ def test_count(app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post(model, context, app): app.authmodel(model, ['insert', 'getone']) @@ -725,8 +725,8 @@ def test_post(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_invalid_json(model, context, app): # tests 400 response on invalid json @@ -741,8 +741,8 @@ def test_post_invalid_json(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_empty_content(model, context, app): # tests posting empty content @@ -757,8 +757,8 @@ def test_post_empty_content(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_id(model, context, app): # tests 400 response when trying to create object with id @@ -778,8 +778,8 @@ def test_post_id(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_insufficient_scope(model, context, app): # tests 400 response when trying to create object with id @@ -798,8 +798,8 @@ def test_insufficient_scope(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_update_postgres(model, context, app): # tests if update works with `id` present in the json @@ -846,8 +846,8 @@ def test_post_update_postgres(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_revision(model, context, app): # tests 400 response when trying to create object with revision @@ -862,8 +862,8 @@ def test_post_revision(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_duplicate_id(model, app): # tests 400 response when trying to create object with id which exists @@ -887,8 +887,8 @@ def test_post_duplicate_id(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_patch_duplicate_id(model, context, app): # tests that duplicate ID detection works with PATCH requests @@ -897,13 +897,13 @@ def test_patch_duplicate_id(model, context, app): # create extra report resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'status': '1', }) assert resp.status_code == 201 data = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'status': '1', }).json() id_ = data['_id'] @@ -932,7 +932,7 @@ def test_patch_duplicate_id(model, context, app): # this should not be successful because id that we want to setup # already exists in database resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'status': '1', }) existing_id = resp.json()['_id'] @@ -946,8 +946,8 @@ def test_patch_duplicate_id(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_non_json_content_type(model, app): # tests 400 response when trying to make non-json request @@ -962,8 +962,8 @@ def test_post_non_json_content_type(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_bad_auth_header(model, app): # tests 400 response when authorization header is missing `Bearer ` @@ -977,8 +977,8 @@ def test_post_bad_auth_header(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_missing_auth_header(model, context, app, mocker): mocker.patch.object(context.get('config'), 'default_auth_client', None) @@ -991,8 +991,8 @@ def test_post_missing_auth_header(model, context, app, mocker): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_post_invalid_report_schema(model, app): # tests validation of correct value types according to manifest's schema @@ -1072,8 +1072,8 @@ def test_post_invalid_report_schema(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_streaming_response(model, app): app.authmodel(model, ['insert', 'getall']) @@ -1104,8 +1104,8 @@ def test_streaming_response(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_multi_backends(model, app): app.authmodel(model, ['insert', 'getone', 'getall', 'search']) @@ -1150,8 +1150,8 @@ def test_multi_backends(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_location_header(model, app, context): app.authmodel(model, ['insert']) @@ -1160,12 +1160,12 @@ def test_location_header(model, app, context): assert 'location' in resp.headers id_ = resp.json()['_id'] server_url = context.get('config').server_url - assert resp.headers['location'] == f'{server_url}{model}s/{id_}' + assert resp.headers['location'] == f'{server_url}{model}/{id_}' @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_upsert_where_ast(model, app): app.authmodel(model, ['upsert', 'changes']) @@ -1240,8 +1240,8 @@ def test_head_method( @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_delete( model: str, @@ -1257,8 +1257,8 @@ def test_delete( @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_delete_batch( model: str, @@ -1296,13 +1296,13 @@ def test_delete_batch( def test_get_gt_ge_lt_le_ne(app): app.authorize(['spinta_set_meta_fields']) - app.authmodel('/datasets/json/rinkimai', ['insert', 'upsert', 'search']) + app.authmodel('/datasets/json/Rinkimai', ['insert', 'upsert', 'search']) - resp = app.post('/datasets/json/rinkimai', json={'_data': [ + resp = app.post('/datasets/json/Rinkimai', json={'_data': [ { '_id': '3ba54cb1-2099-49c8-9b6e-629f6ad60a3b', '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="1"', 'id': '1', 'pavadinimas': 'Rinkimai 1', @@ -1310,7 +1310,7 @@ def test_get_gt_ge_lt_le_ne(app): { '_id': '1e4fff26-0082-4ce8-87b8-dd8abf7fadae', '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="2"', 'id': '2', 'pavadinimas': 'Rinkimai 2', @@ -1318,7 +1318,7 @@ def test_get_gt_ge_lt_le_ne(app): { '_id': '7109da99-6c02-49bf-97aa-a602e23b6659', '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="3"', 'id': '3', 'pavadinimas': 'Rinkimai 3', @@ -1326,7 +1326,7 @@ def test_get_gt_ge_lt_le_ne(app): { '_id': '959e8881-9d84-4b44-a0e8-31183aac0de6', '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="4"', 'id': '4', 'pavadinimas': 'Rinkimai 4', @@ -1334,7 +1334,7 @@ def test_get_gt_ge_lt_le_ne(app): { '_id': '24e613cc-3b3d-4075-96dd-093f2edbdf08', '_op': 'upsert', - '_type': 'datasets/json/rinkimai', + '_type': 'datasets/json/Rinkimai', '_where': f'id="5"', 'id': '5', 'pavadinimas': 'Rinkimai 5', @@ -1344,26 +1344,26 @@ def test_get_gt_ge_lt_le_ne(app): assert resp.status_code == 200, resp.json() eq_id_for_gt_ge = resp.json()['_data'][0]['_id'] - request_line = '/datasets/json/rinkimai?_id>"{0}"'.format(eq_id_for_gt_ge) + request_line = '/datasets/json/Rinkimai?_id>"{0}"'.format(eq_id_for_gt_ge) resp = app.get(request_line) assert len(resp.json()['_data']) == 2 - request_line = '/datasets/json/rinkimai?_id>="{0}"'.format(eq_id_for_gt_ge) + request_line = '/datasets/json/Rinkimai?_id>="{0}"'.format(eq_id_for_gt_ge) resp = app.get(request_line) assert len(resp.json()['_data']) == 3 eq_id_for_gt_ge = resp.json()['_data'][1]['_id'] - request_line = '/datasets/json/rinkimai?_id<="{0}"'.format(eq_id_for_gt_ge) + request_line = '/datasets/json/Rinkimai?_id<="{0}"'.format(eq_id_for_gt_ge) resp = app.get(request_line) assert len(resp.json()['_data']) == 4 - request_line = '/datasets/json/rinkimai?_id<"{0}"'.format(eq_id_for_gt_ge) + request_line = '/datasets/json/Rinkimai?_id<"{0}"'.format(eq_id_for_gt_ge) resp = app.get(request_line) assert len(resp.json()['_data']) == 3 - request_line = '/datasets/json/rinkimai?_id!="{0}"'.format(eq_id_for_gt_ge) + request_line = '/datasets/json/Rinkimai?_id!="{0}"'.format(eq_id_for_gt_ge) resp = app.get(request_line) assert len(resp.json()['_data']) == 4 diff --git a/tests/test_auth.py b/tests/test_auth.py index f68703028..b5d18c8c8 100644 --- a/tests/test_auth.py +++ b/tests/test_auth.py @@ -10,7 +10,7 @@ from authlib.jose import jwk from authlib.jose import jwt -from spinta import auth +from spinta import auth, commands from spinta.auth import get_client_file_path, query_client, get_clients_path from spinta.components import Action, Context from spinta.testing.cli import SpintaCliRunner @@ -168,26 +168,26 @@ def test_invalid_client(app): @pytest.mark.parametrize('client, scope, node, action, authorized', [ - ('default-client', 'spinta_getone', 'backends/mongo/subitem', 'getone', False), - ('test-client', 'spinta_getone', 'backends/mongo/subitem', 'getone', True), - ('test-client', 'spinta_getone', 'backends/mongo/subitem', 'insert', False), - ('test-client', 'spinta_getone', 'backends/mongo/subitem', 'update', False), - ('test-client', 'spinta_backends_getone', 'backends/mongo/subitem', 'getone', True), - ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/subitem', 'getone', True), - ('default-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/subitem', 'getone', False), - ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/subitem', 'insert', False), - ('test-client', 'spinta_getone', 'backends/mongo/subitem.subobj', 'getone', True), - ('test-client', 'spinta_backends_mongo_getone', 'backends/mongo/subitem.subobj', 'getone', True), - ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/subitem.subobj', 'getone', True), - ('test-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/subitem.subobj', 'getone', True), - ('test-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/subitem.subobj', 'insert', False), - ('default-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/subitem.subobj', 'getone', False), - ('test-client', 'spinta_getone', 'backends/mongo/subitem.hidden_subobj', 'getone', False), - ('test-client', 'spinta_backends_mongo_getone', 'backends/mongo/subitem.hidden_subobj', 'getone', False), - ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/subitem.hidden_subobj', 'getone', False), - ('test-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/subitem.hidden_subobj', 'getone', True), - ('test-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/subitem.hidden_subobj', 'update', False), - ('default-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/subitem.hidden_subobj', 'getone', False), + ('default-client', 'spinta_getone', 'backends/mongo/Subitem', 'getone', False), + ('test-client', 'spinta_getone', 'backends/mongo/Subitem', 'getone', True), + ('test-client', 'spinta_getone', 'backends/mongo/Subitem', 'insert', False), + ('test-client', 'spinta_getone', 'backends/mongo/Subitem', 'update', False), + ('test-client', 'spinta_backends_getone', 'backends/mongo/Subitem', 'getone', True), + ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/Subitem', 'getone', True), + ('default-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/Subitem', 'getone', False), + ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/Subitem', 'insert', False), + ('test-client', 'spinta_getone', 'backends/mongo/Subitem.subobj', 'getone', True), + ('test-client', 'spinta_backends_mongo_getone', 'backends/mongo/Subitem.subobj', 'getone', True), + ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/Subitem.subobj', 'getone', True), + ('test-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/Subitem.subobj', 'getone', True), + ('test-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/Subitem.subobj', 'insert', False), + ('default-client', 'spinta_backends_mongo_subitem_subobj_getone', 'backends/mongo/Subitem.subobj', 'getone', False), + ('test-client', 'spinta_getone', 'backends/mongo/Subitem.hidden_subobj', 'getone', False), + ('test-client', 'spinta_backends_mongo_getone', 'backends/mongo/Subitem.hidden_subobj', 'getone', False), + ('test-client', 'spinta_backends_mongo_subitem_getone', 'backends/mongo/Subitem.hidden_subobj', 'getone', False), + ('test-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/Subitem.hidden_subobj', 'getone', True), + ('test-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/Subitem.hidden_subobj', 'update', False), + ('default-client', 'spinta_backends_mongo_subitem_hidden_subobj_getone', 'backends/mongo/Subitem.hidden_subobj', 'getone', False), ]) def test_authorized(context, client, scope, node, action, authorized): if client == 'default-client': @@ -200,18 +200,18 @@ def test_authorized(context, client, scope, node, action, authorized): store = context.get('store') if '.' in node: model, prop = node.split('.', 1) - node = store.manifest.models[model].flatprops[prop] - elif node in store.manifest.models: - node = store.manifest.models[node] + node = commands.get_model(context, store.manifest, model).flatprops[prop] + elif commands.has_model(context, store.manifest, node): + node = commands.get_model(context, store.manifest, node) else: - node = store.manifest.objects['ns'][node] + node = commands.get_namespace(context, store.manifest, node) action = getattr(Action, action.upper()) assert auth.authorized(context, node, action) is authorized def test_invalid_access_token(app): app.headers.update({"Authorization": "Bearer FAKE_TOKEN"}) - resp = app.get('/reports') + resp = app.get('/Report') assert resp.status_code == 401 assert 'WWW-Authenticate' in resp.headers assert resp.headers['WWW-Authenticate'] == 'Bearer error="invalid_token"' @@ -238,7 +238,7 @@ def test_token_validation_key_config(backends, rc, tmp_path, request): token = auth.create_access_token(context, prvkey, client, scopes=scopes) client = create_test_client(context) - resp = client.get('/reports', headers={'Authorization': f'Bearer {token}'}) + resp = client.get('/Report', headers={'Authorization': f'Bearer {token}'}) assert resp.status_code == 200 @@ -274,7 +274,7 @@ def basic_auth(backends, rc, tmp_path, request): def test_http_basic_auth_unauthorized(basic_auth): client = basic_auth - resp = client.get('/reports') + resp = client.get('/Report') assert resp.status_code == 401, resp.json() assert resp.headers['www-authenticate'] == 'Basic realm="Authentication required."' assert resp.json() == { @@ -292,21 +292,21 @@ def test_http_basic_auth_unauthorized(basic_auth): def test_http_basic_auth_invalid_secret(basic_auth): client = basic_auth - resp = client.get('/reports', auth=('default', 'invalid')) + resp = client.get('/Report', auth=('default', 'invalid')) assert resp.status_code == 401, resp.json() assert resp.headers['www-authenticate'] == 'Basic realm="Authentication required."' def test_http_basic_auth_invalid_client(basic_auth): client = basic_auth - resp = client.get('/reports', auth=('invalid', 'secret')) + resp = client.get('/Report', auth=('invalid', 'secret')) assert resp.status_code == 401, resp.json() assert resp.headers['www-authenticate'] == 'Basic realm="Authentication required."' def test_http_basic_auth(basic_auth): client = basic_auth - resp = client.get('/reports', auth=('default', 'secret')) + resp = client.get('/Report', auth=('default', 'secret')) assert resp.status_code == 200, resp.json() diff --git a/tests/test_changes.py b/tests/test_changes.py index 7c37a38aa..90888c3c5 100644 --- a/tests/test_changes.py +++ b/tests/test_changes.py @@ -4,8 +4,8 @@ @pytest.mark.models( - 'backends/postgres/report', - # 'backends/mongo/report', + 'backends/postgres/Report', + # 'backends/mongo/Report', ) def test_changes(model, context, app): app.authmodel(model, ['insert', 'patch', 'changes']) @@ -37,8 +37,8 @@ def test_changes(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - # 'backends/mongo/report', + 'backends/postgres/Report', + # 'backends/mongo/Report', ) def test_changes_negative_offset(model, context, app): app.authmodel(model, ['insert', 'patch', 'changes']) @@ -73,8 +73,8 @@ def test_changes_negative_offset(model, context, app): @pytest.mark.models( - 'backends/postgres/report', - # 'backends/mongo/report', + 'backends/postgres/Report', + # 'backends/mongo/Report', ) def test_changes_empty_patch(model, context, app): app.authmodel(model, ['insert', 'patch', 'changes']) @@ -88,11 +88,11 @@ def test_changes_empty_patch(model, context, app): def test_changes_with_ref(context, app): - model = 'backends/postgres/country' + model = 'backends/postgres/Country' app.authmodel(model, ['insert']) country = send(app, model, 'insert', {'title': 'Lithuania'}) - model = 'backends/postgres/city' + model = 'backends/postgres/City' app.authmodel(model, ['insert', 'changes']) send(app, model, 'insert', {'title': 'Vilnius', 'country': {'_id': country.id}}) send(app, model, 'insert', {'title': 'Kaunas', 'country': {'_id': country.id}}) diff --git a/tests/test_checks.py b/tests/test_checks.py index edb15cb36..0a13349ba 100644 --- a/tests/test_checks.py +++ b/tests/test_checks.py @@ -4,7 +4,7 @@ from spinta import commands from spinta.core.config import RawConfig -from spinta.testing.manifest import load_manifest_and_context +from spinta.testing.manifest import load_manifest_and_context, load_manifest from spinta.testing.manifest import load_manifest_get_context from spinta.manifests.tabular.helpers import TabularManifestError from spinta.exceptions import InvalidValue @@ -12,9 +12,14 @@ from spinta.testing.tabular import create_tabular_manifest -def test_enum_level(tmp_path, rc): +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_level( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): with pytest.raises(TabularManifestError) as e: - context, manifest = load_manifest_and_context(rc, ''' + load_manifest(rc, ''' d | r | b | m | property | type | prepare | level | title datasets/gov/example | | | | | | | | @@ -22,49 +27,62 @@ def test_enum_level(tmp_path, rc): | | | | value | integer | | | | enum | 1 | 3 | Positive | | 2 | 3 | Negative - ''') - assert str(e.value) == ( - "None:6: Enum's do not have a level, but level '3' is given." + ''', manifest_type=manifest_type, tmp_path=tmp_path) + assert str(e.value).endswith( + ":6: Enum's do not have a level, but level '3' is given." ) -def test_enum_type_integer(tmp_path, rc): - context, manifest = load_manifest_and_context(rc, ''' - d | r | b | m | property | type | prepare - datasets/gov/example | | - | | - | | | Data | | - | | | | value | integer | - | enum | "1" - | | "2" - ''') +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_type_integer( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): with pytest.raises(InvalidValue) as e: - commands.check(context, manifest) + load_manifest(rc, ''' + d | r | b | m | property | type | prepare + datasets/gov/example | | + | | + | | | Data | | + | | | | value | integer | + | enum | "1" + | | "2" + ''', manifest_type=manifest_type, tmp_path=tmp_path) assert str(e.value.context['error']) == ( "Given enum value 1 of type does not match property " "type, which is 'integer'." ) -def test_enum_type_string(tmp_path, rc): - context, manifest = load_manifest_and_context(rc, ''' - d | r | b | m | property | type | prepare - datasets/gov/example | | - | | - | | | Data | | - | | | | value | string | - | enum | 1 - | | 2 - ''') +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_type_string( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): with pytest.raises(InvalidValue) as e: - commands.check(context, manifest) + load_manifest(rc, ''' + d | r | b | m | property | type | prepare + datasets/gov/example | | + | | + | | | Data | | + | | | | value | string | + | enum | 1 + | | 2 + ''', manifest_type=manifest_type, tmp_path=tmp_path) assert str(e.value.context['error']) == ( "Given enum value 1 of type does not match property " "type, which is 'string'." ) -def test_enum_type_none(tmp_path, rc): +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_type_none( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | source datasets/gov/example | | @@ -73,11 +91,16 @@ def test_enum_type_none(tmp_path, rc): | | | | value | string | | enum | 1 | | 2 - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) commands.check(context, manifest) -def test_enum_type_integer_negative(tmp_path, rc): +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_type_integer_negative( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | prepare datasets/gov/example | | @@ -85,11 +108,16 @@ def test_enum_type_integer_negative(tmp_path, rc): | | | | value | integer | | enum | 1 | | -2 - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) commands.check(context, manifest) -def test_enum_type_boolean(tmp_path, rc): +@pytest.mark.manifests('internal_sql', 'csv') +def test_enum_type_boolean( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | prepare datasets/gov/example | | @@ -97,12 +125,13 @@ def test_enum_type_boolean(tmp_path, rc): | | | | value | boolean | | enum | true | | false - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) commands.check(context, manifest) -def test_check_names_model(tmp_path: Path, rc: RawConfig): - create_tabular_manifest(tmp_path / 'hidrologija.csv', ''' +@pytest.mark.skip("SKIP FOR NOW, SINCE CHECK SHOULD ALSO BE ON LOAD") +def test_check_names_model(context, tmp_path: Path, rc: RawConfig): + create_tabular_manifest(context, tmp_path / 'hidrologija.csv', ''' d | r | b | m | property | type | source datasets/gov/example | | | | @@ -121,16 +150,19 @@ def test_check_names_model(tmp_path: Path, rc: RawConfig): assert e.value.message == "Invalid 'data' model code name." -def test_check_names_property(tmp_path: Path, rc: RawConfig): - create_tabular_manifest(tmp_path / 'hidrologija.csv', ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_check_names_property( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + context = load_manifest_get_context(rc, ''' d | r | b | m | property | type | source datasets/gov/example | | | | | | | Data | | | | | | value_Value | string | - ''') - - context = load_manifest_get_context(rc, tmp_path / 'hidrologija.csv', check_names=True) + ''', manifest_type=manifest_type, tmp_path=tmp_path, check_names=True) store = context.get('store') manifest = store.manifest @@ -141,16 +173,20 @@ def test_check_names_property(tmp_path: Path, rc: RawConfig): assert e.value.message == "Invalid 'value_Value' property code name." -def test_check_names_dataset(tmp_path: Path, rc: RawConfig): - create_tabular_manifest(tmp_path / 'hidrologija.csv', ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_check_names_dataset( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + + context = load_manifest_get_context(rc, ''' d | r | b | m | property | type | source datasets/gov/Example | | | | | | | Data | | | | | | value | string | - ''') - - context = load_manifest_get_context(rc, tmp_path / 'hidrologija.csv', check_names=True) + ''', manifest_type=manifest_type, tmp_path=tmp_path, check_names=True) store = context.get('store') manifest = store.manifest diff --git a/tests/test_concurency.py b/tests/test_concurency.py index afae869da..545fed027 100644 --- a/tests/test_concurency.py +++ b/tests/test_concurency.py @@ -4,8 +4,8 @@ @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_concurency(model, app): app.authmodel(model, ["insert", "getone"]) diff --git a/tests/test_exceptions.py b/tests/test_exceptions.py index 386a279fc..87b6d1186 100644 --- a/tests/test_exceptions.py +++ b/tests/test_exceptions.py @@ -1,6 +1,7 @@ # Register get_error_context commands. import spinta.commands # noqa import spinta.manifests.commands.error # noqa +from spinta import commands from spinta.exceptions import BaseError, error_response from spinta.components import Node @@ -126,7 +127,7 @@ class Error(BaseError): def test_this_model(context): - model = context.get('store').manifest.objects['model']['org'] + model = commands.get_model(context, context.get('store').manifest, 'Org') model.path = 'manifest/models/org.yml' error = Error(model) assert str(error) == ( @@ -135,12 +136,12 @@ def test_this_model(context): ' component: spinta.components.Model\n' ' manifest: default\n' ' schema: models/org.yml\n' - ' model: org\n' + ' model: Org\n' ) def test_this_model_property(context): - prop = context.get('store').manifest.objects['model']['org'].properties['title'] + prop = commands.get_model(context, context.get('store').manifest, 'Org').properties['title'] prop.model.path = 'manifest/models/org.yml' error = Error(prop) assert str(error) == ( @@ -149,13 +150,13 @@ def test_this_model_property(context): ' component: spinta.components.Property\n' ' manifest: default\n' ' schema: models/org.yml\n' - ' model: org\n' + ' model: Org\n' ' property: title\n' ) def test_this_model_property_dtype(context): - dtype = context.get('store').manifest.objects['model']['org'].properties['title'].dtype + dtype = commands.get_model(context, context.get('store').manifest, 'Org').properties['title'].dtype dtype.prop.model.path = 'manifest/models/org.yml' error = Error(dtype) assert str(error) == ( @@ -164,14 +165,14 @@ def test_this_model_property_dtype(context): ' component: spinta.types.datatype.String\n' ' manifest: default\n' ' schema: models/org.yml\n' - ' model: org\n' + ' model: Org\n' ' property: title\n' ' type: string\n' ) def test_this_dataset_model(context): - model = context.get('store').manifest.models['datasets/backends/postgres/dataset/report'] + model = commands.get_model(context, context.get('store').manifest, 'datasets/backends/postgres/dataset/Report') model.path = 'manifest/backends/postgres/dataset/report.yml' error = Error(model) assert str(error) == ( @@ -182,14 +183,14 @@ def test_this_dataset_model(context): ' schema: backends/postgres/dataset/report.yml\n' ' dataset: datasets/backends/postgres/dataset\n' ' resource: sql\n' - ' model: datasets/backends/postgres/dataset/report\n' + ' model: datasets/backends/postgres/dataset/Report\n' ' entity: reports\n' ' resource.backend: datasets/backends/postgres/dataset/sql\n' ) def test_this_dataset_model_property(context): - prop = context.get('store').manifest.models['datasets/backends/postgres/dataset/report'].properties['status'] + prop = commands.get_model(context, context.get('store').manifest, 'datasets/backends/postgres/dataset/Report').properties['status'] prop.model.path = 'manifest/backends/postgres/dataset/report.yml' error = Error(prop) assert str(error) == ( @@ -200,7 +201,7 @@ def test_this_dataset_model_property(context): ' schema: backends/postgres/dataset/report.yml\n' ' dataset: datasets/backends/postgres/dataset\n' ' resource: sql\n' - ' model: datasets/backends/postgres/dataset/report\n' + ' model: datasets/backends/postgres/dataset/Report\n' ' entity: reports\n' ' property: status\n' ' attribute: None\n' diff --git a/tests/test_freeze.py b/tests/test_freeze.py index 0964187c1..5ef745d86 100644 --- a/tests/test_freeze.py +++ b/tests/test_freeze.py @@ -27,7 +27,7 @@ def test_create_model(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, @@ -41,29 +41,29 @@ def test_create_model(rc, cli: SpintaCliRunner): 'country.yml': [ { 'type': 'model', - 'name': 'country', - 'id': 'country:0', - 'version': 'country:1', + 'name': 'Country', + 'id': 'Country:0', + 'version': 'Country:1', 'properties': { 'name': {'type': 'string'}, }, }, { - 'id': 'country:1', + 'id': 'Country:1', 'parents': [], 'migrate': [ { 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('name', string())", ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ], @@ -78,7 +78,7 @@ def test_add_column(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, @@ -102,25 +102,25 @@ def test_add_column(rc, cli: SpintaCliRunner): 'country.yml': [ { 'type': 'model', - 'name': 'country', - 'id': 'country:0', - 'version': 'country:2', + 'name': 'Country', + 'id': 'Country:0', + 'version': 'Country:2', 'properties': { 'name': {'type': 'string'}, 'code': {'type': 'string'}, }, }, { - 'id': 'country:2', + 'id': 'Country:2', 'parents': [], 'migrate': [ { 'type': 'schema', 'upgrade': [ - "add_column('country', column('code', string()))", + "add_column('Country', column('code', string()))", ], 'downgrade': [ - "drop_column('country', 'code')", + "drop_column('Country', 'code')", ], }, ], @@ -135,7 +135,7 @@ def test_freeze_no_changes(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, @@ -150,29 +150,29 @@ def test_freeze_no_changes(rc, cli: SpintaCliRunner): 'country.yml': [ { 'type': 'model', - 'name': 'country', - 'id': 'country:0', - 'version': 'country:1', + 'name': 'Country', + 'id': 'Country:0', + 'version': 'Country:1', 'properties': { 'name': {'type': 'string'}, }, }, { - 'id': 'country:1', + 'id': 'Country:1', 'parents': [], 'migrate': [ { 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('name', string())", ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ], @@ -187,7 +187,7 @@ def test_freeze_array(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'names': { 'type': 'array', @@ -207,28 +207,28 @@ def test_freeze_array(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country/:list/names',", + " 'Country/:list/names',", " column('_txn', uuid()),", - " column('_rid', ref('country._id', ondelete: 'CASCADE')),", + " column('_rid', ref('Country._id', ondelete: 'CASCADE')),", " column('names', string())", ")", ], 'downgrade': [ - "drop_table('country/:list/names')", + "drop_table('Country/:list/names')", ], }, { 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('names', json())", ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ] @@ -240,7 +240,7 @@ def test_freeze_array_with_object(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'notes': { 'type': 'array', @@ -260,13 +260,13 @@ def test_freeze_array_with_object(rc, cli: SpintaCliRunner): manifest = read_manifest_files(tmpdir) assert readable_manifest_files(manifest)['country.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('country/:list/notes')"], + 'downgrade': ["drop_table('Country/:list/notes')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'country/:list/notes',", + " 'Country/:list/notes',", " column('_txn', uuid()),", - " column('_rid', ref('country._id', ondelete: 'CASCADE')),", + " column('_rid', ref('Country._id', ondelete: 'CASCADE')),", " column('notes.note', string())", ')', ], @@ -275,14 +275,14 @@ def test_freeze_array_with_object(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('notes', json())", ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ] @@ -294,7 +294,7 @@ def test_freeze_object(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'report.yml': { 'type': 'model', - 'name': 'report', + 'name': 'Report', 'properties': { 'str': {'type': 'string'}, 'note': { @@ -319,13 +319,13 @@ def test_freeze_object(rc, cli: SpintaCliRunner): manifest = read_manifest_files(tmpdir) assert readable_manifest_files(manifest)['report.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('report/:list/note.list')"], + 'downgrade': ["drop_table('Report/:list/note.list')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'report/:list/note.list',", + " 'Report/:list/note.list',", " column('_txn', uuid()),", - " column('_rid', ref('report._id', ondelete: 'CASCADE')),", + " column('_rid', ref('Report._id', ondelete: 'CASCADE')),", " column('note.list', string())", ')', ], @@ -334,7 +334,7 @@ def test_freeze_object(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'report',", + " 'Report',", " column('_id', pk()),", " column('_revision', string()),", " column('note.list', json()),", @@ -344,7 +344,7 @@ def test_freeze_object(rc, cli: SpintaCliRunner): ")" ], 'downgrade': [ - "drop_table('report')", + "drop_table('Report')", ], }, ] @@ -356,7 +356,7 @@ def test_freeze_file(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'flag': {'type': 'file'}, 'anthem': { @@ -372,11 +372,11 @@ def test_freeze_file(rc, cli: SpintaCliRunner): manifest = read_manifest_files(tmpdir) assert readable_manifest_files(manifest)['country.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('country/:file/flag')"], + 'downgrade': ["drop_table('Country/:file/flag')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'country/:file/flag',", + " 'Country/:file/flag',", " column('_id', uuid()),", " column('_block', binary())", ')', @@ -386,7 +386,7 @@ def test_freeze_file(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('anthem._id', string()),", @@ -400,7 +400,7 @@ def test_freeze_file(rc, cli: SpintaCliRunner): ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ] @@ -412,7 +412,7 @@ def test_freeze_list_of_files(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'flags': { 'type': 'array', @@ -429,24 +429,24 @@ def test_freeze_list_of_files(rc, cli: SpintaCliRunner): manifest = read_manifest_files(tmpdir) assert readable_manifest_files(manifest)['country.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('country/:file/flags')"], + 'downgrade': ["drop_table('Country/:file/flags')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'country/:file/flags',", + " 'Country/:file/flags',", " column('_id', uuid()),", " column('_block', binary())", ')', ], }, { - 'downgrade': ["drop_table('country/:list/flags')"], + 'downgrade': ["drop_table('Country/:list/flags')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'country/:list/flags',", + " 'Country/:list/flags',", " column('_txn', uuid()),", - " column('_rid', ref('country._id', ondelete: 'CASCADE')),", + " column('_rid', ref('Country._id', ondelete: 'CASCADE')),", " column('flags._id', string()),", " column('flags._content_type', string()),", " column('flags._size', integer()),", @@ -459,14 +459,14 @@ def test_freeze_list_of_files(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('flags', json())", ")" ], 'downgrade': [ - "drop_table('country')", + "drop_table('Country')", ], }, ] @@ -478,18 +478,18 @@ def test_freeze_ref(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, }, 'city.yml': { 'type': 'model', - 'name': 'city', + 'name': 'City', 'properties': { 'country': { 'type': 'ref', - 'model': 'country' + 'model': 'Country' }, 'name': {'type': 'string'} } @@ -502,14 +502,14 @@ def test_freeze_ref(rc, cli: SpintaCliRunner): manifest_files = readable_manifest_files(manifest) assert manifest_files['city.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('city')"], + 'downgrade': ["drop_table('City')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'city',", + " 'City',", " column('_id', pk()),", " column('_revision', string()),", - " column('country._id', ref('country._id')),", + " column('country._id', ref('Country._id')),", " column('name', string())", ')', ], @@ -520,13 +520,13 @@ def test_freeze_ref(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('name', string())", ")" ], - 'downgrade': ["drop_table('country')"], + 'downgrade': ["drop_table('Country')"], }, ] @@ -537,14 +537,14 @@ def test_add_reference_column(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, }, 'city.yml': { 'type': 'model', - 'name': 'city', + 'name': 'City', 'properties': { 'name': {'type': 'string'} }, @@ -559,7 +559,7 @@ def test_add_reference_column(rc, cli: SpintaCliRunner): 'path': '/properties/city', 'value': { 'type': 'ref', - 'model': 'city', + 'model': 'City', }, } ] @@ -571,13 +571,13 @@ def test_add_reference_column(rc, cli: SpintaCliRunner): assert manifest_files['country.yml'][-1]['migrate'] == [ { 'downgrade': [ - "drop_column('country', 'city._id')", + "drop_column('Country', 'city._id')", ], 'type': 'schema', 'upgrade': [ "add_column(", - " 'country',", - " column('city._id', ref('city._id'))", + " 'Country',", + " column('city._id', ref('City._id'))", ")", ], }, @@ -590,25 +590,25 @@ def test_change_ref_model(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, }, 'continent.yml': { 'type': 'model', - 'name': 'continent', + 'name': 'Continent', 'properties': { 'name': {'type': 'string'}, }, }, 'city.yml': { 'type': 'model', - 'name': 'city', + 'name': 'City', 'properties': { 'country': { 'type': 'ref', - 'model': 'country', + 'model': 'Country', }, 'name': {'type': 'string'}, } @@ -621,7 +621,7 @@ def test_change_ref_model(rc, cli: SpintaCliRunner): { 'op': 'replace', 'path': '/properties/country/model', - 'value': 'continent', + 'value': 'Continent', } ] }) @@ -636,20 +636,20 @@ def test_freeze_ref_in_array(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'cities': { 'type': 'array', 'items': { 'type': 'ref', - 'model': 'city' + 'model': 'City' } }, }, }, 'city.yml': { 'type': 'model', - 'name': 'city', + 'name': 'City', 'properties': { 'name': {'type': 'string'} } @@ -662,11 +662,11 @@ def test_freeze_ref_in_array(rc, cli: SpintaCliRunner): manifest_files = readable_manifest_files(manifest) assert manifest_files['city.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_table('city')"], + 'downgrade': ["drop_table('City')"], 'type': 'schema', 'upgrade': [ 'create_table(', - " 'city',", + " 'City',", " column('_id', pk()),", " column('_revision', string()),", " column('name', string())", @@ -679,25 +679,25 @@ def test_freeze_ref_in_array(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "create_table(", - " 'country/:list/cities',", + " 'Country/:list/cities',", " column('_txn', uuid()),", - " column('_rid', ref('country._id', ondelete: 'CASCADE')),", - " column('cities._id', ref('city._id'))", + " column('_rid', ref('Country._id', ondelete: 'CASCADE')),", + " column('cities._id', ref('City._id'))", ")" ], - 'downgrade': ["drop_table('country/:list/cities')"], + 'downgrade': ["drop_table('Country/:list/cities')"], }, { 'type': 'schema', 'upgrade': [ "create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('cities', json())", ")" ], - 'downgrade': ["drop_table('country')"], + 'downgrade': ["drop_table('Country')"], } ] @@ -708,7 +708,7 @@ def test_change_field_type_in_object(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'population': { 'type': 'object', @@ -737,7 +737,7 @@ def test_change_field_type_in_object(rc, cli: SpintaCliRunner): { 'downgrade': [ 'alter_column(', - " 'country',", + " 'Country',", " 'population.amount',", ' type_: string()', ')', @@ -745,7 +745,7 @@ def test_change_field_type_in_object(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ 'alter_column(', - " 'country',", + " 'Country',", " 'population.amount',", ' type_: integer()', ')', @@ -760,7 +760,7 @@ def test_change_field_type_in_list(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'flags': { 'type': 'array', @@ -787,7 +787,7 @@ def test_change_field_type_in_list(rc, cli: SpintaCliRunner): { 'downgrade': [ 'alter_column(', - " 'country/:list/flags',", + " 'Country/:list/flags',", " 'flags',", ' type_: string()', ')', @@ -795,7 +795,7 @@ def test_change_field_type_in_list(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ 'alter_column(', - " 'country/:list/flags',", + " 'Country/:list/flags',", " 'flags',", ' type_: integer()', ')', @@ -810,7 +810,7 @@ def test_add_field_to_object(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'population': { 'type': 'object', @@ -840,10 +840,10 @@ def test_add_field_to_object(rc, cli: SpintaCliRunner): 'type': 'schema', 'upgrade': [ "add_column(", - " 'country',", + " 'Country',", " column('population.code', string())", ")"], - 'downgrade': ["drop_column('country', 'population.code')"], + 'downgrade': ["drop_column('Country', 'population.code')"], }, ] @@ -854,7 +854,7 @@ def test_add_field(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'} } @@ -882,8 +882,8 @@ def test_add_field(rc, cli: SpintaCliRunner): assert readable_manifest_files(manifest)['country.yml'][-1]['migrate'] == [ { 'type': 'schema', - 'upgrade': ["add_column('country', column('cities.name', string()))"], - 'downgrade': ["drop_column('country', 'cities.name')"], + 'upgrade': ["add_column('Country', column('cities.name', string()))"], + 'downgrade': ["drop_column('Country', 'cities.name')"], } ] @@ -894,7 +894,7 @@ def test_freeze_nullable(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': { 'type': 'string', @@ -910,12 +910,12 @@ def test_freeze_nullable(rc, cli: SpintaCliRunner): { 'type': 'schema', 'upgrade': ["create_table(", - " 'country',", + " 'Country',", " column('_id', pk()),", " column('_revision', string()),", " column('name', string(), nullable: true)", ")"], - 'downgrade': ["drop_table('country')"], + 'downgrade': ["drop_table('Country')"], } ] @@ -926,7 +926,7 @@ def test_change_nullable(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': { 'type': 'string', @@ -951,9 +951,9 @@ def test_change_nullable(rc, cli: SpintaCliRunner): manifest = read_manifest_files(tmpdir) assert readable_manifest_files(manifest)['country.yml'][-1]['migrate'] == [ { - 'downgrade': ["alter_column('country', 'name', nullable: true)"], + 'downgrade': ["alter_column('Country', 'name', nullable: true)"], 'type': 'schema', - 'upgrade': ["alter_column('country', 'name', nullable: false)"], + 'upgrade': ["alter_column('Country', 'name', nullable: false)"], }, ] @@ -964,7 +964,7 @@ def test_add_nullable_column(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, @@ -991,12 +991,12 @@ def test_add_nullable_column(rc, cli: SpintaCliRunner): assert manifest_files['country.yml'][-1]['migrate'] == [ { 'downgrade': [ - "drop_column('country', 'flag')", + "drop_column('Country', 'flag')", ], 'type': 'schema', 'upgrade': [ "add_column(", - " 'country',", + " 'Country',", " column('flag', string(), nullable: true)", ")", ], @@ -1010,7 +1010,7 @@ def test_delete_property(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, 'flag': {'type': 'string'} @@ -1040,7 +1040,7 @@ def test_delete_property_from_object(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, 'info': { @@ -1076,7 +1076,7 @@ def test_replace_all_properties(rc, cli: SpintaCliRunner): create_manifest_files(tmpdir, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, 'flag': {'type': 'string'} @@ -1100,8 +1100,8 @@ def test_replace_all_properties(rc, cli: SpintaCliRunner): manifest_files = readable_manifest_files(manifest) assert manifest_files['country.yml'][-1]['migrate'] == [ { - 'downgrade': ["drop_column('country', 'capital')"], + 'downgrade': ["drop_column('Country', 'capital')"], 'type': 'schema', - 'upgrade': ["add_column('country', column('capital', string()))"], + 'upgrade': ["add_column('Country', column('capital', string()))"], } ] diff --git a/tests/test_inspect.py b/tests/test_inspect.py index 79234f33d..c3de28d6c 100644 --- a/tests/test_inspect.py +++ b/tests/test_inspect.py @@ -8,13 +8,15 @@ import pytest +from spinta import commands from spinta.core.config import RawConfig from spinta.datasets.inspect.helpers import PriorityKey from spinta.manifests.tabular.helpers import striptable from spinta.testing.cli import SpintaCliRunner from spinta.testing.config import configure +from spinta.testing.context import create_test_context from spinta.testing.datasets import Sqlite -from spinta.testing.manifest import compare_manifest +from spinta.testing.manifest import compare_manifest, load_manifest_and_context from spinta.testing.tabular import create_tabular_manifest from spinta.testing.manifest import load_manifest @@ -29,7 +31,8 @@ def sqlite_new(): def rc_new(rc, tmp_path: pathlib.Path): # Need to have a clean slate, ignoring testing context manifests path = f'{tmp_path}/manifest.csv' - create_tabular_manifest(path, striptable(''' + context = create_test_context(rc) + create_tabular_manifest(context, path, striptable(''' d | r | b | m | property | type | ref | source | prepare ''')) return rc.fork({ @@ -72,8 +75,8 @@ def test_inspect( cli.invoke(rc_new, ['inspect', sqlite.dsn, '-o', tmp_path / 'result.csv']) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source | prepare dbsqlite | | | | @@ -103,7 +106,8 @@ def test_inspect_from_manifest_table( sa.Column('NAME', sa.Text), ], }) - create_tabular_manifest(tmp_path / 'manifest.csv', f''' + context = create_test_context(rc_new) + create_tabular_manifest(context, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | access dbsqlite | | | | | resource1 | sql | | {sqlite.dsn} | @@ -115,8 +119,8 @@ def test_inspect_from_manifest_table( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source | prepare dbsqlite | | | | @@ -153,8 +157,8 @@ def test_inspect_format( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'manifest.csv') - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'manifest.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | b | m | property | type | ref | source | prepare dbsqlite | | | | @@ -168,7 +172,7 @@ def test_inspect_format( | | | | code | string | | CODE | | | | | id | integer | | ID | | | | | name | string | | NAME | - ''') + ''', context) assert a == b @@ -199,8 +203,8 @@ def test_inspect_cyclic_refs( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'manifest.csv') - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'manifest.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source | prepare dbsqlite | | | | @@ -246,8 +250,8 @@ def test_inspect_self_refs( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'manifest.csv') - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'manifest.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source | prepare dbsqlite | | | | @@ -308,7 +312,7 @@ def test_inspect_oracle_sqldump_stdin( ''') # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'manifest.csv') + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'manifest.csv') assert manifest == ''' id | d | r | b | m | property | type | ref | source | prepare | level | access | uri | title | description | datasets/gov/example | | | | | | | | | @@ -341,8 +345,8 @@ def test_inspect_oracle_sqldump_file_with_formula( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'manifest.csv') - dataset = manifest.datasets['datasets/gov/example'] + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'manifest.csv') + dataset = commands.get_dataset(context, manifest, 'datasets/gov/example') dataset.resources['resource1'].external = 'dump.sql' assert manifest == ''' d | r | b | m | property | type | ref | source | prepare @@ -360,6 +364,7 @@ def test_inspect_with_schema( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'CITY': [ sa.Column('ID', sa.Integer, primary_key=True), @@ -368,7 +373,7 @@ def test_inspect_with_schema( }) # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | source | prepare dataset | | | | schema | sql | {sqlite.dsn} | connect(self, schema: null) @@ -377,8 +382,8 @@ def test_inspect_with_schema( cli.invoke(rc_new, ['inspect', tmp_path / 'manifest.csv', '-o', tmp_path / 'result.csv']) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') - manifest.datasets['dataset'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') + commands.get_dataset(context, manifest, 'dataset').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, ''' d | r | b | m | property | type | ref | source | prepare dataset | | | | @@ -387,7 +392,7 @@ def test_inspect_with_schema( | | | City | | id | CITY | | | | | id | integer | | ID | | | | | name | string | | NAME | - ''') + ''', context) assert a == b @@ -398,6 +403,7 @@ def test_inspect_update_existing_manifest( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('ID', sa.Integer, primary_key=True), @@ -412,14 +418,14 @@ def test_inspect_update_existing_manifest( }) # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' - d | r | m | property | type | ref | source | prepare | access | title - datasets/gov/example | | | | | | Example - | schema | sql | sql | | | | - | | | | | | - | | City | | id | CITY | id > 1 | | City - | | | id | integer | | ID | | private | - | | | name | string | | NAME | strip() | open | City name + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' + d | r | m | property | type | ref | source | prepare | access | title + datasets/gov/example | | | | | | Example + | schema | sql | sql | | | | + | | | | | | + | | City | | id | CITY | id > 1 | | City + | | | id | integer | | ID | | private | + | | | name | string | | NAME | strip() | open | City name ''') cli.invoke(rc_new, [ @@ -429,7 +435,7 @@ def test_inspect_update_existing_manifest( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') a, b = compare_manifest(manifest, ''' d | r | b | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -443,7 +449,7 @@ def test_inspect_update_existing_manifest( | | | Country | | id | COUNTRY | | | | | | | id | integer | | ID | | | | | | | name | string | | NAME | | | - ''') + ''', context) assert a == b @@ -454,6 +460,7 @@ def test_inspect_update_existing_ref_manifest_priority( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('ID', sa.Integer, primary_key=True), @@ -467,10 +474,10 @@ def test_inspect_update_existing_ref_manifest_priority( }) # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | Country | | id | COUNTRY | | | Country | | | id | integer | | ID | | private | Primary key @@ -488,12 +495,11 @@ def test_inspect_update_existing_ref_manifest_priority( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') a, b = compare_manifest(manifest, ''' d | r | b | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | | Country | | id | COUNTRY | | | Country | | | | id | integer | | ID | | private | Primary key @@ -503,7 +509,7 @@ def test_inspect_update_existing_ref_manifest_priority( | | | | id | integer | | ID | | private | | | | | name | string | | NAME | strip() | open | City name | | | | country | integer | | COUNTRY | | open | Country id - ''') + ''', context) assert a == b @@ -514,6 +520,7 @@ def test_inspect_update_existing_ref_external_priority( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('ID', sa.Integer, primary_key=True), @@ -527,10 +534,10 @@ def test_inspect_update_existing_ref_external_priority( }) # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | Country | | id | COUNTRY | | | Country | | | id | integer | | ID | | private | Primary key @@ -549,12 +556,11 @@ def test_inspect_update_existing_ref_external_priority( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv') - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') a, b = compare_manifest(manifest, ''' d | r | b | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | | Country | | id | COUNTRY | | | Country | | | | id | integer | | ID | | private | Primary key @@ -564,7 +570,7 @@ def test_inspect_update_existing_ref_external_priority( | | | | id | integer | | ID | | private | | | | | name | string | | NAME | strip() | open | City name | | | | country | ref | Country | COUNTRY | | open | Country id - ''') + ''', context) assert a == b @@ -595,8 +601,8 @@ def test_inspect_with_empty_config_dir( ]) # Check what was detected. - manifest = load_manifest(rc_new, tmp_path / 'result.csv', ensure_config_dir=True) - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, tmp_path / 'result.csv') + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source dbsqlite | | | @@ -629,8 +635,8 @@ def test_inspect_duplicate_table_names( ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source dbsqlite | | | @@ -670,8 +676,8 @@ def test_inspect_duplicate_column_names( ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' assert manifest == f''' d | r | b | m | property | type | ref | source dbsqlite | | | @@ -691,6 +697,7 @@ def test_inspect_existing_duplicate_table_names( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ '__COUNTRY': [sa.Column('NAME', sa.Text)], '_COUNTRY': [sa.Column('NAME', sa.Text)], @@ -699,10 +706,10 @@ def test_inspect_existing_duplicate_table_names( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | Country | | id | | | | Country | | | id | integer | | | | private | Primary key @@ -714,12 +721,11 @@ def test_inspect_existing_duplicate_table_names( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | Country | | id | | | | Country | | | id | integer | | | | private | Primary key @@ -733,7 +739,7 @@ def test_inspect_existing_duplicate_table_names( | | | | | | | | Country2 | | | __COUNTRY | | | | | | name | string | | NAME | | | - ''') + ''', context) assert a == b @@ -744,6 +750,7 @@ def test_inspect_existing_duplicate_column_names( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('__NAME', sa.Text), @@ -754,10 +761,10 @@ def test_inspect_existing_duplicate_column_names( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | Country | | | COUNTRY | | | Country | | | name | string | | | | open | Country name @@ -768,19 +775,18 @@ def test_inspect_existing_duplicate_column_names( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | Country | | | COUNTRY | | | Country | | | name | string | | | | open | Country name | | | name_2 | string | | NAME | | | | | | name_1 | string | | _NAME | | | | | | name_3 | string | | __NAME | | | - ''') + ''', context) assert a == b @@ -791,6 +797,7 @@ def test_inspect_insert_new_dataset( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -799,7 +806,7 @@ def test_inspect_insert_new_dataset( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | | | | | | @@ -813,8 +820,8 @@ def test_inspect_insert_new_dataset( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['dbsqlite'].resources['resource1'].external = "sqlite" + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = "sqlite" a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -826,7 +833,7 @@ def test_inspect_insert_new_dataset( | | | | | | | | Country | | | COUNTRY | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -837,6 +844,7 @@ def test_inspect_delete_model_source( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -845,10 +853,10 @@ def test_inspect_delete_model_source( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | City | | | CITY | | | City | | | name | string | | NAME | | open | City name @@ -859,19 +867,18 @@ def test_inspect_delete_model_source( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | City | | | | | | City | | | name | string | | | | open | City name | | | | | | | | Country | | | COUNTRY | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -882,6 +889,7 @@ def test_inspect_delete_property_source( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -890,10 +898,10 @@ def test_inspect_delete_property_source( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | {sqlite.dsn} | | | + | schema | sql | sql | | | | | | | | | | | | Country | | | COUNTRY | | | Country | | | name | string | | NAME | | open | Country name @@ -905,17 +913,16 @@ def test_inspect_delete_property_source( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example - | schema | sql | | sqlite | | | + | schema | sql | sql | | | | | | | | | | | | Country | | | COUNTRY | | | Country | | | name | string | | NAME | | open | Country name | | | code | string | | | | open | Country code -''') +''', context) assert a == b @@ -927,6 +934,7 @@ def test_inspect_multiple_resources_all_new( sqlite_new: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -941,7 +949,7 @@ def test_inspect_multiple_resources_all_new( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -954,9 +962,9 @@ def test_inspect_multiple_resources_all_new( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/example'].resources['schema_1'].external = 'sqlite_new' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema_1'].external = 'sqlite_new' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -969,7 +977,7 @@ def test_inspect_multiple_resources_all_new( | | Country1 | | | COUNTRY | | | | | | code | string | | CODE | | | -''') +''', context) assert a == b @@ -981,6 +989,7 @@ def test_inspect_multiple_resources_specific( sqlite_new: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -1005,7 +1014,7 @@ def test_inspect_multiple_resources_specific( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1024,9 +1033,9 @@ def test_inspect_multiple_resources_specific( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/example'].resources['schema_1'].external = 'sqlite_new' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema_1'].external = 'sqlite_new' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1045,7 +1054,7 @@ def test_inspect_multiple_resources_specific( | | | id | integer | | ID | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -1057,6 +1066,7 @@ def test_inspect_multiple_resources_advanced( sqlite_new: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -1081,7 +1091,7 @@ def test_inspect_multiple_resources_advanced( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | | Location | | | | | | @@ -1118,9 +1128,9 @@ def test_inspect_multiple_resources_advanced( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/example'].resources['schema_1'].external = 'sqlite_new' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema_1'].external = 'sqlite_new' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1163,7 +1173,7 @@ def test_inspect_multiple_resources_advanced( | | | code | string | | CODE | | | | | | continent | ref | Continent | CONTINENT | | | -''') +''', context) assert a == b @@ -1174,6 +1184,7 @@ def test_inspect_multiple_datasets( sqlite: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -1186,7 +1197,7 @@ def test_inspect_multiple_datasets( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1206,9 +1217,9 @@ def test_inspect_multiple_datasets( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/loc'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/loc').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1231,7 +1242,7 @@ def test_inspect_multiple_datasets( | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -1242,6 +1253,7 @@ def test_inspect_multiple_datasets_advanced_manifest_priority( sqlite: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('CODE', sa.Text), @@ -1256,7 +1268,7 @@ def test_inspect_multiple_datasets_advanced_manifest_priority( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1280,9 +1292,9 @@ def test_inspect_multiple_datasets_advanced_manifest_priority( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/loc'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/loc').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1309,7 +1321,7 @@ def test_inspect_multiple_datasets_advanced_manifest_priority( | | | new_id | string | | ID | | | | | | code | string | | CODE | | | | | | name_1 | string | | NAME | | | -''') +''', context) assert a == b @@ -1320,6 +1332,7 @@ def test_inspect_multiple_datasets_advanced_external_priority( sqlite: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('CODE', sa.Text), @@ -1334,7 +1347,7 @@ def test_inspect_multiple_datasets_advanced_external_priority( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1359,9 +1372,9 @@ def test_inspect_multiple_datasets_advanced_external_priority( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['schema'].external = 'sqlite' - manifest.datasets['datasets/gov/loc'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['schema'].external = 'sqlite' + commands.get_dataset(context, manifest, 'datasets/gov/loc').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1388,7 +1401,7 @@ def test_inspect_multiple_datasets_advanced_external_priority( | | | new_id | integer | | ID | | | | | | code | string | | CODE | | | | | | name_1 | string | | NAME | | | -''') +''', context) assert a == b @@ -1400,6 +1413,7 @@ def test_inspect_multiple_datasets_different_resources( sqlite_new: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('CODE', sa.Text), @@ -1425,7 +1439,7 @@ def test_inspect_multiple_datasets_different_resources( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/loc | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1440,9 +1454,9 @@ def test_inspect_multiple_datasets_different_resources( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/car'].resources['schema'].external = 'sqlite_new' - manifest.datasets['datasets/gov/loc'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/car').resources['schema'].external = 'sqlite_new' + commands.get_dataset(context, manifest, 'datasets/gov/loc').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/loc | | | | | | Example @@ -1468,7 +1482,7 @@ def test_inspect_multiple_datasets_different_resources( | | | id | integer | | ID | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -1480,6 +1494,7 @@ def test_inspect_multiple_datasets_different_resources_specific( sqlite_new: Sqlite ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('CODE', sa.Text), @@ -1505,7 +1520,7 @@ def test_inspect_multiple_datasets_different_resources_specific( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/loc | | | | | | Example | schema | sql | | {sqlite.dsn} | | | @@ -1528,9 +1543,9 @@ def test_inspect_multiple_datasets_different_resources_specific( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/car'].resources['schema'].external = 'sqlite_new' - manifest.datasets['datasets/gov/loc'].resources['schema'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/car').resources['schema'].external = 'sqlite_new' + commands.get_dataset(context, manifest, 'datasets/gov/loc').resources['schema'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/loc | | | | | | Example @@ -1552,7 +1567,7 @@ def test_inspect_multiple_datasets_different_resources_specific( | | | code | string | | CODE | | | | | | id | integer | | ID | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -1588,9 +1603,9 @@ def test_inspect_with_views( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['dbsqlite'].resources['resource1'].external = 'sqlite' - manifest.datasets['dbsqlite/views'].resources['resource1'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'dbsqlite').resources['resource1'].external = 'sqlite' + commands.get_dataset(context, manifest, 'dbsqlite/views').resources['resource1'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title dbsqlite | | | | | | @@ -1612,7 +1627,7 @@ def test_inspect_with_views( | | | continent | integer | | CONTINENT | | | | | | name | string | | NAME | | | -''') +''', context) assert a == b @@ -1624,6 +1639,7 @@ def test_inspect_with_manifest_backends( sqlite: Sqlite, ): # Prepare source data. + context = create_test_context(rc_new) sqlite.init({ 'COUNTRY': [ sa.Column('NAME', sa.Text), @@ -1632,7 +1648,7 @@ def test_inspect_with_manifest_backends( result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, sqlite, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, sqlite, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source | prepare | access | title | test | sql | | {sqlite.dsn} | | | | | | | | | @@ -1649,8 +1665,8 @@ def test_inspect_with_manifest_backends( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/gov/example'].resources['test'].external = 'sqlite' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/gov/example').resources['test'].external = 'sqlite' a, b = compare_manifest(manifest, f''' d | r | m | property | type | ref | source | prepare | access | title datasets/gov/example | | | | | | Example @@ -1661,7 +1677,7 @@ def test_inspect_with_manifest_backends( | | Country | | | COUNTRY | | | Country | | | name | string | | NAME | | open | Country name | | | code | string | | | | open | Country code -''') +''', context) assert a == b @@ -1706,10 +1722,11 @@ def test_inspect_json_model_ref_change( ] path = tmp_path / 'manifest.json' path.write_text(json.dumps(json_manifest)) + context = create_test_context(rc_new) result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc_new = configure(rc_new, None, tmp_path / 'manifest.csv', f''' + rc_new = configure(context, rc_new, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source datasets/json/inspect | | | | resource | json | | {path} @@ -1732,8 +1749,8 @@ def test_inspect_json_model_ref_change( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc_new, result_file_path) - manifest.datasets['datasets/json/inspect'].resources['resource'].external = 'resource.json' + context, manifest = load_manifest_and_context(rc_new, result_file_path) + commands.get_dataset(context, manifest, 'datasets/json/inspect').resources['resource'].external = 'resource.json' a, b = compare_manifest(manifest, f''' d | r | model | property | type | ref | source datasets/json/inspect | | | @@ -1750,7 +1767,7 @@ def test_inspect_json_model_ref_change( | | | weather_temperature | number unique | | weather.temperature | | | weather_wind_speed | number unique | | weather.wind_speed | | | parent | ref | Pos | .. - ''') + ''', context) assert a == b @@ -1783,10 +1800,11 @@ def test_inspect_xml_model_ref_change( ''' path = tmp_path / 'manifest.xml' path.write_text(xml) + context = create_test_context(rc) result_file_path = tmp_path / 'result.csv' # Configure Spinta. - rc = configure(rc, None, tmp_path / 'manifest.csv', f''' + rc = configure(context, rc, None, tmp_path / 'manifest.csv', f''' d | r | m | property | type | ref | source datasets/xml/inspect | | | | resource | xml | | {path} @@ -1809,8 +1827,8 @@ def test_inspect_xml_model_ref_change( '-o', tmp_path / 'result.csv', ]) # Check what was detected. - manifest = load_manifest(rc, result_file_path) - manifest.datasets['datasets/xml/inspect'].resources['resource'].external = 'resource.xml' + context, manifest = load_manifest_and_context(rc, result_file_path) + commands.get_dataset(context, manifest, 'datasets/xml/inspect').resources['resource'].external = 'resource.xml' a, b = compare_manifest(manifest, f''' d | r | model | property | type | ref | source datasets/xml/inspect | | | @@ -1827,7 +1845,7 @@ def test_inspect_xml_model_ref_change( | | | weather_temperature | number unique | | weather/temperature | | | weather_wind_speed | number unique | | weather/wind_speed | | | country | ref | Country | .. - ''') + ''', context) assert a == b diff --git a/tests/test_joins.py b/tests/test_joins.py index 14a5fab2d..0193163ea 100644 --- a/tests/test_joins.py +++ b/tests/test_joins.py @@ -20,35 +20,35 @@ def create_cities(app: TestClient, backend: str): +-----------+-----------+---------+ """ - app.authmodel(f'backends/{backend}/continent', ['insert']) - app.authmodel(f'backends/{backend}/country', ['insert']) - app.authmodel(f'backends/{backend}/city', ['insert', 'search']) + app.authmodel(f'backends/{backend}/Continent', ['insert']) + app.authmodel(f'backends/{backend}/Country', ['insert']) + app.authmodel(f'backends/{backend}/City', ['insert', 'search']) # Add a continent - eu = pushdata(app, f'/backends/{backend}/continent', { + eu = pushdata(app, f'/backends/{backend}/Continent', { 'title': 'Europe', }) # Add countries - lt = pushdata(app, f'/backends/{backend}/country', { + lt = pushdata(app, f'/backends/{backend}/Country', { 'title': 'Lithuania', 'continent': {'_id': eu['_id']}, }) - lv = pushdata(app, f'/backends/{backend}/country', { + lv = pushdata(app, f'/backends/{backend}/Country', { 'title': 'Latvia', 'continent': {'_id': eu['_id']}, }) # Add cities - pushdata(app, f'/backends/{backend}/city', { + pushdata(app, f'/backends/{backend}/City', { 'title': 'Vilnius', 'country': {'_id': lt['_id']}, }) - pushdata(app, f'/backends/{backend}/city', { + pushdata(app, f'/backends/{backend}/City', { 'title': 'Kaunas', 'country': {'_id': lt['_id']}, }) - pushdata(app, f'/backends/{backend}/city', { + pushdata(app, f'/backends/{backend}/City', { 'title': 'Riga', 'country': {'_id': lv['_id']}, }) @@ -57,12 +57,12 @@ def create_cities(app: TestClient, backend: str): @pytest.mark.parametrize('backend', ['postgres']) def test_select_with_joins(app, backend): create_cities(app, backend) - app.authmodel(f'backends/{backend}/city', ['search']) + app.authmodel(f'backends/{backend}/City', ['search']) # XXX: Maybe we should require `search` scope also for linked models? Now, # we only have access to `continent`, but using foreign keys, we can # also access country and continent. resp = app.get( - f'/backends/{backend}/city' + f'/backends/{backend}/City' '?select(title,country.title,country.continent.title)' '&sort(+_id)' ) diff --git a/tests/test_manifests.py b/tests/test_manifests.py index 2345c823e..8087753bf 100644 --- a/tests/test_manifests.py +++ b/tests/test_manifests.py @@ -2,22 +2,23 @@ from spinta.testing.cli import SpintaCliRunner from spinta.testing.utils import create_manifest_files from spinta.testing.context import create_test_context -from spinta.components import Model -from spinta.manifests.components import Manifest +from spinta.components import Model, Context +from spinta.manifests.components import Manifest, get_manifest_object_names -def show(c: Manifest): +def show(context: Context, c: Manifest): if isinstance(c, Manifest): res = { 'type': c.type, 'nodes': {}, } - for group, nodes in c.objects.items(): - if nodes: - res['nodes'][group] = { - name: show(node) - for name, node in nodes.items() - } + for group in get_manifest_object_names(): + res['nodes'][group] = { + name: show(context, node) + for name, node in commands.get_nodes(context, c, group).items() + } + if not res['nodes'][group]: + res['nodes'].pop(group) return res if isinstance(c, Model): return { @@ -44,7 +45,7 @@ def test_manifest_loading(postgresql, rc, cli: SpintaCliRunner, tmp_path, reques create_manifest_files(tmp_path, { 'country.yml': { 'type': 'model', - 'name': 'country', + 'name': 'Country', 'properties': { 'name': {'type': 'string'}, }, @@ -65,7 +66,7 @@ def test_manifest_loading(postgresql, rc, cli: SpintaCliRunner, tmp_path, reques request.addfinalizer(context.wipe_all) - assert show(store.manifest) == { + assert show(context, store.manifest) == { 'type': 'backend', 'nodes': { 'ns': { @@ -85,7 +86,7 @@ def test_manifest_loading(postgresql, rc, cli: SpintaCliRunner, tmp_path, reques '_txn': { 'backend': 'default', }, - 'country': { + 'Country': { 'backend': 'default', }, }, diff --git a/tests/test_migrations.py b/tests/test_migrations.py new file mode 100644 index 000000000..e69de29bb diff --git a/tests/test_models.py b/tests/test_models.py deleted file mode 100644 index 554f12f66..000000000 --- a/tests/test_models.py +++ /dev/null @@ -1,61 +0,0 @@ -import traceback - -import pytest - -from spinta.testing.utils import create_manifest_files -from spinta.testing.context import create_test_context - - -def check_store(rc, tmp_path, files): - create_manifest_files(tmp_path, files) - context = create_test_context(rc) - context.load({ - 'manifests': { - 'default': { - 'path': str(tmp_path), - } - } - }) - - -def test_engine_name_overshadow(rc, tmp_path): - with pytest.raises(Exception) as e: - check_store(rc, tmp_path, { - 'models/report.yml': { - 'type': 'model', - 'name': 'report', - 'endpoint': 'report', - }, - }) - assert str(e.value) == ( - "Endpoint name can't overshadow existing model names and 'report' is " - "already a model name." - ) - - -@pytest.mark.skip('datasets') -def test_engine_name_overshadow_other(rc, tmp_path): - with pytest.raises(Exception) as e: - check_store(rc, tmp_path, { - 'models/report.yml': { - 'type': 'model', - 'name': 'report', - }, - 'datasets/report.yml': { - 'type': 'dataset', - 'name': 'report', - 'resources': { - 'res': { - 'objects': { - '': { - 'rep': { - 'endpoint': 'report', - } - } - } - } - } - }, - }) - traceback.print_exception(e.type, e.value, e.tb) - assert "Endpoint name can't overshadow existing model names and 'report' is already a model name." in str(e.value) diff --git a/tests/test_namespace.py b/tests/test_namespace.py index 2d4116c23..9fe5ab516 100644 --- a/tests/test_namespace.py +++ b/tests/test_namespace.py @@ -1,23 +1,24 @@ import hashlib +from pathlib import Path from typing import Tuple import pytest +from spinta import commands from spinta.core.config import RawConfig from spinta.testing.client import TestClient from spinta.testing.client import create_test_client from spinta.testing.data import listdata from spinta.testing.data import pushdata -from spinta.testing.manifest import bootstrap_manifest -from spinta.testing.manifest import load_manifest +from spinta.testing.manifest import bootstrap_manifest, load_manifest_and_context from spinta.types.namespace import sort_models_by_refs from spinta.utils.data import take def _create_data(app: TestClient, ns: str) -> Tuple[str, str]: - continent = ns + '/continent' - country = ns + '/country' - capital = ns + '/capital' + continent = ns + '/Continent' + country = ns + '/Country' + capital = ns + '/Capital' app.authmodel(continent, ['insert']) app.authmodel(country, ['insert']) @@ -66,18 +67,22 @@ def test_getall_ns(model, app): resp = app.get('/datasets/backends/postgres/dataset/:ns/:all') assert listdata(resp, 'name') == [ - 'datasets/backends/postgres/dataset/capital', - 'datasets/backends/postgres/dataset/continent', - 'datasets/backends/postgres/dataset/country', - 'datasets/backends/postgres/dataset/org', - 'datasets/backends/postgres/dataset/report', + 'datasets/backends/postgres/dataset/Capital', + 'datasets/backends/postgres/dataset/Continent', + 'datasets/backends/postgres/dataset/Country', + 'datasets/backends/postgres/dataset/Org', + 'datasets/backends/postgres/dataset/Report', ] +@pytest.mark.manifests('internal_sql', 'csv') def test_ns_titles( + manifest_type: str, + tmp_path: Path, rc: RawConfig, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | title | description | ns | datasets | All datasets | All external datasets. | ns | datasets/gov | Government datasets | All external government datasets. @@ -87,7 +92,11 @@ def test_ns_titles( | | | | name | string | | Country name | Name of a country. | | | City | | | Cities | All cities. | | | | name | string | | City name | Name of a city. - ''') + ''', + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) app = create_test_client(context, scope=['spinta_getall']) assert listdata(app.get('/:ns'), 'title', 'description') == [ ("All datasets", "All external datasets."), @@ -102,10 +111,14 @@ def test_ns_titles( ] +@pytest.mark.manifests('internal_sql', 'csv') def test_ns_titles_bare_models( + manifest_type: str, + tmp_path: Path, rc: RawConfig, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref | title | description | ns | datasets | All datasets | All external datasets. | | datasets/gov | Government datasets | All external government datasets. @@ -115,7 +128,11 @@ def test_ns_titles_bare_models( | | | | name | string | | Country name | Name of a country. | | | datasets/gov/vpt/new/City | | | Cities | All cities. | | | | name | string | | City name | Name of a city. - ''') + ''', + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True + ) app = create_test_client(context, scope=['spinta_getall']) assert listdata(app.get('/:ns'), 'title', 'description') == [ ("All datasets", "All external datasets."), @@ -130,8 +147,13 @@ def test_ns_titles_bare_models( ] -def test_sort_models_by_refs(rc: RawConfig): - manifest = load_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_sort_models_by_refs( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access datasets/gov/example | | | | | | | | @@ -146,9 +168,9 @@ def test_sort_models_by_refs(rc: RawConfig): | | | City | | | | | | | name | string | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) - models = sort_models_by_refs(manifest.models.values()) + models = sort_models_by_refs(commands.get_models(context, manifest).values()) names = [model.name for model in models] assert names == [ 'datasets/gov/example/City', @@ -157,8 +179,13 @@ def test_sort_models_by_refs(rc: RawConfig): ] -def test_sort_models_by_ref_with_base(rc: RawConfig): - manifest = load_manifest(rc, ''' +@pytest.mark.manifests('internal_sql', 'csv') +def test_sort_models_by_ref_with_base( + manifest_type: str, + tmp_path: Path, + rc: RawConfig, +): + context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | ref | access datasets/basetest | | | | | | Place | | id | @@ -175,9 +202,9 @@ def test_sort_models_by_ref_with_base(rc: RawConfig): | | | | id | integer | | open | | | | name | | | open | | | | country | ref | Country | open - ''') + ''', manifest_type=manifest_type, tmp_path=tmp_path) - models = sort_models_by_refs(manifest.models.values()) + models = sort_models_by_refs(commands.get_models(context, manifest).values()) names = [model.name for model in models] assert names == [ 'datasets/basetest/City', diff --git a/tests/test_nestedprops.py b/tests/test_nestedprops.py index 1db9f8268..3385221b7 100644 --- a/tests/test_nestedprops.py +++ b/tests/test_nestedprops.py @@ -2,8 +2,8 @@ @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_update_object(model, app): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -15,7 +15,7 @@ def test_update_object(model, app): 'sync_resources': [ { 'sync_id': '2', - 'sync_source': 'report' + 'sync_source': 'Report' } ] } @@ -40,15 +40,15 @@ def test_update_object(model, app): 'sync_resources': [ { 'sync_id': '2', - 'sync_source': 'report' + 'sync_source': 'Report' } ] } @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_update_object_array(model, app): app.authmodel(model, ['insert', 'patch', 'getone']) @@ -60,7 +60,7 @@ def test_update_object_array(model, app): 'sync_resources': [ { 'sync_id': '2', - 'sync_source': 'report' + 'sync_source': 'Report' } ] } diff --git a/tests/test_pii.py b/tests/test_pii.py index d95646b6c..8f68d9868 100644 --- a/tests/test_pii.py +++ b/tests/test_pii.py @@ -5,7 +5,7 @@ from spinta.testing.manifest import load_manifest -def test_detect_pii(rc, cli: SpintaCliRunner, tmp_path, sqlite): +def test_detect_pii(context, rc, cli: SpintaCliRunner, tmp_path, sqlite): # Prepare source data. sqlite.init({ 'PERSON': [ @@ -38,7 +38,7 @@ def test_detect_pii(rc, cli: SpintaCliRunner, tmp_path, sqlite): ]) # Configure Spinta. - rc = configure(rc, sqlite, tmp_path / 'manifest.csv', ''' + rc = configure(context, rc, sqlite, tmp_path / 'manifest.csv', ''' d | r | m | property | type | ref | source | access datasets/ds | | | | | rs | sql | sql | | diff --git a/tests/test_push.py b/tests/test_push.py index 001e328de..6f2de4955 100644 --- a/tests/test_push.py +++ b/tests/test_push.py @@ -16,6 +16,7 @@ from responses import POST from responses import RequestsMock +from spinta import commands from spinta.cli.helpers.errors import ErrorCounter from spinta.cli.push import _PushRow, _reset_pushed from spinta.cli.push import _get_row_for_error @@ -150,14 +151,14 @@ def test_push_different_models(app): def test__map_sent_and_recv__no_recv(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | access datasets/gov/example | | | | | Country | | | | | | name | string | open ''') - model = manifest.models['datasets/gov/example/Country'] + model = commands.get_model(context, manifest, 'datasets/gov/example/Country') sent = [ _PushRow(model, {'name': 'Vilnius'}), ] @@ -166,14 +167,14 @@ def test__map_sent_and_recv__no_recv(rc: RawConfig): def test__get_row_for_error__errors(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | access datasets/gov/example | | | | | Country | | | | | | name | string | open ''') - model = manifest.models['datasets/gov/example/Country'] + model = commands.get_model(context, manifest, 'datasets/gov/example/Country') rows = [ _PushRow(model, { '_id': '4d741843-4e94-4890-81d9-5af7c5b5989a', @@ -251,7 +252,7 @@ def test_push_state__create(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -310,7 +311,7 @@ def test_push_state__create_error(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -352,7 +353,7 @@ def test_push_state__update(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -425,7 +426,7 @@ def test_push_state__update_without_sync(rc: RawConfig, responses: RequestsMock) | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -493,7 +494,7 @@ def test_push_state__update_sync_first_time(rc: RawConfig, responses: RequestsMo | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -560,7 +561,7 @@ def test_push_state__update_sync(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -625,7 +626,7 @@ def test_push_state__update_error(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -674,6 +675,7 @@ def test_push_state__update_error(rc: RawConfig, responses: RequestsMock): def test_push_delete_with_dependent_objects( + context, postgresql, rc, cli: SpintaCliRunner, @@ -695,7 +697,7 @@ def test_push_delete_with_dependent_objects( | | | | name | string | | pavadinimas| | | | | country | ref | Country | salis | ''' - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(table)) + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(table)) localrc = create_rc(rc, tmp_path, geodb) @@ -751,7 +753,7 @@ def test_push_state__delete(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -825,7 +827,7 @@ def test_push_state__retry(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -892,7 +894,7 @@ def test_push_state__max_errors(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) @@ -972,7 +974,7 @@ def test_push_init_state(rc: RawConfig, sqlite: Sqlite): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] sqlite.init({ @@ -1022,7 +1024,7 @@ def test_push_state__paginate(rc: RawConfig, responses: RequestsMock): | name | string | open ''') - model = manifest.models['City'] + model = commands.get_model(context, manifest, 'City') models = [model] state = _State(*_init_push_state('sqlite://', models)) diff --git a/tests/test_search.py b/tests/test_search.py index 7e5f7833c..477e9fbdd 100644 --- a/tests/test_search.py +++ b/tests/test_search.py @@ -18,7 +18,7 @@ test_data = [ { - '_type': 'report', + '_type': 'Report', '_id': '1845aa15-c8e0-4368-a325-c31c6ac3bfac', 'status': 'OK', 'report_type': 'STV', @@ -33,7 +33,7 @@ }], }, { - '_type': 'report', + '_type': 'Report', '_id': '2dc605ba-f4ff-4240-9db6-51e1a7ad7d28', 'status': 'invalid', 'report_type': 'VMI', @@ -48,7 +48,7 @@ }], }, { - '_type': 'report', + '_type': 'Report', '_id': '34e9f68d-7291-4b63-ad61-0480ae3941c4', 'status': 'invalid', 'report_type': 'STV', @@ -80,8 +80,8 @@ def _push_test_data(app, model, data=None): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_exact(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -96,8 +96,8 @@ def test_search_exact(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_exact_lower(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -109,8 +109,8 @@ def test_search_exact_lower(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_exact_non_string(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -141,8 +141,8 @@ def test_search_exact_non_string(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_exact_multiple_props(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -154,8 +154,8 @@ def test_search_exact_multiple_props(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_exact_same_prop_multiple_times(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -166,8 +166,8 @@ def test_search_exact_same_prop_multiple_times(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_gt(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -201,8 +201,8 @@ def test_search_gt(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_gt_with_nested_date(model, context, app): ids = RowIds(_push_test_data(app, model)) @@ -212,8 +212,8 @@ def test_search_gt_with_nested_date(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_gte(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -248,8 +248,8 @@ def test_search_gte(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ge_with_nested_date(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -261,8 +261,8 @@ def test_search_ge_with_nested_date(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_lt(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -296,8 +296,8 @@ def test_search_lt(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_lt_with_nested_date(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -309,8 +309,8 @@ def test_search_lt_with_nested_date(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_lte(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -345,8 +345,8 @@ def test_search_lte(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_le_with_nested_date(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -358,8 +358,8 @@ def test_search_le_with_nested_date(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne(model, context, app): app.authmodel(model, ['search']) @@ -371,8 +371,8 @@ def test_search_ne(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne_lower(model, context, app): app.authmodel(model, ['search']) @@ -383,8 +383,8 @@ def test_search_ne_lower(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne_multiple_props(model, context, app): app.authmodel(model, ['search']) @@ -396,8 +396,8 @@ def test_search_ne_multiple_props(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne_multiple_props_and_logic(model, context, app): app.authmodel(model, ['search']) @@ -409,8 +409,8 @@ def test_search_ne_multiple_props_and_logic(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne_nested(model, context, app): app.authmodel(model, ['search']) @@ -421,8 +421,8 @@ def test_search_ne_nested(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_ne_nested_missing_data(model, context, app): app.authmodel(model, ['search']) @@ -433,8 +433,8 @@ def test_search_ne_nested_missing_data(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_contains(model, context, app, mocker): r1, r2, r3, = _push_test_data(app, model) @@ -449,8 +449,8 @@ def test_search_contains(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_contains_case_insensitive(model, context, app, mocker): r1, r2, r3, = _push_test_data(app, model) @@ -463,8 +463,8 @@ def test_search_contains_case_insensitive(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_contains_multi_field(model, context, app, mocker): r1, r2, r3, = _push_test_data(app, model) @@ -500,8 +500,8 @@ def test_search_contains_multi_field(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_contains_type_check(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -512,8 +512,8 @@ def test_search_contains_type_check(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_contains_with_select(model, context, app, mocker): r1, r2, r3, = _push_test_data(app, model) @@ -550,8 +550,8 @@ def test_search_contains_with_select(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_select_unknown_property(model, context, app, mocker): _push_test_data(app, model) @@ -561,8 +561,8 @@ def test_select_unknown_property(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_select_unknown_property_in_object(model, context, app, mocker): _push_test_data(app, model) @@ -572,8 +572,8 @@ def test_select_unknown_property_in_object(model, context, app, mocker): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_startswith(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -618,8 +618,8 @@ def test_search_startswith(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_nested(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -663,8 +663,8 @@ def test_search_nested(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_nested_contains(model, context, app): app.authmodel(model, ['search']) @@ -674,8 +674,8 @@ def test_search_nested_contains(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_nested_startswith(model, context, app): app.authmodel(model, ['search']) @@ -702,8 +702,8 @@ def ids(resources): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_or(model, context, app): ids = RowIds(_push_test_data(app, model)) @@ -716,8 +716,8 @@ def test_or(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_nested_recurse(model, context, app): r1, r2, r3, = _push_test_data(app, model) @@ -729,8 +729,8 @@ def test_search_nested_recurse(model, context, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_nested_recurse_lower(model, context, app): r1, r2, r3, = ids(_push_test_data(app, model)) @@ -740,8 +740,8 @@ def test_search_nested_recurse_lower(model, context, app): @pytest.mark.models( - 'backends/mongo/recurse', - 'backends/postgres/recurse', + 'backends/mongo/Recurse', + 'backends/postgres/Recurse', ) def test_search_nested_recurse_multiple_props(model, context, app): r1, r2, = ids(_push_test_data(app, model, [ @@ -774,8 +774,8 @@ def test_search_nested_recurse_multiple_props(model, context, app): @pytest.mark.models( - 'backends/mongo/recurse', - 'backends/postgres/recurse', + 'backends/mongo/Recurse', + 'backends/postgres/Recurse', ) def test_search_recurse_multiple_props_lower(model, app): r1, r2, = ids(_push_test_data(app, model, [ @@ -809,7 +809,7 @@ def test_search_recurse_multiple_props_lower(model, app): # TODO: add mongo def test_search_any(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("eq",count,10,42)') @@ -821,7 +821,7 @@ def test_search_any(app): # TODO: add mongo def test_search_any_in_list(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("eq",notes.note,"hello","world")') @@ -833,7 +833,7 @@ def test_search_any_in_list(app): # TODO: add mongo def test_search_any_in_list_of_scalars(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("eq",operating_licenses.license_types,"valid","invalid","expired")') @@ -845,7 +845,7 @@ def test_search_any_in_list_of_scalars(app): # TODO: add mongo def test_search_any_recurse(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("eq",recurse(status),"OK","none")') @@ -854,7 +854,7 @@ def test_search_any_recurse(app): # TODO: add mongo def test_search_any_recurse_lower(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("eq",recurse(status).lower(),"ok","none")') @@ -863,7 +863,7 @@ def test_search_any_recurse_lower(app): # TODO: add mongo def test_search_any_contains(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("contains",status,"inv","val","lid")') @@ -872,7 +872,7 @@ def test_search_any_contains(app): # TODO: add mongo def test_search_any_contains_nested(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("contains",notes.note,"hel","wor")') @@ -881,7 +881,7 @@ def test_search_any_contains_nested(app): # TODO: add mongo def test_search_any_contains_recurse_lower(app): - model = 'backends/postgres/report' + model = 'backends/postgres/Report' app.authmodel(model, ['search']) ids = RowIds(_push_test_data(app, model)) resp = app.get(f'/{model}?any("contains",recurse(status).lower(),"o","k")') @@ -889,8 +889,8 @@ def test_search_any_contains_recurse_lower(app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_id_contains(model, app): app.authmodel(model, ['search', 'getall']) @@ -904,8 +904,8 @@ def test_search_id_contains(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_id_not_contains(model, app): app.authmodel(model, ['search', 'getall']) @@ -915,8 +915,8 @@ def test_search_id_not_contains(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_id_startswith(model, app): app.authmodel(model, ['search']) @@ -927,8 +927,8 @@ def test_search_id_startswith(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_id_not_startswith(model, app): app.authmodel(model, ['search']) @@ -939,8 +939,8 @@ def test_search_id_not_startswith(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_revision_contains(model, app): app.authmodel(model, ['search']) @@ -950,8 +950,8 @@ def test_search_revision_contains(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_revision_startswith(model, app): app.authmodel(model, ['search', 'getone']) @@ -964,8 +964,8 @@ def test_search_revision_startswith(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_group(model, app): app.authmodel(model, ['search', 'getone']) @@ -975,8 +975,8 @@ def test_search_group(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_select_in_or(model, app): app.authmodel(model, ['search', 'getone']) @@ -987,8 +987,8 @@ def test_search_select_in_or(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_lower_contains(model, app): app.authmodel(model, ['search', 'getone']) @@ -999,8 +999,8 @@ def test_search_lower_contains(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_null(model, app): app.authmodel(model, ['search']) @@ -1013,8 +1013,8 @@ def test_search_null(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_search_not_null(model, app): app.authmodel(model, ['search']) @@ -1027,7 +1027,7 @@ def test_search_not_null(model, app): @pytest.mark.parametrize('backend', ['default', 'mongo']) -def test_extra_fields(postgresql, mongo, backend, rc, tmp_path, request): +def test_extra_fields(context, postgresql, mongo, backend, rc, tmp_path, request): rc = rc.fork({ 'backends': [backend], 'manifests.default': { @@ -1038,9 +1038,9 @@ def test_extra_fields(postgresql, mongo, backend, rc, tmp_path, request): }) # Create data into a extrafields model with code and name properties. - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' m | property | type - extrafields | + Extrafields | | code | string | name | string ''')) @@ -1048,8 +1048,8 @@ def test_extra_fields(postgresql, mongo, backend, rc, tmp_path, request): request.addfinalizer(context.wipe_all) app = create_test_client(context) app.authorize(['spinta_set_meta_fields']) - app.authmodel('extrafields', ['insert']) - resp = app.post('/extrafields', json={'_data': [ + app.authmodel('Extrafields', ['insert']) + resp = app.post('/Extrafields', json={'_data': [ {'_op': 'insert', 'code': 'lt', 'name': 'Lietuva', '_id': '00600aa5-1629-4ead-a4fc-736a2e64f5ce'}, {'_op': 'insert', 'code': 'lv', 'name': 'Latvija', '_id': '0411734b-8658-40c5-bf32-9ade37cbb401'}, {'_op': 'insert', 'code': 'ee', 'name': 'Estija', '_id': '0a3afc49-263c-40a0-a278-b58514ed961a'}, @@ -1057,15 +1057,15 @@ def test_extra_fields(postgresql, mongo, backend, rc, tmp_path, request): assert resp.status_code == 200, resp.json() # Now try to read from same model, but loaded with just one property. - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' m | property | type - extrafields | + Extrafields | | name | string ''')) context = create_test_context(rc) app = create_test_client(context) - app.authmodel('extrafields', ['getall', 'getone']) - resp = app.get('/extrafields') + app.authmodel('Extrafields', ['getall', 'getone']) + resp = app.get('/Extrafields') assert listdata(resp, sort=True) == [ "Estija", "Latvija", @@ -1073,14 +1073,14 @@ def test_extra_fields(postgresql, mongo, backend, rc, tmp_path, request): ] pk = resp.json()['_data'][0]['_id'] - resp = app.get(f'/extrafields/{pk}') + resp = app.get(f'/Extrafields/{pk}') data = resp.json() assert resp.status_code == 200, data assert take(data) == {'name': 'Lietuva'} @pytest.mark.parametrize('backend', ['mongo']) -def test_missing_fields(postgresql, mongo, backend, rc, tmp_path): +def test_missing_fields(context, postgresql, mongo, backend, rc, tmp_path): rc = rc.fork({ 'backends': [backend], 'manifests.default': { @@ -1091,15 +1091,15 @@ def test_missing_fields(postgresql, mongo, backend, rc, tmp_path): }) # Create data into a extrafields model with code and name properties. - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' m | property | type - missingfields | + Missingfields | | code | string ''')) context = create_test_context(rc) app = create_test_client(context) - app.authmodel('missingfields', ['insert']) - resp = app.post('/missingfields', json={'_data': [ + app.authmodel('Missingfields', ['insert']) + resp = app.post('/Missingfields', json={'_data': [ {'_op': 'insert', 'code': 'lt'}, {'_op': 'insert', 'code': 'lv'}, {'_op': 'insert', 'code': 'ee'}, @@ -1107,16 +1107,16 @@ def test_missing_fields(postgresql, mongo, backend, rc, tmp_path): assert resp.status_code == 200, resp.json() # Now try to read from same model, but loaded with just one property. - create_tabular_manifest(tmp_path / 'manifest.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'manifest.csv', striptable(''' m | property | type - missingfields | + Missingfields | | code | string | name | string ''')) context = create_test_context(rc) app = create_test_client(context) - app.authmodel('missingfields', ['search', 'getone']) - resp = app.get('/missingfields?select(_id,code,name)') + app.authmodel('Missingfields', ['search', 'getone']) + resp = app.get('/Missingfields?select(_id,code,name)') assert listdata(resp, sort=True) == [ ('ee', None), ('lt', None), @@ -1124,7 +1124,7 @@ def test_missing_fields(postgresql, mongo, backend, rc, tmp_path): ] pk = resp.json()['_data'][0]['_id'] - resp = app.get(f'/missingfields/{pk}') + resp = app.get(f'/Missingfields/{pk}') data = resp.json() assert resp.status_code == 200, data assert take(data) == {'code': 'lt'} @@ -1177,8 +1177,8 @@ def test_base_select(rc, postgresql, request): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_select_revision(model, app): app.authmodel(model, ['search', 'getone', 'getall']) diff --git a/tests/test_sort.py b/tests/test_sort.py index 47072394c..9c71a2173 100644 --- a/tests/test_sort.py +++ b/tests/test_sort.py @@ -2,8 +2,8 @@ @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_sort(model, app): app.authmodel(model, ['insert', 'search']) @@ -31,8 +31,8 @@ def test_sort(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_sort_with_nested_prop(model, app): app.authmodel(model, ['insert', 'search']) diff --git a/tests/test_spyna.py b/tests/test_spyna.py index 0bc4e5c8d..578f3d9da 100644 --- a/tests/test_spyna.py +++ b/tests/test_spyna.py @@ -182,3 +182,7 @@ def test_escale_quotes_one(): def test_escale_quotes_both(): check("swap('\\\'TEST\\\' \"TEST\"', '\"TEST\" \\\'TEST\\\'')") + + +def test_normal_text(): + check("'TEST'") diff --git a/tests/test_store.py b/tests/test_store.py index 76ac97f18..1f4b6718b 100644 --- a/tests/test_store.py +++ b/tests/test_store.py @@ -17,8 +17,8 @@ 'backends/postgres/{}', ) def test_schema_loader(model, app): - model_org = model.format('org') - model_country = model.format('country') + model_org = model.format('Org') + model_country = model.format('Country') app.authmodel(model_org, ['insert']) app.authmodel(model_country, ['insert']) @@ -78,8 +78,8 @@ def test_schema_loader(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_nested(model, app): app.authmodel(model, ['insert', 'getone']) @@ -104,8 +104,8 @@ def test_nested(model, app): }] -def test_root(rc: RawConfig, tmp_path: Path): - rc = configure(rc, None, tmp_path / 'manifest.csv', ''' +def test_root(context, rc: RawConfig, tmp_path: Path): + rc = configure(context, rc, None, tmp_path / 'manifest.csv', ''' d | r | b | m | property | type | title datasets/gov/vpt/old | | Old data | sql | sql | @@ -127,6 +127,7 @@ def test_root(rc: RawConfig, tmp_path: Path): def test_resource_backends( + context, rc: RawConfig, tmp_path: Path, sqlite: Sqlite, @@ -145,7 +146,7 @@ def test_resource_backends( {'ID': 2, 'NAME': 'Latvia'}, ]) - create_tabular_manifest(tmp_path / 'old.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'old.csv', striptable(''' d | r | b | m | property | type | ref | source | access | title datasets/gov/vpt/old | | | | | Old data | sql | | old | | | @@ -154,7 +155,7 @@ def test_resource_backends( | | | | name | string | | NAME | open | ''')) - create_tabular_manifest(tmp_path / 'new.csv', striptable(''' + create_tabular_manifest(context, tmp_path / 'new.csv', striptable(''' d | r | b | m | property | type | ref | source | access | title datasets/gov/vpt/new | | | | | New data | sql | | new | | | diff --git a/tests/test_subresources.py b/tests/test_subresources.py index a7ae1ceb9..2ca8f1074 100644 --- a/tests/test_subresources.py +++ b/tests/test_subresources.py @@ -6,8 +6,8 @@ @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_get_subresource(model, app): app.authmodel(model, ['insert', 'getone', @@ -71,8 +71,8 @@ def test_get_subresource(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_put_subresource(model, app): app.authmodel(model, [ @@ -168,8 +168,8 @@ def test_put_subresource(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_patch_subresource(model, app): app.authmodel(model, [ @@ -285,8 +285,8 @@ def test_patch_subresource(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_subresource_scopes(model, app): app.authmodel(model, ['insert', 'hidden_subobj_update']) @@ -368,8 +368,8 @@ def test_subresource_scopes(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_get_subresource_file(model, app, tmp_path): app.authmodel(model, ['insert', 'getone', 'hidden_subobj_update', @@ -417,8 +417,8 @@ def test_get_subresource_file(model, app, tmp_path): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_put_hidden_subresource_on_model(model, app): app.authmodel(model, ['insert', 'getone', 'update', 'hidden_subobj_update']) @@ -450,8 +450,8 @@ def test_put_hidden_subresource_on_model(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_patch_hidden_subresource_on_model(model, app): app.authmodel(model, ['insert', 'getone', 'patch', 'hidden_subobj_update']) @@ -483,8 +483,8 @@ def test_patch_hidden_subresource_on_model(model, app): @pytest.mark.models( - 'backends/mongo/subitem', - 'backends/postgres/subitem', + 'backends/mongo/Subitem', + 'backends/postgres/Subitem', ) def test_hidden_subresource_after_put(model, app): # tests that hidden subresource is not changed when @@ -532,14 +532,14 @@ def test_hidden_subresource_after_put(model, app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_file_without_content_type(model, app): app.authmodel(model, ['insert', 'pdf_update', 'pdf_getone']) # Create a new report resource. - resp = app.post(f'/{model}s', json={ + resp = app.post(f'/{model}', json={ '_type': model, 'report_type': 'pdf', }) diff --git a/tests/test_ufuncs.py b/tests/test_ufuncs.py index 7e83eb91d..1e1a4a553 100644 --- a/tests/test_ufuncs.py +++ b/tests/test_ufuncs.py @@ -2,6 +2,7 @@ import pytest +from spinta import commands from spinta.components import Store from spinta.core.config import RawConfig from spinta.core.ufuncs import Bind @@ -10,7 +11,7 @@ from spinta.core.ufuncs import Pair from spinta.core.ufuncs import UFuncRegistry from spinta.exceptions import IncompatibleForeignProperties -from spinta.testing.manifest import load_manifest +from spinta.testing.manifest import load_manifest, load_manifest_and_context from spinta.testing.manifest import load_manifest_get_context from spinta.testing.ufuncs import UFuncTester from spinta.types.datatype import Ref @@ -153,7 +154,7 @@ def getattr(env, item, bind): # noqa def test_fpr_get_bind_expr(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | m | property | type | ref datasets/gov/example | | | resource | sql | @@ -173,10 +174,10 @@ def test_fpr_get_bind_expr(rc: RawConfig): | | | country | ref | Country ''') - planet = manifest.models['datasets/gov/example/Planet'] - continent = manifest.models['datasets/gov/example/Continent'] - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + planet = commands.get_model(context, manifest, 'datasets/gov/example/Planet') + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr = ForeignProperty( None, @@ -199,7 +200,7 @@ def test_fpr_get_bind_expr(rc: RawConfig): def test_fpr_join(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | m | property | type | ref datasets/gov/example | | | resource | sql | @@ -215,9 +216,9 @@ def test_fpr_join(rc: RawConfig): | | | country | ref | Country ''') - continent = manifest.models['datasets/gov/example/Continent'] - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr1 = ForeignProperty( None, @@ -238,7 +239,7 @@ def test_fpr_join(rc: RawConfig): def test_fpr_join_no_right(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | m | property | type | ref datasets/gov/example | | | resource | sql | @@ -254,9 +255,9 @@ def test_fpr_join_no_right(rc: RawConfig): | | | country | ref | Country ''') - continent = manifest.models['datasets/gov/example/Continent'] - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr1 = ForeignProperty( None, @@ -276,7 +277,7 @@ def test_fpr_join_no_right(rc: RawConfig): def test_fpr_join_incompatible_refs(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | m | property | type | ref datasets/gov/example | | | resource | sql | @@ -296,9 +297,9 @@ def test_fpr_join_incompatible_refs(rc: RawConfig): | | | country | ref | Country ''') - continent = manifest.models['datasets/gov/example/Continent'] - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr1 = ForeignProperty( None, @@ -315,7 +316,7 @@ def test_fpr_join_incompatible_refs(rc: RawConfig): def test_fpr_join_incompatible_refs_no_right(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | m | property | type | ref datasets/gov/example | | | resource | sql | @@ -335,8 +336,8 @@ def test_fpr_join_incompatible_refs_no_right(rc: RawConfig): | | | country | ref | Country ''') - continent = manifest.models['datasets/gov/example/Continent'] - city = manifest.models['datasets/gov/example/City'] + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr1 = ForeignProperty( None, @@ -371,8 +372,8 @@ def test_change_base_model(rc: RawConfig): store: Store = context.get('store') manifest = store.manifest - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr = ForeignProperty(None, cast(Ref, city.properties['country'].dtype)) assert str(change_base_model(context, country, fpr)) == ( @@ -400,9 +401,9 @@ def test_change_base_model_non_ref(rc: RawConfig): store: Store = context.get('store') manifest = store.manifest - continent = manifest.models['datasets/gov/example/Continent'] - country = manifest.models['datasets/gov/example/Country'] - city = manifest.models['datasets/gov/example/City'] + continent = commands.get_model(context, manifest, 'datasets/gov/example/Continent') + country = commands.get_model(context, manifest, 'datasets/gov/example/Country') + city = commands.get_model(context, manifest, 'datasets/gov/example/City') fpr = ForeignProperty(None, cast(Ref, city.properties['country'].dtype)) fpr = fpr.push(country.properties['continent']) diff --git a/tests/test_validation.py b/tests/test_validation.py index c7e41eee4..aeef3242c 100644 --- a/tests/test_validation.py +++ b/tests/test_validation.py @@ -4,8 +4,8 @@ @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_report(model, app): app.authmodel(model, ['insert', 'getone']) @@ -55,8 +55,8 @@ def test_report(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_report_int(model, app): app.authmodel(model, ['insert']) @@ -86,8 +86,8 @@ def test_invalid_report_int(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_report_date(model, app): app.authmodel(model, ['insert']) @@ -104,14 +104,14 @@ def test_invalid_report_date(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_non_string_report_date(model, app): app.authmodel(model, ['insert']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'report_type': 'simple', 'status': 'valid', 'valid_from_date': 42, # invalid conversion to date @@ -122,14 +122,14 @@ def test_non_string_report_date(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_report_datetime(model, app): app.authmodel(model, ['insert']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'report_type': 'simple', 'status': 'valid', 'update_time': '2019-04', # invalid conversion to datetime @@ -140,14 +140,14 @@ def test_invalid_report_datetime(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_non_string_report_datetime(model, app): app.authmodel(model, ['insert']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'report_type': 'simple', 'status': 'valid', 'update_time': 42, # invalid conversion to datetime @@ -158,14 +158,14 @@ def test_non_string_report_datetime(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_report_array(model, app): app.authmodel(model, ['insert']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'report_type': 'simple', 'status': 'valid', 'notes': {'foo': 'bar'}, # invalid conversion to array @@ -176,14 +176,14 @@ def test_invalid_report_array(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_report_array_object(model, app): app.authmodel(model, ['insert']) resp = app.post(f'/{model}', json={ - '_type': 'report', + '_type': 'Report', 'report_type': 'simple', 'status': 'valid', 'notes': ['hello', 'world'], # invalid array item type @@ -194,8 +194,8 @@ def test_invalid_report_array_object(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_invalid_nested_object_property(model, app): app.authmodel(model, ['insert']) @@ -211,8 +211,8 @@ def test_invalid_nested_object_property(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_missing_report_object_property(model, app): app.authmodel(model, ['insert', 'getone']) @@ -233,8 +233,8 @@ def test_missing_report_object_property(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_unknown_report_property(model, app): app.authmodel(model, ['insert', 'getone']) @@ -267,8 +267,8 @@ def test_unknown_report_property(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_unknown_report_object_property(model, app): app.authmodel(model, ['insert', 'getone']) @@ -299,8 +299,8 @@ def test_unknown_report_object_property(model, app): @pytest.mark.models( - 'backends/postgres/report', - 'backends/mongo/report', + 'backends/postgres/Report', + 'backends/mongo/Report', ) def test_wrong_revision_with_subresource(model, app): app.authmodel(model, ['insert', 'update']) diff --git a/tests/test_wipe.py b/tests/test_wipe.py index cdee2d0cb..3f607e3a5 100644 --- a/tests/test_wipe.py +++ b/tests/test_wipe.py @@ -1,3 +1,4 @@ +from pathlib import Path from typing import List from typing import Tuple @@ -29,9 +30,9 @@ def test_wipe_all(app): # Create some data in different models resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, ]}) assert resp.status_code == 200, resp.json() @@ -39,10 +40,10 @@ def test_wipe_all(app): resp = app.get('/:all') data = sorted([(r['_type'], r.get('status')) for r in resp.json()['_data']]) assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] # Wipe all data @@ -56,27 +57,27 @@ def test_wipe_all(app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_model(model, app): app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe']) # Create some data in different models resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, ]}) assert resp.status_code == 200, resp.json() # Get data from all models resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] # Wipe model data @@ -88,42 +89,42 @@ def test_wipe_model(model, app): # Check the data again resp = app.get('/:all') assert listdata(resp, '_type', 'status') == _excluding(model, 'ok', [ - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('Report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ]) @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_row(model: str, app: TestClient): app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe']) # Create some data in different models resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'nb'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'nb'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'nb'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'nb'}, ]}) _id_idx = { - 'backends/mongo/report': 1, - 'backends/postgres/report': 3, + 'backends/mongo/Report': 1, + 'backends/postgres/Report': 3, } _id = listdata(resp, '_id')[_id_idx[model]] # Get data from all models resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'nb'), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'nb'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'nb'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'nb'), + ('backends/postgres/Report', 'ok'), ] # Wipe model row data @@ -136,17 +137,17 @@ def test_wipe_row(model: str, app: TestClient): # Check the data again. resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ - ('backends/mongo/report', 'nb'), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'nb'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('Report', 'ok'), + ('backends/mongo/Report', 'nb'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'nb'), + ('backends/postgres/Report', 'ok'), ] @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_check_scope(model, app): app.authorize(['spinta_insert', 'spinta_getall', 'spinta_delete']) @@ -161,8 +162,8 @@ def test_wipe_check_ns_scope(app): @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_in_batch(model, app): app.authorize(['spinta_wipe']) @@ -180,19 +181,19 @@ def test_wipe_all_access(app: TestClient): # Create some data in different models. resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, ]}) assert resp.status_code == 200, resp.json() # Get data from all models. resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] # Wipe all data @@ -202,35 +203,35 @@ def test_wipe_all_access(app: TestClient): # Check the data again. resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_model_access(model, app): app.authorize(['spinta_insert', 'spinta_getall', 'spinta_delete']) # Create some data in different models resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, ]}) assert resp.status_code == 200, resp.json() # Get data from all models resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] # Wipe model data @@ -240,25 +241,25 @@ def test_wipe_model_access(model, app): # Check what data again resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] @pytest.mark.models( - 'backends/mongo/report', - 'backends/postgres/report', + 'backends/mongo/Report', + 'backends/postgres/Report', ) def test_wipe_row_access(model, app): app.authorize(['spinta_insert', 'spinta_getall', 'spinta_delete']) # Create some data in different models resp = app.post('/', json={'_data': [ - {'_op': 'insert', '_type': 'report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/mongo/report', 'status': 'ok'}, - {'_op': 'insert', '_type': 'backends/postgres/report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/mongo/Report', 'status': 'ok'}, + {'_op': 'insert', '_type': 'backends/postgres/Report', 'status': 'ok'}, ]}) ids = dict(listdata(resp, '_type', '_id')) _id = ids[model] @@ -266,10 +267,10 @@ def test_wipe_row_access(model, app): # Get data from all models resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] # Wipe model row data @@ -280,50 +281,60 @@ def test_wipe_row_access(model, app): # Check what data again resp = app.get('/:all') assert listdata(resp, '_type', 'status') == [ + ('Report', 'ok'), ('_txn', NA), - ('backends/mongo/report', 'ok'), - ('backends/postgres/report', 'ok'), - ('report', 'ok'), + ('backends/mongo/Report', 'ok'), + ('backends/postgres/Report', 'ok'), ] +@pytest.mark.manifests('internal_sql', 'csv') def test_wipe_with_long_names( + manifest_type: str, + tmp_path: Path, rc: RawConfig, postgresql: str, request: FixtureRequest, ): - context = bootstrap_manifest(rc, ''' + context = bootstrap_manifest( + rc, ''' d | r | b | m | property | type | ref backends/postgres/very/long/name/models | | | | | ModelWithVeryVeryVeryLongName| | | | | | status | string | - ''', backend=postgresql, request=request) - - app = create_test_client(context) - app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe']) - - # Create some data - resp = app.post('/', json={'_data': [ - { - '_op': 'insert', - '_type': 'backends/postgres/very/long/name/models/ModelWithVeryVeryVeryLongName', - 'status': 'ok' - }, - ]}) - assert resp.status_code == 200, resp.json() - - # Get data from all models - resp = app.get('/:all') - assert listdata(resp, '_type', 'status') == [ - ('_txn', NA), - ('backends/postgres/very/long/name/models/ModelWithVeryVeryVeryLongName', 'ok') - ] + ''', + backend=postgresql, + tmp_path=tmp_path, + manifest_type=manifest_type, + full_load=True, + request=request + ) + with context: + app = create_test_client(context) + app.authorize(['spinta_insert', 'spinta_getall', 'spinta_wipe']) + + # Create some data + resp = app.post('/', json={'_data': [ + { + '_op': 'insert', + '_type': 'backends/postgres/very/long/name/models/ModelWithVeryVeryVeryLongName', + 'status': 'ok' + }, + ]}) + assert resp.status_code == 200, resp.json() + + # Get data from all models + resp = app.get('/:all') + assert listdata(resp, '_type', 'status') == [ + ('_txn', NA), + ('backends/postgres/very/long/name/models/ModelWithVeryVeryVeryLongName', 'ok') + ] - # Wipe all data - resp = app.delete('/:wipe') - assert resp.status_code == 200, resp.json() + # Wipe all data + resp = app.delete('/:wipe') + assert resp.status_code == 200, resp.json() - # Check what data again - resp = app.get('/:all') - assert resp.status_code == 200, resp.json() - assert len(resp.json()['_data']) == 0 + # Check what data again + resp = app.get('/:all') + assert resp.status_code == 200, resp.json() + assert len(resp.json()['_data']) == 0 diff --git a/tests/testing/test_dtypes.py b/tests/testing/test_dtypes.py index 714e26ca5..1cd1b4bdd 100644 --- a/tests/testing/test_dtypes.py +++ b/tests/testing/test_dtypes.py @@ -2,13 +2,13 @@ def test_path(): - assert path('backends/mongo/dtypes/string') == 'string' - assert path('backends/postgres/dtypes/object/string') == 'object.string' + assert path('backends/mongo/dtypes/String') == 'string' + assert path('backends/postgres/dtypes/object/String') == 'object.string' def test_path_array(): - assert path('backends/postgres/dtypes/array/string') == 'array' - assert path('backends/postgres/dtypes/array/object/string') == 'array.string' + assert path('backends/postgres/dtypes/array/String') == 'array' + assert path('backends/postgres/dtypes/array/object/String') == 'array.string' def test_nest(): @@ -18,13 +18,13 @@ def test_nest(): '_revision': '', 'string': 'test', } - assert nest('backends/mongo/dtypes/string', data) == { + assert nest('backends/mongo/dtypes/String', data) == { '_type': '', '_id': '', '_revision': '', 'string': 'test', } - assert nest('backends/postgres/dtypes/object/string', data) == { + assert nest('backends/postgres/dtypes/object/String', data) == { '_type': '', '_id': '', '_revision': '', @@ -41,7 +41,7 @@ def test_nest_array(): '_revision': '', 'string': 'test', } - assert nest('backends/postgres/dtypes/array/string', data) == { + assert nest('backends/postgres/dtypes/array/String', data) == { '_type': '', '_id': '', '_revision': '', @@ -56,8 +56,8 @@ def test_flat(): '_revision': '', 'string': 'test', } - assert flat('backends/postgres/dtypes/string', data) == data - assert flat('backends/postgres/dtypes/object/string', { + assert flat('backends/postgres/dtypes/String', data) == data + assert flat('backends/postgres/dtypes/object/String', { '_type': '', '_id': '', '_revision': '', @@ -74,7 +74,7 @@ def test_flat_array(): '_revision': '', 'string': 'test', } - assert flat('backends/postgres/dtypes/array/string', { + assert flat('backends/postgres/dtypes/array/String', { '_type': '', '_id': '', '_revision': '', diff --git a/tests/utils/test_errors.py b/tests/utils/test_errors.py index 5f715405f..6aca85911 100644 --- a/tests/utils/test_errors.py +++ b/tests/utils/test_errors.py @@ -1,18 +1,18 @@ -from spinta import exceptions +from spinta import exceptions, commands from spinta.core.config import RawConfig -from spinta.testing.manifest import load_manifest +from spinta.testing.manifest import load_manifest, load_manifest_and_context from spinta.utils.errors import report_error def test_report_error__id(rc: RawConfig): - manifest = load_manifest(rc, ''' + context, manifest = load_manifest_and_context(rc, ''' d | r | b | m | property | type | access datasets/gov/example | | | | | Country | | | | | | name | string | open ''') - model = manifest.models['datasets/gov/example/Country'] + model = commands.get_model(context, manifest, 'datasets/gov/example/Country') prop = model.properties['name'] exc = exceptions.InvalidValue(prop.dtype, value=42)