From 9273d49bb2221914f87817b9cba31c5bf48b920c Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Mon, 4 Mar 2024 11:17:36 +0000 Subject: [PATCH 1/9] Update Github Actions and resolve merge conflicts --- .editorconfig | 10 + .github/workflows/pr.yaml | 34 +++ .gitignore | 5 +- .pre-commit-config.yaml | 4 +- poetry.lock | 402 +++++++++++++++++++++++++++++++++- pyproject.toml | 36 ++- src/neo4j_genai/__init__.py | 4 + src/neo4j_genai/client.py | 15 +- src/neo4j_genai/embeddings.py | 2 +- src/neo4j_genai/types.py | 4 +- tests/conftest.py | 14 ++ tests/test_client.py | 132 +++++++++++ 12 files changed, 630 insertions(+), 32 deletions(-) create mode 100644 .editorconfig create mode 100644 .github/workflows/pr.yaml create mode 100644 tests/conftest.py diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 00000000..41ff5cda --- /dev/null +++ b/.editorconfig @@ -0,0 +1,10 @@ +root = true + +[*] +indent_style = space +indent_size = 4 +insert_final_newline = true +trim_trailing_whitespace = true +end_of_line = lf +charset = utf-8 +max_line_length = 88 diff --git a/.github/workflows/pr.yaml b/.github/workflows/pr.yaml new file mode 100644 index 00000000..16625cf3 --- /dev/null +++ b/.github/workflows/pr.yaml @@ -0,0 +1,34 @@ +name: neo4j_genai PR +on: pull_request + +jobs: + test: + runs-on: ubuntu-latest + steps: + - name: Check out repository code + uses: actions/checkout@v4 + - name: Install Poetry + uses: snok/install-poetry@v1 + with: + virtualenvs-create: true + virtualenvs-in-project: true + installer-parallel: true + - name: Load cached venv + id: cached-poetry-dependencies + uses: actions/cache@v4 + with: + path: .venv + key: venv-${{ runner.os }}-${{ steps.setup-python.outputs.python-version }}-${{ hashFiles('**/poetry.lock') }} + - name: Install dependencies + if: steps.cached-poetry-dependencies.outputs.cache-hit != 'true' + run: poetry install --no-interaction --no-root + - name: Install root project + run: poetry install --no-interaction + - name: Check format and linting + run: | + poetry run ruff format --check . + poetry run ruff check . + - name: Run tests and check coverage + run: | + poetry run coverage run -m pytest + poetry run coverage report --fail-under=85 diff --git a/.gitignore b/.gitignore index d16f774f..d2991ae6 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,7 @@ dist/ **/__pycache__/* *.py[cod] -.mypy_cache/ \ No newline at end of file +.mypy_cache/ +.coverage +htmlcov/ +.idea/ diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 34f891f9..3980af8b 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,6 +1,6 @@ repos: - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.5.0 # Use the ref you want to point at + rev: v4.5.0 hooks: - id: trailing-whitespace - # - id: ... \ No newline at end of file + - id: end-of-file-fixer diff --git a/poetry.lock b/poetry.lock index 5a61a6ea..1f553fad 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,6 +1,11 @@ -# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand. [[package]] +<<<<<<< HEAD +======= +<<<<<<< HEAD +======= +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "annotated-types" version = "0.6.0" description = "Reusable constraint types to use with typing.Annotated" @@ -29,6 +34,7 @@ files = [ typing-extensions = {version = ">=4.0.0", markers = "python_version < \"3.11\""} [[package]] +<<<<<<< HEAD name = "black" version = "24.2.0" description = "The uncompromising code formatter." @@ -88,6 +94,18 @@ files = [ [package.dependencies] colorama = {version = "*", markers = "platform_system == \"Windows\""} +======= +name = "cfgv" +version = "3.4.0" +description = "Validate configuration and produce human readable error messages." +optional = false +python-versions = ">=3.8" +files = [ + {file = "cfgv-3.4.0-py2.py3-none-any.whl", hash = "sha256:b7265b1f29fd3316bfcd2b330d63d024f2bfd8bcb8b0272f8e19a504856c48f9"}, + {file = "cfgv-3.4.0.tar.gz", hash = "sha256:e52591d4c5f5dead8e0f673fb16db7949d2cfb3f7da4582893288f0ded8fe560"}, +] + +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) [[package]] name = "colorama" version = "0.4.6" @@ -100,6 +118,73 @@ files = [ ] [[package]] +<<<<<<< HEAD +======= +name = "coverage" +version = "7.4.3" +description = "Code coverage measurement for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "coverage-7.4.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8580b827d4746d47294c0e0b92854c85a92c2227927433998f0d3320ae8a71b6"}, + {file = "coverage-7.4.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:718187eeb9849fc6cc23e0d9b092bc2348821c5e1a901c9f8975df0bc785bfd4"}, + {file = "coverage-7.4.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:767b35c3a246bcb55b8044fd3a43b8cd553dd1f9f2c1eeb87a302b1f8daa0524"}, + {file = "coverage-7.4.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ae7f19afe0cce50039e2c782bff379c7e347cba335429678450b8fe81c4ef96d"}, + {file = "coverage-7.4.3-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba3a8aaed13770e970b3df46980cb068d1c24af1a1968b7818b69af8c4347efb"}, + {file = "coverage-7.4.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:ee866acc0861caebb4f2ab79f0b94dbfbdbfadc19f82e6e9c93930f74e11d7a0"}, + {file = "coverage-7.4.3-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:506edb1dd49e13a2d4cac6a5173317b82a23c9d6e8df63efb4f0380de0fbccbc"}, + {file = "coverage-7.4.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd6545d97c98a192c5ac995d21c894b581f1fd14cf389be90724d21808b657e2"}, + {file = "coverage-7.4.3-cp310-cp310-win32.whl", hash = "sha256:f6a09b360d67e589236a44f0c39218a8efba2593b6abdccc300a8862cffc2f94"}, + {file = "coverage-7.4.3-cp310-cp310-win_amd64.whl", hash = "sha256:18d90523ce7553dd0b7e23cbb28865db23cddfd683a38fb224115f7826de78d0"}, + {file = "coverage-7.4.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:cbbe5e739d45a52f3200a771c6d2c7acf89eb2524890a4a3aa1a7fa0695d2a47"}, + {file = "coverage-7.4.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:489763b2d037b164846ebac0cbd368b8a4ca56385c4090807ff9fad817de4113"}, + {file = "coverage-7.4.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:451f433ad901b3bb00184d83fd83d135fb682d780b38af7944c9faeecb1e0bfe"}, + {file = "coverage-7.4.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:fcc66e222cf4c719fe7722a403888b1f5e1682d1679bd780e2b26c18bb648cdc"}, + {file = "coverage-7.4.3-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b3ec74cfef2d985e145baae90d9b1b32f85e1741b04cd967aaf9cfa84c1334f3"}, + {file = "coverage-7.4.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:abbbd8093c5229c72d4c2926afaee0e6e3140de69d5dcd918b2921f2f0c8baba"}, + {file = "coverage-7.4.3-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:35eb581efdacf7b7422af677b92170da4ef34500467381e805944a3201df2079"}, + {file = "coverage-7.4.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:8249b1c7334be8f8c3abcaaa996e1e4927b0e5a23b65f5bf6cfe3180d8ca7840"}, + {file = "coverage-7.4.3-cp311-cp311-win32.whl", hash = "sha256:cf30900aa1ba595312ae41978b95e256e419d8a823af79ce670835409fc02ad3"}, + {file = "coverage-7.4.3-cp311-cp311-win_amd64.whl", hash = "sha256:18c7320695c949de11a351742ee001849912fd57e62a706d83dfc1581897fa2e"}, + {file = "coverage-7.4.3-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:b51bfc348925e92a9bd9b2e48dad13431b57011fd1038f08316e6bf1df107d10"}, + {file = "coverage-7.4.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d6cdecaedea1ea9e033d8adf6a0ab11107b49571bbb9737175444cea6eb72328"}, + {file = "coverage-7.4.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3b2eccb883368f9e972e216c7b4c7c06cabda925b5f06dde0650281cb7666a30"}, + {file = "coverage-7.4.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6c00cdc8fa4e50e1cc1f941a7f2e3e0f26cb2a1233c9696f26963ff58445bac7"}, + {file = "coverage-7.4.3-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b9a4a8dd3dcf4cbd3165737358e4d7dfbd9d59902ad11e3b15eebb6393b0446e"}, + {file = "coverage-7.4.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:062b0a75d9261e2f9c6d071753f7eef0fc9caf3a2c82d36d76667ba7b6470003"}, + {file = "coverage-7.4.3-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:ebe7c9e67a2d15fa97b77ea6571ce5e1e1f6b0db71d1d5e96f8d2bf134303c1d"}, + {file = "coverage-7.4.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:c0a120238dd71c68484f02562f6d446d736adcc6ca0993712289b102705a9a3a"}, + {file = "coverage-7.4.3-cp312-cp312-win32.whl", hash = "sha256:37389611ba54fd6d278fde86eb2c013c8e50232e38f5c68235d09d0a3f8aa352"}, + {file = "coverage-7.4.3-cp312-cp312-win_amd64.whl", hash = "sha256:d25b937a5d9ffa857d41be042b4238dd61db888533b53bc76dc082cb5a15e914"}, + {file = "coverage-7.4.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:28ca2098939eabab044ad68850aac8f8db6bf0b29bc7f2887d05889b17346454"}, + {file = "coverage-7.4.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:280459f0a03cecbe8800786cdc23067a8fc64c0bd51dc614008d9c36e1659d7e"}, + {file = "coverage-7.4.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6c0cdedd3500e0511eac1517bf560149764b7d8e65cb800d8bf1c63ebf39edd2"}, + {file = "coverage-7.4.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9a9babb9466fe1da12417a4aed923e90124a534736de6201794a3aea9d98484e"}, + {file = "coverage-7.4.3-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dec9de46a33cf2dd87a5254af095a409ea3bf952d85ad339751e7de6d962cde6"}, + {file = "coverage-7.4.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:16bae383a9cc5abab9bb05c10a3e5a52e0a788325dc9ba8499e821885928968c"}, + {file = "coverage-7.4.3-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:2c854ce44e1ee31bda4e318af1dbcfc929026d12c5ed030095ad98197eeeaed0"}, + {file = "coverage-7.4.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:ce8c50520f57ec57aa21a63ea4f325c7b657386b3f02ccaedeccf9ebe27686e1"}, + {file = "coverage-7.4.3-cp38-cp38-win32.whl", hash = "sha256:708a3369dcf055c00ddeeaa2b20f0dd1ce664eeabde6623e516c5228b753654f"}, + {file = "coverage-7.4.3-cp38-cp38-win_amd64.whl", hash = "sha256:1bf25fbca0c8d121a3e92a2a0555c7e5bc981aee5c3fdaf4bb7809f410f696b9"}, + {file = "coverage-7.4.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3b253094dbe1b431d3a4ac2f053b6d7ede2664ac559705a704f621742e034f1f"}, + {file = "coverage-7.4.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:77fbfc5720cceac9c200054b9fab50cb2a7d79660609200ab83f5db96162d20c"}, + {file = "coverage-7.4.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6679060424faa9c11808598504c3ab472de4531c571ab2befa32f4971835788e"}, + {file = "coverage-7.4.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4af154d617c875b52651dd8dd17a31270c495082f3d55f6128e7629658d63765"}, + {file = "coverage-7.4.3-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8640f1fde5e1b8e3439fe482cdc2b0bb6c329f4bb161927c28d2e8879c6029ee"}, + {file = "coverage-7.4.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:69b9f6f66c0af29642e73a520b6fed25ff9fd69a25975ebe6acb297234eda501"}, + {file = "coverage-7.4.3-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:0842571634f39016a6c03e9d4aba502be652a6e4455fadb73cd3a3a49173e38f"}, + {file = "coverage-7.4.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:a78ed23b08e8ab524551f52953a8a05d61c3a760781762aac49f8de6eede8c45"}, + {file = "coverage-7.4.3-cp39-cp39-win32.whl", hash = "sha256:c0524de3ff096e15fcbfe8f056fdb4ea0bf497d584454f344d59fce069d3e6e9"}, + {file = "coverage-7.4.3-cp39-cp39-win_amd64.whl", hash = "sha256:0209a6369ccce576b43bb227dc8322d8ef9e323d089c6f3f26a597b09cb4d2aa"}, + {file = "coverage-7.4.3-pp38.pp39.pp310-none-any.whl", hash = "sha256:7cbde573904625509a3f37b6fecea974e363460b556a627c60dc2f47e2fffa51"}, + {file = "coverage-7.4.3.tar.gz", hash = "sha256:276f6077a5c61447a48d133ed13e759c09e62aff0dc84274a68dc18660104d52"}, +] + +[package.extras] +toml = ["tomli"] + +[[package]] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "dill" version = "0.3.8" description = "serialize all of Python" @@ -115,6 +200,20 @@ graph = ["objgraph (>=1.7.2)"] profile = ["gprof2dot (>=2022.7.29)"] [[package]] +<<<<<<< HEAD +======= +name = "distlib" +version = "0.3.8" +description = "Distribution utilities" +optional = false +python-versions = "*" +files = [ + {file = "distlib-0.3.8-py2.py3-none-any.whl", hash = "sha256:034db59a0b96f8ca18035f36290806a9a6e6bd9d1ff91e45a7f172eb17e51784"}, + {file = "distlib-0.3.8.tar.gz", hash = "sha256:1530ea13e350031b6312d8580ddb6b27a104275a31106523b8f123787f494f64"}, +] + +[[package]] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "exceptiongroup" version = "1.2.0" description = "Backport of PEP 654 (exception groups)" @@ -129,6 +228,39 @@ files = [ test = ["pytest (>=6)"] [[package]] +<<<<<<< HEAD +======= +name = "filelock" +version = "3.13.1" +description = "A platform independent file lock." +optional = false +python-versions = ">=3.8" +files = [ + {file = "filelock-3.13.1-py3-none-any.whl", hash = "sha256:57dbda9b35157b05fb3e58ee91448612eb674172fab98ee235ccb0b5bee19a1c"}, + {file = "filelock-3.13.1.tar.gz", hash = "sha256:521f5f56c50f8426f5e03ad3b281b490a87ef15bc6c526f168290f0c7148d44e"}, +] + +[package.extras] +docs = ["furo (>=2023.9.10)", "sphinx (>=7.2.6)", "sphinx-autodoc-typehints (>=1.24)"] +testing = ["covdefaults (>=2.3)", "coverage (>=7.3.2)", "diff-cover (>=8)", "pytest (>=7.4.3)", "pytest-cov (>=4.1)", "pytest-mock (>=3.12)", "pytest-timeout (>=2.2)"] +typing = ["typing-extensions (>=4.8)"] + +[[package]] +name = "identify" +version = "2.5.35" +description = "File identification library for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "identify-2.5.35-py2.py3-none-any.whl", hash = "sha256:c4de0081837b211594f8e877a6b4fad7ca32bbfc1a9307fdd61c28bfe923f13e"}, + {file = "identify-2.5.35.tar.gz", hash = "sha256:10a7ca245cfcd756a554a7288159f72ff105ad233c7c4b9c6f0f4d108f5f6791"}, +] + +[package.extras] +license = ["ukkonen"] + +[[package]] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "iniconfig" version = "2.0.0" description = "brain-dead simple config-ini parsing" @@ -223,6 +355,10 @@ files = [ ] [[package]] +<<<<<<< HEAD +======= +>>>>>>> 4222460 (Update Github Actions) +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "neo4j" version = "5.18.0" description = "Neo4j Bolt driver for Python" @@ -241,6 +377,25 @@ pandas = ["numpy (>=1.7.0,<2.0.0)", "pandas (>=1.1.0,<3.0.0)"] pyarrow = ["pyarrow (>=1.0.0)"] [[package]] +<<<<<<< HEAD +======= +<<<<<<< HEAD +======= +name = "nodeenv" +version = "1.8.0" +description = "Node.js virtual environment builder" +optional = false +python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*" +files = [ + {file = "nodeenv-1.8.0-py2.py3-none-any.whl", hash = "sha256:df865724bb3c3adc86b3876fa209771517b0cfe596beff01a92700e0e8be4cec"}, + {file = "nodeenv-1.8.0.tar.gz", hash = "sha256:d51e0c37e64fbf47d017feac3145cdbb58836d7eee8c6f6d3b6880c5456227d2"}, +] + +[package.dependencies] +setuptools = "*" + +[[package]] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "packaging" version = "23.2" description = "Core utilities for Python packages" @@ -252,6 +407,7 @@ files = [ ] [[package]] +<<<<<<< HEAD name = "pathspec" version = "0.12.1" description = "Utility library for gitignore style pattern matching of file paths." @@ -263,6 +419,8 @@ files = [ ] [[package]] +======= +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "platformdirs" version = "4.2.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." @@ -293,6 +451,27 @@ dev = ["pre-commit", "tox"] testing = ["pytest", "pytest-benchmark"] [[package]] +<<<<<<< HEAD +======= +name = "pre-commit" +version = "3.6.2" +description = "A framework for managing and maintaining multi-language pre-commit hooks." +optional = false +python-versions = ">=3.9" +files = [ + {file = "pre_commit-3.6.2-py2.py3-none-any.whl", hash = "sha256:ba637c2d7a670c10daedc059f5c49b5bd0aadbccfcd7ec15592cf9665117532c"}, + {file = "pre_commit-3.6.2.tar.gz", hash = "sha256:c3ef34f463045c88658c5b99f38c1e297abdcc0ff13f98d3370055fbbfabc67e"}, +] + +[package.dependencies] +cfgv = ">=2.0.0" +identify = ">=1.0.0" +nodeenv = ">=0.11.1" +pyyaml = ">=5.1" +virtualenv = ">=20.10.0" + +[[package]] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "pydantic" version = "2.6.3" description = "Data validation using Python type hints" @@ -434,13 +613,22 @@ testutils = ["gitpython (>3)"] [[package]] name = "pytest" +<<<<<<< HEAD version = "8.0.2" +======= +version = "8.1.0" +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) description = "pytest: simple powerful testing with Python" optional = false python-versions = ">=3.8" files = [ +<<<<<<< HEAD {file = "pytest-8.0.2-py3-none-any.whl", hash = "sha256:edfaaef32ce5172d5466b5127b42e0d6d35ebbe4453f0e3505d96afd93f6b096"}, {file = "pytest-8.0.2.tar.gz", hash = "sha256:d4051d623a2e0b7e51960ba963193b09ce6daeb9759a451844a21e4ddedfc1bd"}, +======= + {file = "pytest-8.1.0-py3-none-any.whl", hash = "sha256:ee32db7af8de4629a455806befa90559f307424c07b8413ccfc30bf5b221dd7e"}, + {file = "pytest-8.1.0.tar.gz", hash = "sha256:f8fa04ab8f98d185113ae60ea6d79c22f8143b14bc1caeced44a0ab844928323"}, +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) ] [package.dependencies] @@ -448,11 +636,19 @@ colorama = {version = "*", markers = "sys_platform == \"win32\""} exceptiongroup = {version = ">=1.0.0rc8", markers = "python_version < \"3.11\""} iniconfig = "*" packaging = "*" +<<<<<<< HEAD pluggy = ">=1.3.0,<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"] +======= +pluggy = ">=1.4,<2.0" +tomli = {version = ">=1", markers = "python_version < \"3.11\""} + +[package.extras] +testing = ["argcomplete", "attrs (>=19.2)", "hypothesis (>=3.56)", "mock", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"] +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) [[package]] name = "pytest-mock" @@ -472,6 +668,10 @@ pytest = ">=5.0" dev = ["pre-commit", "pytest-asyncio", "tox"] [[package]] +<<<<<<< HEAD +======= +>>>>>>> 4222460 (Update Github Actions) +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "pytz" version = "2024.1" description = "World timezone definitions, modern and historical" @@ -482,6 +682,7 @@ files = [ {file = "pytz-2024.1.tar.gz", hash = "sha256:2a29735ea9c18baf14b448846bde5a48030ed267578472d8955cd0e7443a9812"}, ] +<<<<<<< HEAD [[package]] name = "tomli" version = "2.0.1" @@ -550,3 +751,202 @@ zstd = ["zstandard (>=0.18.0)"] lock-version = "2.0" python-versions = "^3.8" content-hash = "e9571707d11b67cf7226bd97e391882098e9ca6065c92ce148da425824aa7718" +======= +<<<<<<< HEAD +[metadata] +lock-version = "2.0" +python-versions = ">=3.7" +content-hash = "4638595b14c9dedbce42766fd4745866b8011e8cd03967f27d170a0200602437" +======= +[[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_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a08c6f0fe150303c1c6b71ebcd7213c2858041a7e01975da3a99aed1e7a378ef"}, + {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 = "ruff" +version = "0.3.0" +description = "An extremely fast Python linter and code formatter, written in Rust." +optional = false +python-versions = ">=3.7" +files = [ + {file = "ruff-0.3.0-py3-none-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:7deb528029bacf845bdbb3dbb2927d8ef9b4356a5e731b10eef171e3f0a85944"}, + {file = "ruff-0.3.0-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:e1e0d4381ca88fb2b73ea0766008e703f33f460295de658f5467f6f229658c19"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2f7dbba46e2827dfcb0f0cc55fba8e96ba7c8700e0a866eb8cef7d1d66c25dcb"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:23dbb808e2f1d68eeadd5f655485e235c102ac6f12ad31505804edced2a5ae77"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3ef655c51f41d5fa879f98e40c90072b567c666a7114fa2d9fe004dffba00932"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:d0d3d7ef3d4f06433d592e5f7d813314a34601e6c5be8481cccb7fa760aa243e"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b08b356d06a792e49a12074b62222f9d4ea2a11dca9da9f68163b28c71bf1dd4"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9343690f95710f8cf251bee1013bf43030072b9f8d012fbed6ad702ef70d360a"}, + {file = "ruff-0.3.0-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a1f3ed501a42f60f4dedb7805fa8d4534e78b4e196f536bac926f805f0743d49"}, + {file = "ruff-0.3.0-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:cc30a9053ff2f1ffb505a585797c23434d5f6c838bacfe206c0e6cf38c921a1e"}, + {file = "ruff-0.3.0-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:5da894a29ec018a8293d3d17c797e73b374773943e8369cfc50495573d396933"}, + {file = "ruff-0.3.0-py3-none-musllinux_1_2_i686.whl", hash = "sha256:755c22536d7f1889be25f2baf6fedd019d0c51d079e8417d4441159f3bcd30c2"}, + {file = "ruff-0.3.0-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:dd73fe7f4c28d317855da6a7bc4aa29a1500320818dd8f27df95f70a01b8171f"}, + {file = "ruff-0.3.0-py3-none-win32.whl", hash = "sha256:19eacceb4c9406f6c41af806418a26fdb23120dfe53583df76d1401c92b7c14b"}, + {file = "ruff-0.3.0-py3-none-win_amd64.whl", hash = "sha256:128265876c1d703e5f5e5a4543bd8be47c73a9ba223fd3989d4aa87dd06f312f"}, + {file = "ruff-0.3.0-py3-none-win_arm64.whl", hash = "sha256:e3a4a6d46aef0a84b74fcd201a4401ea9a6cd85614f6a9435f2d33dd8cefbf83"}, + {file = "ruff-0.3.0.tar.gz", hash = "sha256:0886184ba2618d815067cf43e005388967b67ab9c80df52b32ec1152ab49f53a"}, +] + +[[package]] +name = "setuptools" +version = "69.1.1" +description = "Easily download, build, install, upgrade, and uninstall Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "setuptools-69.1.1-py3-none-any.whl", hash = "sha256:02fa291a0471b3a18b2b2481ed902af520c69e8ae0919c13da936542754b4c56"}, + {file = "setuptools-69.1.1.tar.gz", hash = "sha256:5c0806c7d9af348e6dd3777b4f4dbb42c7ad85b190104837488eab9a7c945cf8"}, +] + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.2)", "pip (>=19.1)", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.2)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] + +[[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 = "tomlkit" +version = "0.12.4" +description = "Style preserving TOML library" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tomlkit-0.12.4-py3-none-any.whl", hash = "sha256:5cd82d48a3dd89dee1f9d64420aa20ae65cfbd00668d6f094d7578a78efbb77b"}, + {file = "tomlkit-0.12.4.tar.gz", hash = "sha256:7ca1cfc12232806517a8515047ba66a19369e71edf2439d0f5824f91032b6cc3"}, +] + +[[package]] +name = "types-requests" +version = "2.31.0.20240218" +description = "Typing stubs for requests" +optional = false +python-versions = ">=3.8" +files = [ + {file = "types-requests-2.31.0.20240218.tar.gz", hash = "sha256:f1721dba8385958f504a5386240b92de4734e047a08a40751c1654d1ac3349c5"}, + {file = "types_requests-2.31.0.20240218-py3-none-any.whl", hash = "sha256:a82807ec6ddce8f00fe0e949da6d6bc1fbf1715420218a9640d695f70a9e5a9b"}, +] + +[package.dependencies] +urllib3 = ">=2" + +[[package]] +name = "typing-extensions" +version = "4.10.0" +description = "Backported and Experimental Type Hints for Python 3.8+" +optional = false +python-versions = ">=3.8" +files = [ + {file = "typing_extensions-4.10.0-py3-none-any.whl", hash = "sha256:69b1a937c3a517342112fb4c6df7e72fc39a38e7891a5730ed4985b5214b5475"}, + {file = "typing_extensions-4.10.0.tar.gz", hash = "sha256:b0abd7c89e8fb96f98db18d86106ff1d90ab692004eb746cf6eda2682f91b3cb"}, +] + +[[package]] +name = "urllib3" +version = "2.2.1" +description = "HTTP library with thread-safe connection pooling, file post, and more." +optional = false +python-versions = ">=3.8" +files = [ + {file = "urllib3-2.2.1-py3-none-any.whl", hash = "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d"}, + {file = "urllib3-2.2.1.tar.gz", hash = "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19"}, +] + +[package.extras] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] +h2 = ["h2 (>=4,<5)"] +socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] +zstd = ["zstandard (>=0.18.0)"] + +[[package]] +name = "virtualenv" +version = "20.25.1" +description = "Virtual Python Environment builder" +optional = false +python-versions = ">=3.7" +files = [ + {file = "virtualenv-20.25.1-py3-none-any.whl", hash = "sha256:961c026ac520bac5f69acb8ea063e8a4f071bcc9457b9c1f28f6b085c511583a"}, + {file = "virtualenv-20.25.1.tar.gz", hash = "sha256:e08e13ecdca7a0bd53798f356d5831434afa5b07b93f0abdf0797b7a06ffe197"}, +] + +[package.dependencies] +distlib = ">=0.3.7,<1" +filelock = ">=3.12.2,<4" +platformdirs = ">=3.9.1,<5" + +[package.extras] +docs = ["furo (>=2023.7.26)", "proselint (>=0.13)", "sphinx (>=7.1.2)", "sphinx-argparse (>=0.4)", "sphinxcontrib-towncrier (>=0.2.1a0)", "towncrier (>=23.6)"] +test = ["covdefaults (>=2.3)", "coverage (>=7.2.7)", "coverage-enable-subprocess (>=1)", "flaky (>=3.7)", "packaging (>=23.1)", "pytest (>=7.4)", "pytest-env (>=0.8.2)", "pytest-freezer (>=0.4.8)", "pytest-mock (>=3.11.1)", "pytest-randomly (>=3.12)", "pytest-timeout (>=2.1)", "setuptools (>=68)", "time-machine (>=2.10)"] + +[metadata] +lock-version = "2.0" +python-versions = "^3.8" +content-hash = "d62bd60ebf4ec5dee0e62c540d5bd2918394fae8b7c6988367caecf935738109" +>>>>>>> 4222460 (Update Github Actions) +>>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) diff --git a/pyproject.toml b/pyproject.toml index 675c1cee..31549133 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -6,37 +6,35 @@ authors = ["Neo4j, Inc "] license = "Apache License, Version 2.0" readme = "README.md" -[[tool.poetry.packages]] -include = "neo4j_genai" -from = "src" - [tool.poetry.dependencies] python = "^3.8" neo4j = "^5.17.0" types-requests = "^2.31.0.20240218" -pytest = "^8.0.2" -pytest-mock = "^3.12.0" pydantic = "^2.6.3" [tool.poetry.group.dev.dependencies] pylint = "^3.1.0" mypy = "^1.8.0" -black = "^24.2.0" +pytest = "^8.0.2" +pytest-mock = "^3.12.0" +pre-commit = { version = "^3.6.2", python = "^3.9" } +coverage = "^7.4.3" +ruff = "^0.3.0" [build-system] requires = ["poetry-core"] build-backend = "poetry.core.masonry.api" exclude = ["**/tests/"] -[tool.black] -line-length = 88 -target-version = ['py38'] -include = '\.pyi?$' -exclude = ''' -/( - .git - | .venv - | build - | dist -)/ -''' + +[tool.pytest.ini_options] +testpaths = ["tests"] +filterwarnings = [ + "", +] + +[tool.coverage.paths] +source = ["src"] + +[tool.pylint."MESSAGES CONTROL"] +disable="C0114,C0115" diff --git a/src/neo4j_genai/__init__.py b/src/neo4j_genai/__init__.py index e69de29b..c7748c58 100644 --- a/src/neo4j_genai/__init__.py +++ b/src/neo4j_genai/__init__.py @@ -0,0 +1,4 @@ +from .client import GenAIClient + + +__all__ = ["GenAIClient"] diff --git a/src/neo4j_genai/client.py b/src/neo4j_genai/client.py index eb9714e2..605ccfd5 100644 --- a/src/neo4j_genai/client.py +++ b/src/neo4j_genai/client.py @@ -2,8 +2,8 @@ from pydantic import ValidationError from neo4j import Driver from neo4j.exceptions import CypherSyntaxError -from neo4j_genai.embeddings import Embeddings -from neo4j_genai.types import CreateIndexModel, SimilaritySearchModel, Neo4jRecord +from .embeddings import Embeddings +from .types import CreateIndexModel, SimilaritySearchModel, Neo4jRecord class GenAIClient: @@ -31,7 +31,10 @@ def _verify_version(self) -> None: """ version = self.database_query("CALL dbms.components()")[0]["versions"][0] if "aura" in version: - version_tuple = (*tuple(map(int, version.split("-")[0].split("."))), 0) + version_tuple = ( + *tuple(map(int, version.split("-")[0].split("."))), + 0, + ) else: version_tuple = tuple(map(int, version.split("."))) @@ -106,7 +109,7 @@ def create_index( "toInteger($dimensions)," "$similarity_fn )" ) - self.database_query(query, params=index_data.dict()) + self.database_query(query, params=index_data.model_dump()) def drop_index(self, name: str) -> None: """ @@ -160,7 +163,7 @@ def similarity_search( error_details = e.errors() raise ValueError(f"Validation failed: {error_details}") - parameters = validated_data.dict(exclude_none=True) + parameters = validated_data.model_dump(exclude_none=True) if query_text: if not self.embeddings: @@ -169,7 +172,7 @@ def similarity_search( parameters["query_vector"] = query_vector db_query_string = """ - CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) + CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ records = self.database_query(db_query_string, params=parameters) diff --git a/src/neo4j_genai/embeddings.py b/src/neo4j_genai/embeddings.py index 1fe801ff..52b27eb8 100644 --- a/src/neo4j_genai/embeddings.py +++ b/src/neo4j_genai/embeddings.py @@ -1,5 +1,5 @@ from abc import ABC, abstractmethod -from neo4j_genai.types import EmbeddingVector +from .types import EmbeddingVector class Embeddings(ABC): diff --git a/src/neo4j_genai/types.py b/src/neo4j_genai/types.py index 82da6699..e56f2013 100644 --- a/src/neo4j_genai/types.py +++ b/src/neo4j_genai/types.py @@ -1,5 +1,5 @@ from typing import List, Any, Literal, Optional -from pydantic import BaseModel, PositiveInt, Field, root_validator +from pydantic import BaseModel, PositiveInt, Field, model_validator class Neo4jRecord(BaseModel): @@ -25,7 +25,7 @@ class SimilaritySearchModel(BaseModel): query_vector: Optional[List[float]] = None query_text: Optional[str] = None - @root_validator(pre=True) + @model_validator(mode="before") def check_query(cls, values): """ Validates that one of either query_vector or query_text is provided exclusively. diff --git a/tests/conftest.py b/tests/conftest.py new file mode 100644 index 00000000..59791534 --- /dev/null +++ b/tests/conftest.py @@ -0,0 +1,14 @@ +import pytest +from neo4j_genai import GenAIClient +from unittest.mock import Mock, patch + + +@pytest.fixture +def driver(): + return Mock() + + +@pytest.fixture +@patch("neo4j_genai.GenAIClient._verify_version") +def client(_verify_version_mock, driver): + return GenAIClient(driver) diff --git a/tests/test_client.py b/tests/test_client.py index e69de29b..74483307 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -0,0 +1,132 @@ +import pytest +from neo4j_genai import GenAIClient +from unittest.mock import Mock, patch +from neo4j.exceptions import CypherSyntaxError + + +@patch( + "neo4j_genai.GenAIClient.database_query", + return_value=[{"versions": ["5.11-aura"]}], +) +def test_genai_client_supported_aura_version(mock_database_query, driver): + GenAIClient(driver) + mock_database_query.assert_called_once() + + +@patch( + "neo4j_genai.GenAIClient.database_query", + return_value=[{"versions": ["5.3-aura"]}], +) +def test_genai_client_no_supported_aura_version(driver): + with pytest.raises(ValueError): + GenAIClient(driver) + + +@patch( + "neo4j_genai.GenAIClient.database_query", + return_value=[{"versions": ["5.11.5"]}], +) +def test_genai_client_supported_version(mock_database_query, driver): + GenAIClient(driver) + mock_database_query.assert_called_once() + + +@patch( + "neo4j_genai.GenAIClient.database_query", + return_value=[{"versions": ["4.3.5"]}], +) +def test_genai_client_no_supported_version(driver): + with pytest.raises(ValueError): + GenAIClient(driver) + + +@patch("neo4j_genai.GenAIClient.database_query") +def test_create_index_happy_path(mock_database_query, client): + client.create_index("my-index", "People", "name", 2048, "cosine") + query = ( + "CALL db.index.vector.createNodeIndex(" + "$name," + "$label," + "$property," + "toInteger($dimensions)," + "$similarity_fn )" + ) + mock_database_query.assert_called_once_with( + query, + params={ + "name": "my-index", + "label": "People", + "property": "name", + "dimensions": 2048, + "similarity_fn": "cosine", + }, + ) + + +def test_create_index_too_big_dimension(client): + with pytest.raises(ValueError): + client.create_index("my-index", "People", "name", 5024, "cosine") + + +def test_create_index_validation_error_dimensions(client): + with pytest.raises(ValueError) as excinfo: + client.create_index("my-index", "People", "name", "no-dim", "cosine") + assert "Error for inputs to create_index" in str(excinfo) + + +def test_create_index_validation_error_similarity_fn(client): + with pytest.raises(ValueError) as excinfo: + client.create_index("my-index", "People", "name", "no-dim", "algebra") + assert "Error for inputs to create_index" in str(excinfo) + + +@patch("neo4j_genai.GenAIClient.database_query") +def test_drop_index(mock_database_query, client): + client.drop_index("my-index") + + query = "DROP INDEX $name" + + mock_database_query.assert_called_with(query, params={"name": "my-index"}) + + +def test_database_query_happy(client, driver): + class Session: + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def run(self, query, params): + m_list = [] + for i in range(3): + mock = Mock() + mock.data.return_value = i + m_list.append(mock) + + return m_list + + driver.session = Session + res = client.database_query("MATCH (p:$label) RETURN p", {"label": "People"}) + assert res == [0, 1, 2] + + +def test_database_query_cypher_error(client, driver): + class Session: + def __enter__(self): + return self + + def __exit__(self, exc_type, exc_value, traceback): + pass + + def run(self, query, params): + raise CypherSyntaxError + + driver.session = Session + + with pytest.raises(ValueError): + client.database_query("MATCH (p:$label) RETURN p", {"label": "People"}) + + +def test_similarity_search(): + pass From 18c32f3ad1af971fc806c70201e50e57ffdf3558 Mon Sep 17 00:00:00 2001 From: Will Tai Date: Mon, 11 Mar 2024 15:26:11 +0000 Subject: [PATCH 2/9] Fixed conflicts and lock dependencies relock dependencies Linting with ruff fixed linting issues --- examples/similarity_search_for_text.py | 1 - poetry.lock | 313 +++++-------------------- pyproject.toml | 4 + src/neo4j_genai/client.py | 9 +- 4 files changed, 67 insertions(+), 260 deletions(-) diff --git a/examples/similarity_search_for_text.py b/examples/similarity_search_for_text.py index 1b93b225..4b154a9c 100644 --- a/examples/similarity_search_for_text.py +++ b/examples/similarity_search_for_text.py @@ -1,6 +1,5 @@ from typing import List from neo4j import GraphDatabase -from neo4j.exceptions import DatabaseError from neo4j_genai.client import GenAIClient from random import random diff --git a/poetry.lock b/poetry.lock index 1f553fad..813c5c70 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,11 +1,6 @@ -# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. [[package]] -<<<<<<< HEAD -======= -<<<<<<< HEAD -======= ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "annotated-types" version = "0.6.0" description = "Reusable constraint types to use with typing.Annotated" @@ -34,67 +29,6 @@ files = [ typing-extensions = {version = ">=4.0.0", markers = "python_version < \"3.11\""} [[package]] -<<<<<<< HEAD -name = "black" -version = "24.2.0" -description = "The uncompromising code formatter." -optional = false -python-versions = ">=3.8" -files = [ - {file = "black-24.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6981eae48b3b33399c8757036c7f5d48a535b962a7c2310d19361edeef64ce29"}, - {file = "black-24.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d533d5e3259720fdbc1b37444491b024003e012c5173f7d06825a77508085430"}, - {file = "black-24.2.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:61a0391772490ddfb8a693c067df1ef5227257e72b0e4108482b8d41b5aee13f"}, - {file = "black-24.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:992e451b04667116680cb88f63449267c13e1ad134f30087dec8527242e9862a"}, - {file = "black-24.2.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:163baf4ef40e6897a2a9b83890e59141cc8c2a98f2dda5080dc15c00ee1e62cd"}, - {file = "black-24.2.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e37c99f89929af50ffaf912454b3e3b47fd64109659026b678c091a4cd450fb2"}, - {file = "black-24.2.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4f9de21bafcba9683853f6c96c2d515e364aee631b178eaa5145fc1c61a3cc92"}, - {file = "black-24.2.0-cp311-cp311-win_amd64.whl", hash = "sha256:9db528bccb9e8e20c08e716b3b09c6bdd64da0dd129b11e160bf082d4642ac23"}, - {file = "black-24.2.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:d84f29eb3ee44859052073b7636533ec995bd0f64e2fb43aeceefc70090e752b"}, - {file = "black-24.2.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1e08fb9a15c914b81dd734ddd7fb10513016e5ce7e6704bdd5e1251ceee51ac9"}, - {file = "black-24.2.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:810d445ae6069ce64030c78ff6127cd9cd178a9ac3361435708b907d8a04c693"}, - {file = "black-24.2.0-cp312-cp312-win_amd64.whl", hash = "sha256:ba15742a13de85e9b8f3239c8f807723991fbfae24bad92d34a2b12e81904982"}, - {file = "black-24.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:7e53a8c630f71db01b28cd9602a1ada68c937cbf2c333e6ed041390d6968faf4"}, - {file = "black-24.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:93601c2deb321b4bad8f95df408e3fb3943d85012dddb6121336b8e24a0d1218"}, - {file = "black-24.2.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a0057f800de6acc4407fe75bb147b0c2b5cbb7c3ed110d3e5999cd01184d53b0"}, - {file = "black-24.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:faf2ee02e6612577ba0181f4347bcbcf591eb122f7841ae5ba233d12c39dcb4d"}, - {file = "black-24.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:057c3dc602eaa6fdc451069bd027a1b2635028b575a6c3acfd63193ced20d9c8"}, - {file = "black-24.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:08654d0797e65f2423f850fc8e16a0ce50925f9337fb4a4a176a7aa4026e63f8"}, - {file = "black-24.2.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ca610d29415ee1a30a3f30fab7a8f4144e9d34c89a235d81292a1edb2b55f540"}, - {file = "black-24.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:4dd76e9468d5536abd40ffbc7a247f83b2324f0c050556d9c371c2b9a9a95e31"}, - {file = "black-24.2.0-py3-none-any.whl", hash = "sha256:e8a6ae970537e67830776488bca52000eaa37fa63b9988e8c487458d9cd5ace6"}, - {file = "black-24.2.0.tar.gz", hash = "sha256:bce4f25c27c3435e4dace4815bcb2008b87e167e3bf4ee47ccdc5ce906eb4894"}, -] - -[package.dependencies] -click = ">=8.0.0" -mypy-extensions = ">=0.4.3" -packaging = ">=22.0" -pathspec = ">=0.9.0" -platformdirs = ">=2" -tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} -typing-extensions = {version = ">=4.0.1", markers = "python_version < \"3.11\""} - -[package.extras] -colorama = ["colorama (>=0.4.3)"] -d = ["aiohttp (>=3.7.4)", "aiohttp (>=3.7.4,!=3.9.0)"] -jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] -uvloop = ["uvloop (>=0.15.2)"] - -[[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\""} - -======= name = "cfgv" version = "3.4.0" description = "Validate configuration and produce human readable error messages." @@ -105,7 +39,6 @@ files = [ {file = "cfgv-3.4.0.tar.gz", hash = "sha256:e52591d4c5f5dead8e0f673fb16db7949d2cfb3f7da4582893288f0ded8fe560"}, ] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) [[package]] name = "colorama" version = "0.4.6" @@ -118,8 +51,6 @@ files = [ ] [[package]] -<<<<<<< HEAD -======= name = "coverage" version = "7.4.3" description = "Code coverage measurement for Python" @@ -184,7 +115,6 @@ files = [ toml = ["tomli"] [[package]] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "dill" version = "0.3.8" description = "serialize all of Python" @@ -200,8 +130,6 @@ graph = ["objgraph (>=1.7.2)"] profile = ["gprof2dot (>=2022.7.29)"] [[package]] -<<<<<<< HEAD -======= name = "distlib" version = "0.3.8" description = "Distribution utilities" @@ -213,7 +141,6 @@ files = [ ] [[package]] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "exceptiongroup" version = "1.2.0" description = "Backport of PEP 654 (exception groups)" @@ -228,8 +155,6 @@ files = [ test = ["pytest (>=6)"] [[package]] -<<<<<<< HEAD -======= name = "filelock" version = "3.13.1" description = "A platform independent file lock." @@ -260,7 +185,6 @@ files = [ license = ["ukkonen"] [[package]] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "iniconfig" version = "2.0.0" description = "brain-dead simple config-ini parsing" @@ -298,38 +222,38 @@ files = [ [[package]] name = "mypy" -version = "1.8.0" +version = "1.9.0" description = "Optional static typing for Python" optional = false python-versions = ">=3.8" files = [ - {file = "mypy-1.8.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:485a8942f671120f76afffff70f259e1cd0f0cfe08f81c05d8816d958d4577d3"}, - {file = "mypy-1.8.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:df9824ac11deaf007443e7ed2a4a26bebff98d2bc43c6da21b2b64185da011c4"}, - {file = "mypy-1.8.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2afecd6354bbfb6e0160f4e4ad9ba6e4e003b767dd80d85516e71f2e955ab50d"}, - {file = "mypy-1.8.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:8963b83d53ee733a6e4196954502b33567ad07dfd74851f32be18eb932fb1cb9"}, - {file = "mypy-1.8.0-cp310-cp310-win_amd64.whl", hash = "sha256:e46f44b54ebddbeedbd3d5b289a893219065ef805d95094d16a0af6630f5d410"}, - {file = "mypy-1.8.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:855fe27b80375e5c5878492f0729540db47b186509c98dae341254c8f45f42ae"}, - {file = "mypy-1.8.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:4c886c6cce2d070bd7df4ec4a05a13ee20c0aa60cb587e8d1265b6c03cf91da3"}, - {file = "mypy-1.8.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d19c413b3c07cbecf1f991e2221746b0d2a9410b59cb3f4fb9557f0365a1a817"}, - {file = "mypy-1.8.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:9261ed810972061388918c83c3f5cd46079d875026ba97380f3e3978a72f503d"}, - {file = "mypy-1.8.0-cp311-cp311-win_amd64.whl", hash = "sha256:51720c776d148bad2372ca21ca29256ed483aa9a4cdefefcef49006dff2a6835"}, - {file = "mypy-1.8.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:52825b01f5c4c1c4eb0db253ec09c7aa17e1a7304d247c48b6f3599ef40db8bd"}, - {file = "mypy-1.8.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f5ac9a4eeb1ec0f1ccdc6f326bcdb464de5f80eb07fb38b5ddd7b0de6bc61e55"}, - {file = "mypy-1.8.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:afe3fe972c645b4632c563d3f3eff1cdca2fa058f730df2b93a35e3b0c538218"}, - {file = "mypy-1.8.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:42c6680d256ab35637ef88891c6bd02514ccb7e1122133ac96055ff458f93fc3"}, - {file = "mypy-1.8.0-cp312-cp312-win_amd64.whl", hash = "sha256:720a5ca70e136b675af3af63db533c1c8c9181314d207568bbe79051f122669e"}, - {file = "mypy-1.8.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:028cf9f2cae89e202d7b6593cd98db6759379f17a319b5faf4f9978d7084cdc6"}, - {file = "mypy-1.8.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4e6d97288757e1ddba10dd9549ac27982e3e74a49d8d0179fc14d4365c7add66"}, - {file = "mypy-1.8.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f1478736fcebb90f97e40aff11a5f253af890c845ee0c850fe80aa060a267c6"}, - {file = "mypy-1.8.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:42419861b43e6962a649068a61f4a4839205a3ef525b858377a960b9e2de6e0d"}, - {file = "mypy-1.8.0-cp38-cp38-win_amd64.whl", hash = "sha256:2b5b6c721bd4aabaadead3a5e6fa85c11c6c795e0c81a7215776ef8afc66de02"}, - {file = "mypy-1.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5c1538c38584029352878a0466f03a8ee7547d7bd9f641f57a0f3017a7c905b8"}, - {file = "mypy-1.8.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4ef4be7baf08a203170f29e89d79064463b7fc7a0908b9d0d5114e8009c3a259"}, - {file = "mypy-1.8.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7178def594014aa6c35a8ff411cf37d682f428b3b5617ca79029d8ae72f5402b"}, - {file = "mypy-1.8.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ab3c84fa13c04aeeeabb2a7f67a25ef5d77ac9d6486ff33ded762ef353aa5592"}, - {file = "mypy-1.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:99b00bc72855812a60d253420d8a2eae839b0afa4938f09f4d2aa9bb4654263a"}, - {file = "mypy-1.8.0-py3-none-any.whl", hash = "sha256:538fd81bb5e430cc1381a443971c0475582ff9f434c16cd46d2c66763ce85d9d"}, - {file = "mypy-1.8.0.tar.gz", hash = "sha256:6ff8b244d7085a0b425b56d327b480c3b29cafbd2eff27316a004f9a7391ae07"}, + {file = "mypy-1.9.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:f8a67616990062232ee4c3952f41c779afac41405806042a8126fe96e098419f"}, + {file = "mypy-1.9.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d357423fa57a489e8c47b7c85dfb96698caba13d66e086b412298a1a0ea3b0ed"}, + {file = "mypy-1.9.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:49c87c15aed320de9b438ae7b00c1ac91cd393c1b854c2ce538e2a72d55df150"}, + {file = "mypy-1.9.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:48533cdd345c3c2e5ef48ba3b0d3880b257b423e7995dada04248725c6f77374"}, + {file = "mypy-1.9.0-cp310-cp310-win_amd64.whl", hash = "sha256:4d3dbd346cfec7cb98e6cbb6e0f3c23618af826316188d587d1c1bc34f0ede03"}, + {file = "mypy-1.9.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:653265f9a2784db65bfca694d1edd23093ce49740b2244cde583aeb134c008f3"}, + {file = "mypy-1.9.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3a3c007ff3ee90f69cf0a15cbcdf0995749569b86b6d2f327af01fd1b8aee9dc"}, + {file = "mypy-1.9.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2418488264eb41f69cc64a69a745fad4a8f86649af4b1041a4c64ee61fc61129"}, + {file = "mypy-1.9.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:68edad3dc7d70f2f17ae4c6c1b9471a56138ca22722487eebacfd1eb5321d612"}, + {file = "mypy-1.9.0-cp311-cp311-win_amd64.whl", hash = "sha256:85ca5fcc24f0b4aeedc1d02f93707bccc04733f21d41c88334c5482219b1ccb3"}, + {file = "mypy-1.9.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:aceb1db093b04db5cd390821464504111b8ec3e351eb85afd1433490163d60cd"}, + {file = "mypy-1.9.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:0235391f1c6f6ce487b23b9dbd1327b4ec33bb93934aa986efe8a9563d9349e6"}, + {file = "mypy-1.9.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d4d5ddc13421ba3e2e082a6c2d74c2ddb3979c39b582dacd53dd5d9431237185"}, + {file = "mypy-1.9.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:190da1ee69b427d7efa8aa0d5e5ccd67a4fb04038c380237a0d96829cb157913"}, + {file = "mypy-1.9.0-cp312-cp312-win_amd64.whl", hash = "sha256:fe28657de3bfec596bbeef01cb219833ad9d38dd5393fc649f4b366840baefe6"}, + {file = "mypy-1.9.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:e54396d70be04b34f31d2edf3362c1edd023246c82f1730bbf8768c28db5361b"}, + {file = "mypy-1.9.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:5e6061f44f2313b94f920e91b204ec600982961e07a17e0f6cd83371cb23f5c2"}, + {file = "mypy-1.9.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:81a10926e5473c5fc3da8abb04119a1f5811a236dc3a38d92015cb1e6ba4cb9e"}, + {file = "mypy-1.9.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:b685154e22e4e9199fc95f298661deea28aaede5ae16ccc8cbb1045e716b3e04"}, + {file = "mypy-1.9.0-cp38-cp38-win_amd64.whl", hash = "sha256:5d741d3fc7c4da608764073089e5f58ef6352bedc223ff58f2f038c2c4698a89"}, + {file = "mypy-1.9.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:587ce887f75dd9700252a3abbc9c97bbe165a4a630597845c61279cf32dfbf02"}, + {file = "mypy-1.9.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:f88566144752999351725ac623471661c9d1cd8caa0134ff98cceeea181789f4"}, + {file = "mypy-1.9.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:61758fabd58ce4b0720ae1e2fea5cfd4431591d6d590b197775329264f86311d"}, + {file = "mypy-1.9.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:e49499be624dead83927e70c756970a0bc8240e9f769389cdf5714b0784ca6bf"}, + {file = "mypy-1.9.0-cp39-cp39-win_amd64.whl", hash = "sha256:571741dc4194b4f82d344b15e8837e8c5fcc462d66d076748142327626a1b6e9"}, + {file = "mypy-1.9.0-py3-none-any.whl", hash = "sha256:a260627a570559181a9ea5de61ac6297aa5af202f06fd7ab093ce74e7181e43e"}, + {file = "mypy-1.9.0.tar.gz", hash = "sha256:3cc5da0127e6a478cddd906068496a97a7618a21ce9b54bde5bf7e539c7af974"}, ] [package.dependencies] @@ -355,10 +279,6 @@ files = [ ] [[package]] -<<<<<<< HEAD -======= ->>>>>>> 4222460 (Update Github Actions) ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "neo4j" version = "5.18.0" description = "Neo4j Bolt driver for Python" @@ -377,10 +297,6 @@ pandas = ["numpy (>=1.7.0,<2.0.0)", "pandas (>=1.1.0,<3.0.0)"] pyarrow = ["pyarrow (>=1.0.0)"] [[package]] -<<<<<<< HEAD -======= -<<<<<<< HEAD -======= name = "nodeenv" version = "1.8.0" description = "Node.js virtual environment builder" @@ -395,32 +311,17 @@ files = [ setuptools = "*" [[package]] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "packaging" -version = "23.2" +version = "24.0" 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"}, + {file = "packaging-24.0-py3-none-any.whl", hash = "sha256:2ddfb553fdf02fb784c234c7ba6ccc288296ceabec964ad2eae3777778130bc5"}, + {file = "packaging-24.0.tar.gz", hash = "sha256:eb82c5e3e56209074766e6885bb04b8c38a0c015d0a30036ebe7ece34c9989e9"}, ] [[package]] -<<<<<<< HEAD -name = "pathspec" -version = "0.12.1" -description = "Utility library for gitignore style pattern matching of file paths." -optional = false -python-versions = ">=3.8" -files = [ - {file = "pathspec-0.12.1-py3-none-any.whl", hash = "sha256:a0d503e138a4c123b27490a4f7beda6a01c6f288df0e4a8b79c7eb0dc7b4cc08"}, - {file = "pathspec-0.12.1.tar.gz", hash = "sha256:a482d51503a1ab33b1c67a6c3813a26953dbdc71c31dacaef9a838c4e29f5712"}, -] - -[[package]] -======= ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "platformdirs" version = "4.2.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." @@ -451,8 +352,6 @@ dev = ["pre-commit", "tox"] testing = ["pytest", "pytest-benchmark"] [[package]] -<<<<<<< HEAD -======= name = "pre-commit" version = "3.6.2" description = "A framework for managing and maintaining multi-language pre-commit hooks." @@ -471,7 +370,6 @@ pyyaml = ">=5.1" virtualenv = ">=20.10.0" [[package]] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "pydantic" version = "2.6.3" description = "Data validation using Python type hints" @@ -613,22 +511,13 @@ testutils = ["gitpython (>3)"] [[package]] name = "pytest" -<<<<<<< HEAD -version = "8.0.2" -======= -version = "8.1.0" ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) +version = "8.1.1" description = "pytest: simple powerful testing with Python" optional = false python-versions = ">=3.8" files = [ -<<<<<<< HEAD - {file = "pytest-8.0.2-py3-none-any.whl", hash = "sha256:edfaaef32ce5172d5466b5127b42e0d6d35ebbe4453f0e3505d96afd93f6b096"}, - {file = "pytest-8.0.2.tar.gz", hash = "sha256:d4051d623a2e0b7e51960ba963193b09ce6daeb9759a451844a21e4ddedfc1bd"}, -======= - {file = "pytest-8.1.0-py3-none-any.whl", hash = "sha256:ee32db7af8de4629a455806befa90559f307424c07b8413ccfc30bf5b221dd7e"}, - {file = "pytest-8.1.0.tar.gz", hash = "sha256:f8fa04ab8f98d185113ae60ea6d79c22f8143b14bc1caeced44a0ab844928323"}, ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) + {file = "pytest-8.1.1-py3-none-any.whl", hash = "sha256:2a8386cfc11fa9d2c50ee7b2a57e7d898ef90470a7a34c4b949ff59662bb78b7"}, + {file = "pytest-8.1.1.tar.gz", hash = "sha256:ac978141a75948948817d360297b7aae0fcb9d6ff6bc9ec6d514b85d5a65c044"}, ] [package.dependencies] @@ -636,19 +525,11 @@ colorama = {version = "*", markers = "sys_platform == \"win32\""} exceptiongroup = {version = ">=1.0.0rc8", markers = "python_version < \"3.11\""} iniconfig = "*" packaging = "*" -<<<<<<< HEAD -pluggy = ">=1.3.0,<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"] -======= pluggy = ">=1.4,<2.0" tomli = {version = ">=1", markers = "python_version < \"3.11\""} [package.extras] testing = ["argcomplete", "attrs (>=19.2)", "hypothesis (>=3.56)", "mock", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"] ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) [[package]] name = "pytest-mock" @@ -668,10 +549,6 @@ pytest = ">=5.0" dev = ["pre-commit", "pytest-asyncio", "tox"] [[package]] -<<<<<<< HEAD -======= ->>>>>>> 4222460 (Update Github Actions) ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) name = "pytz" version = "2024.1" description = "World timezone definitions, modern and historical" @@ -682,82 +559,6 @@ files = [ {file = "pytz-2024.1.tar.gz", hash = "sha256:2a29735ea9c18baf14b448846bde5a48030ed267578472d8955cd0e7443a9812"}, ] -<<<<<<< HEAD -[[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 = "tomlkit" -version = "0.12.4" -description = "Style preserving TOML library" -optional = false -python-versions = ">=3.7" -files = [ - {file = "tomlkit-0.12.4-py3-none-any.whl", hash = "sha256:5cd82d48a3dd89dee1f9d64420aa20ae65cfbd00668d6f094d7578a78efbb77b"}, - {file = "tomlkit-0.12.4.tar.gz", hash = "sha256:7ca1cfc12232806517a8515047ba66a19369e71edf2439d0f5824f91032b6cc3"}, -] - -[[package]] -name = "types-requests" -version = "2.31.0.20240218" -description = "Typing stubs for requests" -optional = false -python-versions = ">=3.8" -files = [ - {file = "types-requests-2.31.0.20240218.tar.gz", hash = "sha256:f1721dba8385958f504a5386240b92de4734e047a08a40751c1654d1ac3349c5"}, - {file = "types_requests-2.31.0.20240218-py3-none-any.whl", hash = "sha256:a82807ec6ddce8f00fe0e949da6d6bc1fbf1715420218a9640d695f70a9e5a9b"}, -] - -[package.dependencies] -urllib3 = ">=2" - -[[package]] -name = "typing-extensions" -version = "4.10.0" -description = "Backported and Experimental Type Hints for Python 3.8+" -optional = false -python-versions = ">=3.8" -files = [ - {file = "typing_extensions-4.10.0-py3-none-any.whl", hash = "sha256:69b1a937c3a517342112fb4c6df7e72fc39a38e7891a5730ed4985b5214b5475"}, - {file = "typing_extensions-4.10.0.tar.gz", hash = "sha256:b0abd7c89e8fb96f98db18d86106ff1d90ab692004eb746cf6eda2682f91b3cb"}, -] - -[[package]] -name = "urllib3" -version = "2.2.1" -description = "HTTP library with thread-safe connection pooling, file post, and more." -optional = false -python-versions = ">=3.8" -files = [ - {file = "urllib3-2.2.1-py3-none-any.whl", hash = "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d"}, - {file = "urllib3-2.2.1.tar.gz", hash = "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19"}, -] - -[package.extras] -brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] -h2 = ["h2 (>=4,<5)"] -socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] -zstd = ["zstandard (>=0.18.0)"] - -[metadata] -lock-version = "2.0" -python-versions = "^3.8" -content-hash = "e9571707d11b67cf7226bd97e391882098e9ca6065c92ce148da425824aa7718" -======= -<<<<<<< HEAD -[metadata] -lock-version = "2.0" -python-versions = ">=3.7" -content-hash = "4638595b14c9dedbce42766fd4745866b8011e8cd03967f27d170a0200602437" -======= [[package]] name = "pyyaml" version = "6.0.1" @@ -820,28 +621,28 @@ files = [ [[package]] name = "ruff" -version = "0.3.0" +version = "0.3.2" description = "An extremely fast Python linter and code formatter, written in Rust." optional = false python-versions = ">=3.7" files = [ - {file = "ruff-0.3.0-py3-none-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:7deb528029bacf845bdbb3dbb2927d8ef9b4356a5e731b10eef171e3f0a85944"}, - {file = "ruff-0.3.0-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:e1e0d4381ca88fb2b73ea0766008e703f33f460295de658f5467f6f229658c19"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2f7dbba46e2827dfcb0f0cc55fba8e96ba7c8700e0a866eb8cef7d1d66c25dcb"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:23dbb808e2f1d68eeadd5f655485e235c102ac6f12ad31505804edced2a5ae77"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3ef655c51f41d5fa879f98e40c90072b567c666a7114fa2d9fe004dffba00932"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:d0d3d7ef3d4f06433d592e5f7d813314a34601e6c5be8481cccb7fa760aa243e"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b08b356d06a792e49a12074b62222f9d4ea2a11dca9da9f68163b28c71bf1dd4"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9343690f95710f8cf251bee1013bf43030072b9f8d012fbed6ad702ef70d360a"}, - {file = "ruff-0.3.0-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a1f3ed501a42f60f4dedb7805fa8d4534e78b4e196f536bac926f805f0743d49"}, - {file = "ruff-0.3.0-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:cc30a9053ff2f1ffb505a585797c23434d5f6c838bacfe206c0e6cf38c921a1e"}, - {file = "ruff-0.3.0-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:5da894a29ec018a8293d3d17c797e73b374773943e8369cfc50495573d396933"}, - {file = "ruff-0.3.0-py3-none-musllinux_1_2_i686.whl", hash = "sha256:755c22536d7f1889be25f2baf6fedd019d0c51d079e8417d4441159f3bcd30c2"}, - {file = "ruff-0.3.0-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:dd73fe7f4c28d317855da6a7bc4aa29a1500320818dd8f27df95f70a01b8171f"}, - {file = "ruff-0.3.0-py3-none-win32.whl", hash = "sha256:19eacceb4c9406f6c41af806418a26fdb23120dfe53583df76d1401c92b7c14b"}, - {file = "ruff-0.3.0-py3-none-win_amd64.whl", hash = "sha256:128265876c1d703e5f5e5a4543bd8be47c73a9ba223fd3989d4aa87dd06f312f"}, - {file = "ruff-0.3.0-py3-none-win_arm64.whl", hash = "sha256:e3a4a6d46aef0a84b74fcd201a4401ea9a6cd85614f6a9435f2d33dd8cefbf83"}, - {file = "ruff-0.3.0.tar.gz", hash = "sha256:0886184ba2618d815067cf43e005388967b67ab9c80df52b32ec1152ab49f53a"}, + {file = "ruff-0.3.2-py3-none-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:77f2612752e25f730da7421ca5e3147b213dca4f9a0f7e0b534e9562c5441f01"}, + {file = "ruff-0.3.2-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:9966b964b2dd1107797be9ca7195002b874424d1d5472097701ae8f43eadef5d"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b83d17ff166aa0659d1e1deaf9f2f14cbe387293a906de09bc4860717eb2e2da"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bb875c6cc87b3703aeda85f01c9aebdce3d217aeaca3c2e52e38077383f7268a"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:be75e468a6a86426430373d81c041b7605137a28f7014a72d2fc749e47f572aa"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:967978ac2d4506255e2f52afe70dda023fc602b283e97685c8447d036863a302"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1231eacd4510f73222940727ac927bc5d07667a86b0cbe822024dd00343e77e9"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2c6d613b19e9a8021be2ee1d0e27710208d1603b56f47203d0abbde906929a9b"}, + {file = "ruff-0.3.2-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c8439338a6303585d27b66b4626cbde89bb3e50fa3cae86ce52c1db7449330a7"}, + {file = "ruff-0.3.2-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:de8b480d8379620cbb5ea466a9e53bb467d2fb07c7eca54a4aa8576483c35d36"}, + {file = "ruff-0.3.2-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:b74c3de9103bd35df2bb05d8b2899bf2dbe4efda6474ea9681280648ec4d237d"}, + {file = "ruff-0.3.2-py3-none-musllinux_1_2_i686.whl", hash = "sha256:f380be9fc15a99765c9cf316b40b9da1f6ad2ab9639e551703e581a5e6da6745"}, + {file = "ruff-0.3.2-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:0ac06a3759c3ab9ef86bbeca665d31ad3aa9a4b1c17684aadb7e61c10baa0df4"}, + {file = "ruff-0.3.2-py3-none-win32.whl", hash = "sha256:9bd640a8f7dd07a0b6901fcebccedadeb1a705a50350fb86b4003b805c81385a"}, + {file = "ruff-0.3.2-py3-none-win_amd64.whl", hash = "sha256:0c1bdd9920cab5707c26c8b3bf33a064a4ca7842d91a99ec0634fec68f9f4037"}, + {file = "ruff-0.3.2-py3-none-win_arm64.whl", hash = "sha256:5f65103b1d76e0d600cabd577b04179ff592064eaa451a70a81085930e907d0b"}, + {file = "ruff-0.3.2.tar.gz", hash = "sha256:fa78ec9418eb1ca3db392811df3376b46471ae93792a81af2d1cbb0e5dcb5142"}, ] [[package]] @@ -884,13 +685,13 @@ files = [ [[package]] name = "types-requests" -version = "2.31.0.20240218" +version = "2.31.0.20240311" description = "Typing stubs for requests" optional = false python-versions = ">=3.8" files = [ - {file = "types-requests-2.31.0.20240218.tar.gz", hash = "sha256:f1721dba8385958f504a5386240b92de4734e047a08a40751c1654d1ac3349c5"}, - {file = "types_requests-2.31.0.20240218-py3-none-any.whl", hash = "sha256:a82807ec6ddce8f00fe0e949da6d6bc1fbf1715420218a9640d695f70a9e5a9b"}, + {file = "types-requests-2.31.0.20240311.tar.gz", hash = "sha256:b1c1b66abfb7fa79aae09097a811c4aa97130eb8831c60e47aee4ca344731ca5"}, + {file = "types_requests-2.31.0.20240311-py3-none-any.whl", hash = "sha256:47872893d65a38e282ee9f277a4ee50d1b28bd592040df7d1fdaffdf3779937d"}, ] [package.dependencies] @@ -948,5 +749,3 @@ test = ["covdefaults (>=2.3)", "coverage (>=7.2.7)", "coverage-enable-subprocess lock-version = "2.0" python-versions = "^3.8" content-hash = "d62bd60ebf4ec5dee0e62c540d5bd2918394fae8b7c6988367caecf935738109" ->>>>>>> 4222460 (Update Github Actions) ->>>>>>> 0c921fe (Update Github Actions and resolve merge conflicts) diff --git a/pyproject.toml b/pyproject.toml index 31549133..d97f6f49 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -6,6 +6,10 @@ authors = ["Neo4j, Inc "] license = "Apache License, Version 2.0" readme = "README.md" +[[tool.poetry.packages]] +include = "neo4j_genai" +from = "src" + [tool.poetry.dependencies] python = "^3.8" neo4j = "^5.17.0" diff --git a/src/neo4j_genai/client.py b/src/neo4j_genai/client.py index 605ccfd5..f106e632 100644 --- a/src/neo4j_genai/client.py +++ b/src/neo4j_genai/client.py @@ -178,7 +178,12 @@ def similarity_search( records = self.database_query(db_query_string, params=parameters) try: - return [Neo4jRecord(node=record["node"], score=record["score"]) for record in records] + return [ + Neo4jRecord(node=record["node"], score=record["score"]) + for record in records + ] except ValidationError as e: error_details = e.errors() - raise ValueError(f"Validation failed while constructing output: {error_details}") + raise ValueError( + f"Validation failed while constructing output: {error_details}" + ) From f134f7db0285b23501f373955d645eca4d37e73b Mon Sep 17 00:00:00 2001 From: Will Tai Date: Tue, 12 Mar 2024 13:54:27 +0000 Subject: [PATCH 3/9] Adds tests for similarity search --- examples/similarity_search_for_text.py | 10 +-- src/neo4j_genai/client.py | 13 ++-- .../{embeddings.py => embedder.py} | 2 +- tests/conftest.py | 13 ++++ tests/test_client.py | 73 ++++++++++++++++++- 5 files changed, 96 insertions(+), 15 deletions(-) rename src/neo4j_genai/{embeddings.py => embedder.py} (94%) diff --git a/examples/similarity_search_for_text.py b/examples/similarity_search_for_text.py index 4b154a9c..6fc67278 100644 --- a/examples/similarity_search_for_text.py +++ b/examples/similarity_search_for_text.py @@ -3,7 +3,7 @@ from neo4j_genai.client import GenAIClient from random import random -from neo4j_genai.embeddings import Embeddings +from neo4j_genai.embedder import Embedder URI = "neo4j://localhost:7687" AUTH = ("neo4j", "password") @@ -15,16 +15,16 @@ driver = GraphDatabase.driver(URI, auth=AUTH) -# Create Embeddings object -class CustomEmbeddings(Embeddings): +# Create Embedder object +class CustomEmbedder(Embedder): def embed_query(self, text: str) -> List[float]: return [random() for _ in range(DIMENSION)] -embeddings = CustomEmbeddings() +embedder = CustomEmbedder() # Initialize the client -client = GenAIClient(driver, embeddings) +client = GenAIClient(driver, embedder) # Creating the index client.create_index( diff --git a/src/neo4j_genai/client.py b/src/neo4j_genai/client.py index f106e632..f1eb5102 100644 --- a/src/neo4j_genai/client.py +++ b/src/neo4j_genai/client.py @@ -2,7 +2,7 @@ from pydantic import ValidationError from neo4j import Driver from neo4j.exceptions import CypherSyntaxError -from .embeddings import Embeddings +from .embedder import Embedder from .types import CreateIndexModel, SimilaritySearchModel, Neo4jRecord @@ -14,11 +14,11 @@ class GenAIClient: def __init__( self, driver: Driver, - embeddings: Optional[Embeddings] = None, + embedder: Optional[Embedder] = None, ) -> None: self.driver = driver self._verify_version() - self.embeddings = embeddings + self.embedder = embedder def _verify_version(self) -> None: """ @@ -147,7 +147,7 @@ def similarity_search( Raises: ValueError: If validation of the input arguments fail. - ValueError: If no embeddings is provided. + ValueError: If no embedder is provided. Returns: List[Neo4jRecord]: The `top_k` neighbors found in vector search with their nodes and scores. @@ -166,10 +166,11 @@ def similarity_search( parameters = validated_data.model_dump(exclude_none=True) if query_text: - if not self.embeddings: + if not self.embedder: raise ValueError("Embedding method required for text query.") - query_vector = self.embeddings.embed_query(query_text) + query_vector = self.embedder.embed_query(query_text) parameters["query_vector"] = query_vector + del parameters["query_text"] db_query_string = """ CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) diff --git a/src/neo4j_genai/embeddings.py b/src/neo4j_genai/embedder.py similarity index 94% rename from src/neo4j_genai/embeddings.py rename to src/neo4j_genai/embedder.py index 52b27eb8..a4c7fb23 100644 --- a/src/neo4j_genai/embeddings.py +++ b/src/neo4j_genai/embedder.py @@ -2,7 +2,7 @@ from .types import EmbeddingVector -class Embeddings(ABC): +class Embedder(ABC): """Interface for embedding models.""" @abstractmethod diff --git a/tests/conftest.py b/tests/conftest.py index 59791534..729e3547 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,6 +1,8 @@ import pytest from neo4j_genai import GenAIClient from unittest.mock import Mock, patch +from typing import List +from neo4j_genai.embedder import Embedder @pytest.fixture @@ -12,3 +14,14 @@ def driver(): @patch("neo4j_genai.GenAIClient._verify_version") def client(_verify_version_mock, driver): return GenAIClient(driver) + + +@pytest.fixture +@patch("neo4j_genai.GenAIClient._verify_version") +def client_with_embedder(_verify_version_mock, driver): + class CustomEmbedder(Embedder): + def embed_query(self, text: str) -> List[float]: + return [1.0 for _ in range(1536)] + + embedder = CustomEmbedder() + return GenAIClient(driver, embedder) diff --git a/tests/test_client.py b/tests/test_client.py index 74483307..a82e917f 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -76,7 +76,7 @@ def test_create_index_validation_error_dimensions(client): def test_create_index_validation_error_similarity_fn(client): with pytest.raises(ValueError) as excinfo: - client.create_index("my-index", "People", "name", "no-dim", "algebra") + client.create_index("my-index", "People", "name", 1536, "algebra") assert "Error for inputs to create_index" in str(excinfo) @@ -128,5 +128,72 @@ def run(self, query, params): client.database_query("MATCH (p:$label) RETURN p", {"label": "People"}) -def test_similarity_search(): - pass +@patch("neo4j_genai.GenAIClient.database_query") +def test_similarity_search_vector_happy_path(mock_database_query, client): + index_name = "my-index" + query_vector = [1.1, 2.2, 3.3] + top_k = 5 + + client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + + query = """ + CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) + YIELD node, score + """ + mock_database_query.assert_called_once_with( + query, + params={ + "index_name": index_name, + "top_k": top_k, + "query_vector": query_vector, + }, + ) + + +@patch("neo4j_genai.GenAIClient.database_query") +def test_similarity_search_text_happy_path(mock_database_query, client_with_embedder): + index_name = "my-index" + query_text = "may thy knife chip and shatter" + query_vector = [1.0 for _ in range(1536)] + top_k = 5 + + client_with_embedder.similarity_search( + name=index_name, query_text=query_text, top_k=top_k + ) + + query = """ + CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) + YIELD node, score + """ + mock_database_query.assert_called_once_with( + query, + params={ + "index_name": index_name, + "top_k": top_k, + "query_vector": query_vector, + }, + ) + + +def test_similarity_search_missing_embedder_for_text(client): + index_name = "my-index" + query_text = "may thy knife chip and shatter" + top_k = 5 + + with pytest.raises(ValueError): + client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) + + +def test_similarity_search_both_text_and_vector(client): + index_name = "my-index" + query_text = "may thy knife chip and shatter" + query_vector = [1.1, 2.2, 3.3] + top_k = 5 + + with pytest.raises(ValueError): + client.similarity_search( + name=index_name, + query_text=query_text, + query_vector=query_vector, + top_k=top_k, + ) From 4332fcda8be7f432df4148088b46b865f5c0adc4 Mon Sep 17 00:00:00 2001 From: Will Tai Date: Tue, 12 Mar 2024 17:57:40 +0000 Subject: [PATCH 4/9] Changed test to mock neo4j driver's execute_driver instead of the client's _database_query() Changed test_client.py to have mocks at execute_query level Changed precommit config to include ruff linting and formatting Update pre-commit Corrected cypher query error test and dimensions constraint in CreateIndexModel data type ruff formatting remove drop index in example --- examples/similarity_search_for_text.py | 2 +- examples/similarity_search_for_vector.py | 2 +- src/neo4j_genai/client.py | 23 +-- src/neo4j_genai/types.py | 4 +- tests/conftest.py | 14 +- tests/test_client.py | 182 +++++++++++------------ 6 files changed, 111 insertions(+), 116 deletions(-) diff --git a/examples/similarity_search_for_text.py b/examples/similarity_search_for_text.py index 6fc67278..9f5e9276 100644 --- a/examples/similarity_search_for_text.py +++ b/examples/similarity_search_for_text.py @@ -46,7 +46,7 @@ def embed_query(self, text: str) -> List[float]: parameters = { "vector": vector, } -client.database_query(insert_query, params=parameters) +client._database_query(insert_query, params=parameters) # Perform the similarity search for a text query query_text = "hello world" diff --git a/examples/similarity_search_for_vector.py b/examples/similarity_search_for_vector.py index 157dc29b..159b81ae 100644 --- a/examples/similarity_search_for_vector.py +++ b/examples/similarity_search_for_vector.py @@ -35,7 +35,7 @@ parameters = { "vector": vector, } -client.database_query(insert_query, params=parameters) +client._database_query(insert_query, params=parameters) # Perform the similarity search for a vector query query_vector = [random() for _ in range(DIMENSION)] diff --git a/src/neo4j_genai/client.py b/src/neo4j_genai/client.py index f1eb5102..7676c570 100644 --- a/src/neo4j_genai/client.py +++ b/src/neo4j_genai/client.py @@ -29,7 +29,7 @@ def _verify_version(self) -> None: indexing. Raises a ValueError if the connected Neo4j version is not supported. """ - version = self.database_query("CALL dbms.components()")[0]["versions"][0] + version = self._database_query("CALL dbms.components()")[0]["versions"][0] if "aura" in version: version_tuple = ( *tuple(map(int, version.split("-")[0].split("."))), @@ -45,7 +45,9 @@ def _verify_version(self) -> None: "Version index is only supported in Neo4j version 5.11 or greater" ) - def database_query(self, query: str, params: Dict = {}) -> List[Dict[str, Any]]: + def _database_query( + self, query: str, params: Optional[Dict[str, Any]] = None + ) -> List[Dict[str, Any]]: """ This method sends a Cypher query to the connected Neo4j database and returns the results as a list of dictionaries. @@ -57,12 +59,11 @@ def database_query(self, query: str, params: Dict = {}) -> List[Dict[str, Any]]: Returns: List[Dict[str, Any]]: List of dictionaries containing the query results. """ - with self.driver.session() as session: - try: - data = session.run(query, params) - return [r.data() for r in data] - except CypherSyntaxError as e: - raise ValueError(f"Cypher Statement is not valid\n{e}") + try: + records, _, _ = self.driver.execute_query(query, params) + return records + except CypherSyntaxError as e: + raise ValueError(f"Cypher Statement is not valid\n{e}") def create_index( self, @@ -109,7 +110,7 @@ def create_index( "toInteger($dimensions)," "$similarity_fn )" ) - self.database_query(query, params=index_data.model_dump()) + self._database_query(query, params=index_data.model_dump()) def drop_index(self, name: str) -> None: """ @@ -124,7 +125,7 @@ def drop_index(self, name: str) -> None: parameters = { "name": name, } - self.database_query(query, params=parameters) + self._database_query(query, params=parameters) def similarity_search( self, @@ -176,7 +177,7 @@ def similarity_search( CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ - records = self.database_query(db_query_string, params=parameters) + records = self._database_query(db_query_string, params=parameters) try: return [ diff --git a/src/neo4j_genai/types.py b/src/neo4j_genai/types.py index e56f2013..3182a49c 100644 --- a/src/neo4j_genai/types.py +++ b/src/neo4j_genai/types.py @@ -1,5 +1,5 @@ from typing import List, Any, Literal, Optional -from pydantic import BaseModel, PositiveInt, Field, model_validator +from pydantic import BaseModel, PositiveInt, model_validator class Neo4jRecord(BaseModel): @@ -15,7 +15,7 @@ class CreateIndexModel(BaseModel): name: str label: str property: str - dimensions: int = Field(ge=1, le=2048) + dimensions: int = PositiveInt similarity_fn: Literal["euclidean", "cosine"] diff --git a/tests/conftest.py b/tests/conftest.py index 729e3547..a57c6252 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,13 +1,13 @@ import pytest from neo4j_genai import GenAIClient -from unittest.mock import Mock, patch +from unittest.mock import MagicMock, patch from typing import List from neo4j_genai.embedder import Embedder @pytest.fixture def driver(): - return Mock() + return MagicMock() @pytest.fixture @@ -20,8 +20,14 @@ def client(_verify_version_mock, driver): @patch("neo4j_genai.GenAIClient._verify_version") def client_with_embedder(_verify_version_mock, driver): class CustomEmbedder(Embedder): + def __init__(self): + self.dimension = 1536 + def embed_query(self, text: str) -> List[float]: - return [1.0 for _ in range(1536)] + return [1.0 for _ in range(self.dimension)] + + def set_dimension(self, dimension: int): + self.dimension = dimension embedder = CustomEmbedder() - return GenAIClient(driver, embedder) + return GenAIClient(driver, embedder), embedder diff --git a/tests/test_client.py b/tests/test_client.py index a82e917f..809a47ac 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -1,49 +1,45 @@ import pytest from neo4j_genai import GenAIClient -from unittest.mock import Mock, patch from neo4j.exceptions import CypherSyntaxError -@patch( - "neo4j_genai.GenAIClient.database_query", - return_value=[{"versions": ["5.11-aura"]}], -) -def test_genai_client_supported_aura_version(mock_database_query, driver): - GenAIClient(driver) - mock_database_query.assert_called_once() +def test_genai_client_supported_aura_version(driver): + driver.execute_query.return_value = [[{"versions": ["5.11-aura"]}], None, None] + + GenAIClient(driver=driver) -@patch( - "neo4j_genai.GenAIClient.database_query", - return_value=[{"versions": ["5.3-aura"]}], -) def test_genai_client_no_supported_aura_version(driver): - with pytest.raises(ValueError): - GenAIClient(driver) + driver.execute_query.return_value = [[{"versions": ["5.3-aura"]}], None, None] + + with pytest.raises(ValueError) as excinfo: + GenAIClient(driver=driver) + + assert "Version index is only supported in Neo4j version 5.11 or greater" in str( + excinfo + ) -@patch( - "neo4j_genai.GenAIClient.database_query", - return_value=[{"versions": ["5.11.5"]}], -) -def test_genai_client_supported_version(mock_database_query, driver): - GenAIClient(driver) - mock_database_query.assert_called_once() +def test_genai_client_supported_version(driver): + driver.execute_query.return_value = [[{"versions": ["5.11.5"]}], None, None] + + GenAIClient(driver=driver) -@patch( - "neo4j_genai.GenAIClient.database_query", - return_value=[{"versions": ["4.3.5"]}], -) def test_genai_client_no_supported_version(driver): - with pytest.raises(ValueError): - GenAIClient(driver) + driver.execute_query.return_value = [[{"versions": ["4.3.5"]}], None, None] + with pytest.raises(ValueError) as excinfo: + GenAIClient(driver=driver) -@patch("neo4j_genai.GenAIClient.database_query") -def test_create_index_happy_path(mock_database_query, client): - client.create_index("my-index", "People", "name", 2048, "cosine") - query = ( + assert "Version index is only supported in Neo4j version 5.11 or greater" in str( + excinfo + ) + + +def test_create_index_happy_path(driver, client): + driver.execute_query.return_value = [None, None, None] + create_query = ( "CALL db.index.vector.createNodeIndex(" "$name," "$label," @@ -51,9 +47,12 @@ def test_create_index_happy_path(mock_database_query, client): "toInteger($dimensions)," "$similarity_fn )" ) - mock_database_query.assert_called_once_with( - query, - params={ + + client.create_index("my-index", "People", "name", 2048, "cosine") + + driver.execute_query.assert_called_once_with( + create_query, + { "name": "my-index", "label": "People", "property": "name", @@ -63,11 +62,6 @@ def test_create_index_happy_path(mock_database_query, client): ) -def test_create_index_too_big_dimension(client): - with pytest.raises(ValueError): - client.create_index("my-index", "People", "name", 5024, "cosine") - - def test_create_index_validation_error_dimensions(client): with pytest.raises(ValueError) as excinfo: client.create_index("my-index", "People", "name", "no-dim", "cosine") @@ -80,69 +74,56 @@ def test_create_index_validation_error_similarity_fn(client): assert "Error for inputs to create_index" in str(excinfo) -@patch("neo4j_genai.GenAIClient.database_query") -def test_drop_index(mock_database_query, client): - client.drop_index("my-index") +def test_drop_index(driver, client): + driver.execute_query.return_value = [None, None, None] + drop_query = "DROP INDEX $name" - query = "DROP INDEX $name" + client.drop_index("my-index") - mock_database_query.assert_called_with(query, params={"name": "my-index"}) + driver.execute_query.assert_called_once_with( + drop_query, + {"name": "my-index"}, + ) def test_database_query_happy(client, driver): - class Session: - def __enter__(self): - return self - - def __exit__(self, exc_type, exc_value, traceback): - pass - - def run(self, query, params): - m_list = [] - for i in range(3): - mock = Mock() - mock.data.return_value = i - m_list.append(mock) + expected_db_result = [0, 1, 2] + driver.execute_query.return_value = [expected_db_result, None, None] - return m_list + res = client._database_query("MATCH (p:$label) RETURN p", {"label": "People"}) - driver.session = Session - res = client.database_query("MATCH (p:$label) RETURN p", {"label": "People"}) - assert res == [0, 1, 2] + assert res == expected_db_result def test_database_query_cypher_error(client, driver): - class Session: - def __enter__(self): - return self - - def __exit__(self, exc_type, exc_value, traceback): - pass - - def run(self, query, params): - raise CypherSyntaxError + driver.execute_query.side_effect = CypherSyntaxError - driver.session = Session + with pytest.raises(ValueError) as excinfo: + client._database_query("MATCH (p:$label) RETURN p", {"label": "People"}) - with pytest.raises(ValueError): - client.database_query("MATCH (p:$label) RETURN p", {"label": "People"}) + assert "Cypher Statement is not valid" in str(excinfo) -@patch("neo4j_genai.GenAIClient.database_query") -def test_similarity_search_vector_happy_path(mock_database_query, client): +def test_similarity_search_vector_happy_path(driver, client): index_name = "my-index" - query_vector = [1.1, 2.2, 3.3] + dimensions = 1536 + query_vector = [1.0 for _ in range(dimensions)] top_k = 5 - - client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) - - query = """ + driver.execute_query.return_value = [ + [{"node": "dummy-node", "score": 1.0}], + None, + None, + ] + search_query = """ CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ - mock_database_query.assert_called_once_with( - query, - params={ + + client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + + driver.execute_query.assert_called_once_with( + search_query, + { "index_name": index_name, "top_k": top_k, "query_vector": query_vector, @@ -150,24 +131,29 @@ def test_similarity_search_vector_happy_path(mock_database_query, client): ) -@patch("neo4j_genai.GenAIClient.database_query") -def test_similarity_search_text_happy_path(mock_database_query, client_with_embedder): +def test_similarity_search_text_happy_path(driver, client_with_embedder): + client, embedder = client_with_embedder index_name = "my-index" query_text = "may thy knife chip and shatter" - query_vector = [1.0 for _ in range(1536)] + dimensions = 1536 + query_vector = [1.0 for _ in range(dimensions)] top_k = 5 - - client_with_embedder.similarity_search( - name=index_name, query_text=query_text, top_k=top_k - ) - - query = """ + driver.execute_query.return_value = [ + [{"node": "dummy-node", "score": 1.0}], + None, + None, + ] + embedder.set_dimension(dimensions) + search_query = """ CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ - mock_database_query.assert_called_once_with( - query, - params={ + + client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) + + driver.execute_query.assert_called_once_with( + search_query, + { "index_name": index_name, "top_k": top_k, "query_vector": query_vector, @@ -180,7 +166,7 @@ def test_similarity_search_missing_embedder_for_text(client): query_text = "may thy knife chip and shatter" top_k = 5 - with pytest.raises(ValueError): + with pytest.raises(ValueError, match="Embedding method required for text query"): client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) @@ -190,7 +176,9 @@ def test_similarity_search_both_text_and_vector(client): query_vector = [1.1, 2.2, 3.3] top_k = 5 - with pytest.raises(ValueError): + with pytest.raises( + ValueError, match="You must provide exactly one of query_vector or query_text." + ): client.similarity_search( name=index_name, query_text=query_text, From d144cf9acac8b8ec1e05a75494349572a00c2822 Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Fri, 15 Mar 2024 16:35:53 +0000 Subject: [PATCH 5/9] Add pypi publishing workflow --- .github/workflows/publish.yaml | 73 ++++++++++++++++++++++++++++++++++ 1 file changed, 73 insertions(+) create mode 100644 .github/workflows/publish.yaml diff --git a/.github/workflows/publish.yaml b/.github/workflows/publish.yaml new file mode 100644 index 00000000..1e3da16b --- /dev/null +++ b/.github/workflows/publish.yaml @@ -0,0 +1,73 @@ +name: Publish Python 🐍 distribution 📦 to PyPI and TestPyPI + +on: push + +jobs: + build: + name: Build distribution 📦 + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + - name: Set up Python + uses: actions/setup-python@v4 + with: + python-version: "3.8" + - name: Install pypa/build + run: >- + python3 -m + pip install + build + --user + - name: Build a binary wheel and a source tarball + run: python3 -m build + - name: Store the distribution packages + uses: actions/upload-artifact@v3 + with: + name: python-package-distributions + path: dist/ + + publish-to-pypi: + name: >- + Publish Python 🐍 distribution 📦 to PyPI + if: startsWith(github.ref, 'refs/tags/') + needs: + - build + runs-on: ubuntu-latest + environment: + name: pypi + url: https://pypi.org/p/neo4j-genai + permissions: + id-token: write + steps: + - name: Download all the dists + uses: actions/download-artifact@v3 + with: + name: python-package-distributions + path: dist/ + - name: Publish distribution 📦 to PyPI + uses: pypa/gh-action-pypi-publish@release/v1 + + publish-to-testpypi: + name: Publish Python 🐍 distribution 📦 to TestPyPI + needs: + - build + runs-on: ubuntu-latest + + environment: + name: testpypi + url: https://test.pypi.org/p/neo4j-genai + + permissions: + id-token: write + + steps: + - name: Download all the dists + uses: actions/download-artifact@v3 + with: + name: python-package-distributions + path: dist/ + - name: Publish distribution 📦 to TestPyPI + uses: pypa/gh-action-pypi-publish@release/v1 + with: + repository-url: https://test.pypi.org/legacy/ From 73574e65325bb483f98470214ec200a62681795a Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Fri, 15 Mar 2024 16:36:25 +0000 Subject: [PATCH 6/9] Remove database_query method and use execute_query --- examples/similarity_search_for_text.py | 2 +- examples/similarity_search_for_vector.py | 3 +- src/neo4j_genai/client.py | 33 +++--------- tests/conftest.py | 17 ------ tests/test_client.py | 66 ++++++++++++------------ 5 files changed, 43 insertions(+), 78 deletions(-) diff --git a/examples/similarity_search_for_text.py b/examples/similarity_search_for_text.py index 9f5e9276..40c09492 100644 --- a/examples/similarity_search_for_text.py +++ b/examples/similarity_search_for_text.py @@ -46,7 +46,7 @@ def embed_query(self, text: str) -> List[float]: parameters = { "vector": vector, } -client._database_query(insert_query, params=parameters) +driver.execute_query(insert_query, parameters) # Perform the similarity search for a text query query_text = "hello world" diff --git a/examples/similarity_search_for_vector.py b/examples/similarity_search_for_vector.py index 159b81ae..6c4c9c3a 100644 --- a/examples/similarity_search_for_vector.py +++ b/examples/similarity_search_for_vector.py @@ -35,7 +35,8 @@ parameters = { "vector": vector, } -client._database_query(insert_query, params=parameters) + +driver.execute_query(insert_query, parameters) # Perform the similarity search for a vector query query_vector = [random() for _ in range(DIMENSION)] diff --git a/src/neo4j_genai/client.py b/src/neo4j_genai/client.py index 7676c570..d7643ecc 100644 --- a/src/neo4j_genai/client.py +++ b/src/neo4j_genai/client.py @@ -1,7 +1,6 @@ -from typing import List, Dict, Any, Optional +from typing import List, Optional from pydantic import ValidationError from neo4j import Driver -from neo4j.exceptions import CypherSyntaxError from .embedder import Embedder from .types import CreateIndexModel, SimilaritySearchModel, Neo4jRecord @@ -29,7 +28,9 @@ def _verify_version(self) -> None: indexing. Raises a ValueError if the connected Neo4j version is not supported. """ - version = self._database_query("CALL dbms.components()")[0]["versions"][0] + records, _, _ = self.driver.execute_query("CALL dbms.components()") + version = records[0]["versions"][0] + if "aura" in version: version_tuple = ( *tuple(map(int, version.split("-")[0].split("."))), @@ -45,26 +46,6 @@ def _verify_version(self) -> None: "Version index is only supported in Neo4j version 5.11 or greater" ) - def _database_query( - self, query: str, params: Optional[Dict[str, Any]] = None - ) -> List[Dict[str, Any]]: - """ - This method sends a Cypher query to the connected Neo4j database - and returns the results as a list of dictionaries. - - Args: - query (str): The Cypher query to execute. - params (Dict, optional): Dictionary of query parameters. Defaults to {}. - - Returns: - List[Dict[str, Any]]: List of dictionaries containing the query results. - """ - try: - records, _, _ = self.driver.execute_query(query, params) - return records - except CypherSyntaxError as e: - raise ValueError(f"Cypher Statement is not valid\n{e}") - def create_index( self, name: str, @@ -110,7 +91,7 @@ def create_index( "toInteger($dimensions)," "$similarity_fn )" ) - self._database_query(query, params=index_data.model_dump()) + self.driver.execute_query(query, index_data.model_dump()) def drop_index(self, name: str) -> None: """ @@ -125,7 +106,7 @@ def drop_index(self, name: str) -> None: parameters = { "name": name, } - self._database_query(query, params=parameters) + self.driver.execute_query(query, parameters) def similarity_search( self, @@ -177,7 +158,7 @@ def similarity_search( CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ - records = self._database_query(db_query_string, params=parameters) + records, _, _ = self.driver.execute_query(db_query_string, parameters) try: return [ diff --git a/tests/conftest.py b/tests/conftest.py index a57c6252..07992d05 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -14,20 +14,3 @@ def driver(): @patch("neo4j_genai.GenAIClient._verify_version") def client(_verify_version_mock, driver): return GenAIClient(driver) - - -@pytest.fixture -@patch("neo4j_genai.GenAIClient._verify_version") -def client_with_embedder(_verify_version_mock, driver): - class CustomEmbedder(Embedder): - def __init__(self): - self.dimension = 1536 - - def embed_query(self, text: str) -> List[float]: - return [1.0 for _ in range(self.dimension)] - - def set_dimension(self, dimension: int): - self.dimension = dimension - - embedder = CustomEmbedder() - return GenAIClient(driver, embedder), embedder diff --git a/tests/test_client.py b/tests/test_client.py index 809a47ac..6d5714ec 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -1,7 +1,7 @@ import pytest from neo4j_genai import GenAIClient -from neo4j.exceptions import CypherSyntaxError - +from neo4j_genai.types import Neo4jRecord +from unittest.mock import patch, MagicMock def test_genai_client_supported_aura_version(driver): driver.execute_query.return_value = [[{"versions": ["5.11-aura"]}], None, None] @@ -74,42 +74,29 @@ def test_create_index_validation_error_similarity_fn(client): assert "Error for inputs to create_index" in str(excinfo) -def test_drop_index(driver, client): - driver.execute_query.return_value = [None, None, None] +def test_drop_index(client): + client.driver.execute_query.return_value = [None, None, None] drop_query = "DROP INDEX $name" client.drop_index("my-index") - driver.execute_query.assert_called_once_with( + client.driver.execute_query.assert_called_once_with( drop_query, {"name": "my-index"}, ) +@patch("neo4j_genai.GenAIClient._verify_version") +def test_similarity_search_vector_happy_path(_verify_version_mock, driver): + custom_embeddings = MagicMock() -def test_database_query_happy(client, driver): - expected_db_result = [0, 1, 2] - driver.execute_query.return_value = [expected_db_result, None, None] - - res = client._database_query("MATCH (p:$label) RETURN p", {"label": "People"}) - - assert res == expected_db_result - - -def test_database_query_cypher_error(client, driver): - driver.execute_query.side_effect = CypherSyntaxError + client = GenAIClient(driver, custom_embeddings) - with pytest.raises(ValueError) as excinfo: - client._database_query("MATCH (p:$label) RETURN p", {"label": "People"}) - - assert "Cypher Statement is not valid" in str(excinfo) - - -def test_similarity_search_vector_happy_path(driver, client): index_name = "my-index" dimensions = 1536 query_vector = [1.0 for _ in range(dimensions)] top_k = 5 - driver.execute_query.return_value = [ + + client.driver.execute_query.return_value = [ [{"node": "dummy-node", "score": 1.0}], None, None, @@ -119,9 +106,11 @@ def test_similarity_search_vector_happy_path(driver, client): YIELD node, score """ - client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + records = client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) - driver.execute_query.assert_called_once_with( + custom_embeddings.embed_query.assert_not_called() + + client.driver.execute_query.assert_called_once_with( search_query, { "index_name": index_name, @@ -130,36 +119,47 @@ def test_similarity_search_vector_happy_path(driver, client): }, ) + assert records == [Neo4jRecord(node="dummy-node", score=1.0)] + + +@patch("neo4j_genai.GenAIClient._verify_version") +def test_similarity_search_text_happy_path(_verify_version_mock, driver): + embed_query_vector = [1.0 for _ in range(1536)] + custom_embeddings = MagicMock() + custom_embeddings.embed_query.return_value = embed_query_vector + + client = GenAIClient(driver, custom_embeddings) -def test_similarity_search_text_happy_path(driver, client_with_embedder): - client, embedder = client_with_embedder index_name = "my-index" query_text = "may thy knife chip and shatter" - dimensions = 1536 - query_vector = [1.0 for _ in range(dimensions)] top_k = 5 + driver.execute_query.return_value = [ [{"node": "dummy-node", "score": 1.0}], None, None, ] - embedder.set_dimension(dimensions) + search_query = """ CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) YIELD node, score """ - client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) + records = client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) + + custom_embeddings.embed_query.assert_called_once_with(query_text) driver.execute_query.assert_called_once_with( search_query, { "index_name": index_name, "top_k": top_k, - "query_vector": query_vector, + "query_vector": embed_query_vector, }, ) + assert records == [Neo4jRecord(node="dummy-node", score=1.0)] + def test_similarity_search_missing_embedder_for_text(client): index_name = "my-index" From b57c646c187fcfb2102aba2b9b6a391e5427ac0e Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Fri, 15 Mar 2024 16:39:46 +0000 Subject: [PATCH 7/9] Add validation error test --- tests/test_client.py | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) diff --git a/tests/test_client.py b/tests/test_client.py index 6d5714ec..8d061490 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -185,3 +185,38 @@ def test_similarity_search_both_text_and_vector(client): query_vector=query_vector, top_k=top_k, ) + +@patch("neo4j_genai.GenAIClient._verify_version") +def test_similarity_search_vector_bad_results(_verify_version_mock, driver): + custom_embeddings = MagicMock() + + client = GenAIClient(driver, custom_embeddings) + + index_name = "my-index" + dimensions = 1536 + query_vector = [1.0 for _ in range(dimensions)] + top_k = 5 + + client.driver.execute_query.return_value = [ + [{"node": "dummy-node", "score": "adsa"}], + None, + None, + ] + search_query = """ + CALL db.index.vector.queryNodes($index_name, $top_k, $query_vector) + YIELD node, score + """ + + with pytest.raises(ValueError): + client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + + custom_embeddings.embed_query.assert_not_called() + + client.driver.execute_query.assert_called_once_with( + search_query, + { + "index_name": index_name, + "top_k": top_k, + "query_vector": query_vector, + }, + ) From 385ce00d9109835925b5b880ff7f15e70331b375 Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Mon, 18 Mar 2024 09:38:41 +0000 Subject: [PATCH 8/9] Remove exclude property from pyproject.toml --- pyproject.toml | 2 -- 1 file changed, 2 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index d97f6f49..b9f9e7da 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -29,8 +29,6 @@ ruff = "^0.3.0" requires = ["poetry-core"] build-backend = "poetry.core.masonry.api" -exclude = ["**/tests/"] - [tool.pytest.ini_options] testpaths = ["tests"] filterwarnings = [ From 9af38027cdb55e6b4a60dd309bcec75898886846 Mon Sep 17 00:00:00 2001 From: Jon Besga Date: Mon, 18 Mar 2024 09:45:51 +0000 Subject: [PATCH 9/9] Bump version and lint --- .github/workflows/publish.yaml | 7 +++---- pyproject.toml | 2 +- tests/conftest.py | 2 -- tests/test_client.py | 15 ++++++++++++--- 4 files changed, 16 insertions(+), 10 deletions(-) diff --git a/.github/workflows/publish.yaml b/.github/workflows/publish.yaml index 1e3da16b..428b1916 100644 --- a/.github/workflows/publish.yaml +++ b/.github/workflows/publish.yaml @@ -1,4 +1,4 @@ -name: Publish Python 🐍 distribution 📦 to PyPI and TestPyPI +name: Publish Python 🐍 distribution 📦 on: push @@ -28,8 +28,7 @@ jobs: path: dist/ publish-to-pypi: - name: >- - Publish Python 🐍 distribution 📦 to PyPI + name: PyPI if: startsWith(github.ref, 'refs/tags/') needs: - build @@ -49,7 +48,7 @@ jobs: uses: pypa/gh-action-pypi-publish@release/v1 publish-to-testpypi: - name: Publish Python 🐍 distribution 📦 to TestPyPI + name: TestPyPI needs: - build runs-on: ubuntu-latest diff --git a/pyproject.toml b/pyproject.toml index b9f9e7da..f09a8ed7 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "neo4j-genai" -version = "0.1.0" +version = "0.1.1" description = "Python package to allow easy integration to Neo4j's GenAI features" authors = ["Neo4j, Inc "] license = "Apache License, Version 2.0" diff --git a/tests/conftest.py b/tests/conftest.py index 07992d05..b3331dec 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,8 +1,6 @@ import pytest from neo4j_genai import GenAIClient from unittest.mock import MagicMock, patch -from typing import List -from neo4j_genai.embedder import Embedder @pytest.fixture diff --git a/tests/test_client.py b/tests/test_client.py index 8d061490..53f14314 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -3,6 +3,7 @@ from neo4j_genai.types import Neo4jRecord from unittest.mock import patch, MagicMock + def test_genai_client_supported_aura_version(driver): driver.execute_query.return_value = [[{"versions": ["5.11-aura"]}], None, None] @@ -85,6 +86,7 @@ def test_drop_index(client): {"name": "my-index"}, ) + @patch("neo4j_genai.GenAIClient._verify_version") def test_similarity_search_vector_happy_path(_verify_version_mock, driver): custom_embeddings = MagicMock() @@ -106,7 +108,9 @@ def test_similarity_search_vector_happy_path(_verify_version_mock, driver): YIELD node, score """ - records = client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + records = client.similarity_search( + name=index_name, query_vector=query_vector, top_k=top_k + ) custom_embeddings.embed_query.assert_not_called() @@ -145,7 +149,9 @@ def test_similarity_search_text_happy_path(_verify_version_mock, driver): YIELD node, score """ - records = client.similarity_search(name=index_name, query_text=query_text, top_k=top_k) + records = client.similarity_search( + name=index_name, query_text=query_text, top_k=top_k + ) custom_embeddings.embed_query.assert_called_once_with(query_text) @@ -186,6 +192,7 @@ def test_similarity_search_both_text_and_vector(client): top_k=top_k, ) + @patch("neo4j_genai.GenAIClient._verify_version") def test_similarity_search_vector_bad_results(_verify_version_mock, driver): custom_embeddings = MagicMock() @@ -208,7 +215,9 @@ def test_similarity_search_vector_bad_results(_verify_version_mock, driver): """ with pytest.raises(ValueError): - client.similarity_search(name=index_name, query_vector=query_vector, top_k=top_k) + client.similarity_search( + name=index_name, query_vector=query_vector, top_k=top_k + ) custom_embeddings.embed_query.assert_not_called()