From 74f94c0dded7d87fe8ab3687ded25339ee2b3d26 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Thu, 30 May 2024 20:41:05 -0500 Subject: [PATCH] AUTH-421 Fargate FastAPI container (#15287) --- .eslintignore | 1 + opentrons-ai-server/.gitignore | 6 +- opentrons-ai-server/Dockerfile | 17 +- opentrons-ai-server/Makefile | 10 +- opentrons-ai-server/Pipfile | 21 +- opentrons-ai-server/Pipfile.lock | 1132 +++++++++++++---- opentrons-ai-server/README.md | 43 +- .../api/domain/openai_predict.py | 28 +- opentrons-ai-server/api/domain/prompts.py | 12 +- opentrons-ai-server/api/handler/fast.py | 252 ++++ opentrons-ai-server/api/handler/function.py | 82 -- opentrons-ai-server/api/handler/local_run.py | 9 + .../api/handler/logging_config.py | 31 + opentrons-ai-server/api/integration/auth.py | 60 + .../api/models/chat_request.py | 17 +- opentrons-ai-server/api/settings.py | 92 +- opentrons-ai-server/deploy.py | 200 ++- opentrons-ai-server/pytest.ini | 2 +- opentrons-ai-server/tests/conftest.py | 2 +- opentrons-ai-server/tests/helpers/client.py | 60 +- opentrons-ai-server/tests/helpers/settings.py | 58 +- .../tests/helpers/token_verifier.py | 4 +- opentrons-ai-server/tests/test_chat_models.py | 4 +- opentrons-ai-server/tests/test_live.py | 12 +- 24 files changed, 1597 insertions(+), 558 deletions(-) create mode 100644 opentrons-ai-server/api/handler/fast.py delete mode 100644 opentrons-ai-server/api/handler/function.py create mode 100644 opentrons-ai-server/api/handler/local_run.py create mode 100644 opentrons-ai-server/api/handler/logging_config.py create mode 100644 opentrons-ai-server/api/integration/auth.py diff --git a/.eslintignore b/.eslintignore index e45bc680540..054e695eb4d 100644 --- a/.eslintignore +++ b/.eslintignore @@ -33,3 +33,4 @@ app-testing/files # app testing don't format the snapshots app-testing/tests/__snapshots__ opentrons-ai-server/package +opentrons-ai-server/api/storage/index/ diff --git a/opentrons-ai-server/.gitignore b/opentrons-ai-server/.gitignore index 78bcfc1a90b..8608492bce5 100644 --- a/opentrons-ai-server/.gitignore +++ b/opentrons-ai-server/.gitignore @@ -1,4 +1,4 @@ -.env +*.env results package function.zip @@ -6,4 +6,8 @@ requirements.txt test.env cached_token.txt tests/helpers/cached_token.txt +tests/helpers/prod_cached_token.txt +tests/helpers/staging_cached_token.txt +tests/helpers/*_cached_token.txt tests/helpers/test.env +tests/helpers/*.env diff --git a/opentrons-ai-server/Dockerfile b/opentrons-ai-server/Dockerfile index 8a4ee3b944d..76eb78f172a 100644 --- a/opentrons-ai-server/Dockerfile +++ b/opentrons-ai-server/Dockerfile @@ -1,13 +1,16 @@ -FROM --platform=linux/amd64 public.ecr.aws/lambda/python:3.12 +FROM --platform=linux/amd64 python:3.12-slim -COPY --from=public.ecr.aws/datadog/lambda-extension:57 /opt/. /opt/ +ENV PYTHONUNBUFFERED True +ENV DOCKER_RUNNING True -WORKDIR ${LAMBDA_TASK_ROOT} +WORKDIR /code -COPY requirements.txt . +COPY ./requirements.txt /code/requirements.txt -RUN pip install -r requirements.txt +RUN pip install --no-cache-dir --upgrade -r /code/requirements.txt -COPY api ${LAMBDA_TASK_ROOT}/api +COPY ./api /code/api -CMD [ "datadog_lambda.handler.handler" ] \ No newline at end of file +EXPOSE 8000 + +CMD ["ddtrace-run", "uvicorn", "api.handler.fast:app", "--proxy-headers", "--host", "0.0.0.0", "--port", "8000", "--timeout-keep-alive", "190", "--workers", "3"] diff --git a/opentrons-ai-server/Makefile b/opentrons-ai-server/Makefile index 48c18c9f6ff..1c16c6e06d0 100644 --- a/opentrons-ai-server/Makefile +++ b/opentrons-ai-server/Makefile @@ -67,13 +67,17 @@ gen-requirements: @echo "Generating requirements.txt from Pipfile.lock..." python -m pipenv requirements > requirements.txt -ENV ?= sandbox +ENV ?= local .PHONY: deploy deploy: gen-requirements @echo "Deploying to environment: $(ENV)" python -m pipenv run python deploy.py --env $(ENV) +.PHONY: prompted-deploy +prompted-deploy: gen-requirements + python -m pipenv run python deploy.py + .PHONY: direct-chat-completion direct-chat-completion: python -m pipenv run python -m api.domain.openai_predict @@ -89,3 +93,7 @@ live-client: .PHONY: test-live test-live: python -m pipenv run python -m pytest tests -m live --env $(ENV) + +.PHONY: run +run: + python -m pipenv run python -m api.handler.local_run --env $(ENV) \ No newline at end of file diff --git a/opentrons-ai-server/Pipfile b/opentrons-ai-server/Pipfile index 2ca1c97821d..8903b0637d7 100644 --- a/opentrons-ai-server/Pipfile +++ b/opentrons-ai-server/Pipfile @@ -4,26 +4,29 @@ verify_ssl = true name = "pypi" [packages] -openai = "==1.25.1" +openai = "==1.30.4" python-dotenv = "==1.0.1" httpx = "==0.27.0" llama-index = "==0.10.24" -datadog-lambda = "==5.94.0" pydantic = "==2.7.1" +fastapi = "==0.111.0" +ddtrace = "==2.8.5" +pydantic-settings = "==2.2.1" +pyjwt = {extras = ["crypto"], version = "*"} +python-json-logger = "==2.0.7" [dev-packages] -docker = "==7.0.0" -pytest = "==8.2.0" -ruff = "==0.4.2" +docker = "==7.1.0" +pytest = "==8.2.1" +ruff = "==0.4.6" mypy = "==1.10.0" black = "==24.4.2" types-requests = "*" -boto3 = "==1.34.97" -boto3-stubs = "==1.34.97" +boto3 = "==1.34.114" +boto3-stubs = "==1.34.114" rich = "==13.7.1" -pyjwt = "==2.8.0" cryptography = "==42.0.7" -types-docker = "==7.0.0.20240513" +types-docker = "==7.0.0.20240528" [requires] python_version = "3.12" diff --git a/opentrons-ai-server/Pipfile.lock b/opentrons-ai-server/Pipfile.lock index 00922b74257..6af3bd4a246 100644 --- a/opentrons-ai-server/Pipfile.lock +++ b/opentrons-ai-server/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "15df1bc32acf890b1eb461ea6e3162be9abcd9d5a6124c81da13eb6201061db9" + "sha256": "799a525d134285522d6a1ea338a23ef883c0fad06c9d0d42cee8828d8df703b8" }, "pipfile-spec": 6, "requires": { @@ -109,19 +109,19 @@ }, "annotated-types": { "hashes": [ - "sha256:0641064de18ba7a25dee8f96403ebc39113d0cb953a01429249d5c7564666a43", - "sha256:563339e807e53ffd9c267e99fc6d9ea23eb8443c08f112651963e24e22f84a5d" + "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53", + "sha256:aff07c09a53a08bc8cfccb9c85b05f1aa9a2a6f23728d790723543408344ce89" ], "markers": "python_version >= '3.8'", - "version": "==0.6.0" + "version": "==0.7.0" }, "anyio": { "hashes": [ - "sha256:048e05d0f6caeed70d731f3db756d35dcc1f35747c8c403364a8332c630441b8", - "sha256:f75253795a87df48568485fd18cdd2a3fa5c4f7c5be8e5e36637733fce06fed6" + "sha256:5aadc6a1bbb7cdb0bede386cac5e2940f5e2ff3aa20277e991cf028e0585ce94", + "sha256:c1b2d8f46a8a812513012e1107cb0e68c17159a7a594208005a57dc776e1bdc7" ], "markers": "python_version >= '3.8'", - "version": "==4.3.0" + "version": "==4.4.0" }, "attrs": { "hashes": [ @@ -163,6 +163,64 @@ "markers": "python_version >= '3.6'", "version": "==2024.2.2" }, + "cffi": { + "hashes": [ + "sha256:0c9ef6ff37e974b73c25eecc13952c55bceed9112be2d9d938ded8e856138bcc", + "sha256:131fd094d1065b19540c3d72594260f118b231090295d8c34e19a7bbcf2e860a", + "sha256:1b8ebc27c014c59692bb2664c7d13ce7a6e9a629be20e54e7271fa696ff2b417", + "sha256:2c56b361916f390cd758a57f2e16233eb4f64bcbeee88a4881ea90fca14dc6ab", + "sha256:2d92b25dbf6cae33f65005baf472d2c245c050b1ce709cc4588cdcdd5495b520", + "sha256:31d13b0f99e0836b7ff893d37af07366ebc90b678b6664c955b54561fc36ef36", + "sha256:32c68ef735dbe5857c810328cb2481e24722a59a2003018885514d4c09af9743", + "sha256:3686dffb02459559c74dd3d81748269ffb0eb027c39a6fc99502de37d501faa8", + "sha256:582215a0e9adbe0e379761260553ba11c58943e4bbe9c36430c4ca6ac74b15ed", + "sha256:5b50bf3f55561dac5438f8e70bfcdfd74543fd60df5fa5f62d94e5867deca684", + "sha256:5bf44d66cdf9e893637896c7faa22298baebcd18d1ddb6d2626a6e39793a1d56", + "sha256:6602bc8dc6f3a9e02b6c22c4fc1e47aa50f8f8e6d3f78a5e16ac33ef5fefa324", + "sha256:673739cb539f8cdaa07d92d02efa93c9ccf87e345b9a0b556e3ecc666718468d", + "sha256:68678abf380b42ce21a5f2abde8efee05c114c2fdb2e9eef2efdb0257fba1235", + "sha256:68e7c44931cc171c54ccb702482e9fc723192e88d25a0e133edd7aff8fcd1f6e", + "sha256:6b3d6606d369fc1da4fd8c357d026317fbb9c9b75d36dc16e90e84c26854b088", + "sha256:748dcd1e3d3d7cd5443ef03ce8685043294ad6bd7c02a38d1bd367cfd968e000", + "sha256:7651c50c8c5ef7bdb41108b7b8c5a83013bfaa8a935590c5d74627c047a583c7", + "sha256:7b78010e7b97fef4bee1e896df8a4bbb6712b7f05b7ef630f9d1da00f6444d2e", + "sha256:7e61e3e4fa664a8588aa25c883eab612a188c725755afff6289454d6362b9673", + "sha256:80876338e19c951fdfed6198e70bc88f1c9758b94578d5a7c4c91a87af3cf31c", + "sha256:8895613bcc094d4a1b2dbe179d88d7fb4a15cee43c052e8885783fac397d91fe", + "sha256:88e2b3c14bdb32e440be531ade29d3c50a1a59cd4e51b1dd8b0865c54ea5d2e2", + "sha256:8f8e709127c6c77446a8c0a8c8bf3c8ee706a06cd44b1e827c3e6a2ee6b8c098", + "sha256:9cb4a35b3642fc5c005a6755a5d17c6c8b6bcb6981baf81cea8bfbc8903e8ba8", + "sha256:9f90389693731ff1f659e55c7d1640e2ec43ff725cc61b04b2f9c6d8d017df6a", + "sha256:a09582f178759ee8128d9270cd1344154fd473bb77d94ce0aeb2a93ebf0feaf0", + "sha256:a6a14b17d7e17fa0d207ac08642c8820f84f25ce17a442fd15e27ea18d67c59b", + "sha256:a72e8961a86d19bdb45851d8f1f08b041ea37d2bd8d4fd19903bc3083d80c896", + "sha256:abd808f9c129ba2beda4cfc53bde801e5bcf9d6e0f22f095e45327c038bfe68e", + "sha256:ac0f5edd2360eea2f1daa9e26a41db02dd4b0451b48f7c318e217ee092a213e9", + "sha256:b29ebffcf550f9da55bec9e02ad430c992a87e5f512cd63388abb76f1036d8d2", + "sha256:b2ca4e77f9f47c55c194982e10f058db063937845bb2b7a86c84a6cfe0aefa8b", + "sha256:b7be2d771cdba2942e13215c4e340bfd76398e9227ad10402a8767ab1865d2e6", + "sha256:b84834d0cf97e7d27dd5b7f3aca7b6e9263c56308ab9dc8aae9784abb774d404", + "sha256:b86851a328eedc692acf81fb05444bdf1891747c25af7529e39ddafaf68a4f3f", + "sha256:bcb3ef43e58665bbda2fb198698fcae6776483e0c4a631aa5647806c25e02cc0", + "sha256:c0f31130ebc2d37cdd8e44605fb5fa7ad59049298b3f745c74fa74c62fbfcfc4", + "sha256:c6a164aa47843fb1b01e941d385aab7215563bb8816d80ff3a363a9f8448a8dc", + "sha256:d8a9d3ebe49f084ad71f9269834ceccbf398253c9fac910c4fd7053ff1386936", + "sha256:db8e577c19c0fda0beb7e0d4e09e0ba74b1e4c092e0e40bfa12fe05b6f6d75ba", + "sha256:dc9b18bf40cc75f66f40a7379f6a9513244fe33c0e8aa72e2d56b0196a7ef872", + "sha256:e09f3ff613345df5e8c3667da1d918f9149bd623cd9070c983c013792a9a62eb", + "sha256:e4108df7fe9b707191e55f33efbcb2d81928e10cea45527879a4749cbe472614", + "sha256:e6024675e67af929088fda399b2094574609396b1decb609c55fa58b028a32a1", + "sha256:e70f54f1796669ef691ca07d046cd81a29cb4deb1e5f942003f401c0c4a2695d", + "sha256:e715596e683d2ce000574bae5d07bd522c781a822866c20495e52520564f0969", + "sha256:e760191dd42581e023a68b758769e2da259b5d52e3103c6060ddc02c9edb8d7b", + "sha256:ed86a35631f7bfbb28e108dd96773b9d5a6ce4811cf6ea468bb6a359b256b1e4", + "sha256:ee07e47c12890ef248766a6e55bd38ebfb2bb8edd4142d56db91b21ea68b7627", + "sha256:fa3a0128b152627161ce47201262d3140edb5a5c3da88d73a1b790a959126956", + "sha256:fcc8eb6d5902bb1cf6dc4f187ee3ea80a1eba0a89aba40a5cb20a5087d961357" + ], + "markers": "platform_python_implementation != 'PyPy'", + "version": "==1.16.0" + }, "charset-normalizer": { "hashes": [ "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027", @@ -267,6 +325,43 @@ "markers": "python_version >= '3.7'", "version": "==8.1.7" }, + "cryptography": { + "hashes": [ + "sha256:02c0eee2d7133bdbbc5e24441258d5d2244beb31da5ed19fbb80315f4bbbff55", + "sha256:0d563795db98b4cd57742a78a288cdbdc9daedac29f2239793071fe114f13785", + "sha256:16268d46086bb8ad5bf0a2b5544d8a9ed87a0e33f5e77dd3c3301e63d941a83b", + "sha256:1a58839984d9cb34c855197043eaae2c187d930ca6d644612843b4fe8513c886", + "sha256:2954fccea107026512b15afb4aa664a5640cd0af630e2ee3962f2602693f0c82", + "sha256:2e47577f9b18723fa294b0ea9a17d5e53a227867a0a4904a1a076d1646d45ca1", + "sha256:31adb7d06fe4383226c3e963471f6837742889b3c4caa55aac20ad951bc8ffda", + "sha256:3577d029bc3f4827dd5bf8bf7710cac13527b470bbf1820a3f394adb38ed7d5f", + "sha256:36017400817987670037fbb0324d71489b6ead6231c9604f8fc1f7d008087c68", + "sha256:362e7197754c231797ec45ee081f3088a27a47c6c01eff2ac83f60f85a50fe60", + "sha256:3de9a45d3b2b7d8088c3fbf1ed4395dfeff79d07842217b38df14ef09ce1d8d7", + "sha256:4f698edacf9c9e0371112792558d2f705b5645076cc0aaae02f816a0171770fd", + "sha256:5482e789294854c28237bba77c4c83be698be740e31a3ae5e879ee5444166582", + "sha256:5e44507bf8d14b36b8389b226665d597bc0f18ea035d75b4e53c7b1ea84583cc", + "sha256:779245e13b9a6638df14641d029add5dc17edbef6ec915688f3acb9e720a5858", + "sha256:789caea816c6704f63f6241a519bfa347f72fbd67ba28d04636b7c6b7da94b0b", + "sha256:7f8b25fa616d8b846aef64b15c606bb0828dbc35faf90566eb139aa9cff67af2", + "sha256:8cb8ce7c3347fcf9446f201dc30e2d5a3c898d009126010cbd1f443f28b52678", + "sha256:93a3209f6bb2b33e725ed08ee0991b92976dfdcf4e8b38646540674fc7508e13", + "sha256:a3a5ac8b56fe37f3125e5b72b61dcde43283e5370827f5233893d461b7360cd4", + "sha256:a47787a5e3649008a1102d3df55424e86606c9bae6fb77ac59afe06d234605f8", + "sha256:a79165431551042cc9d1d90e6145d5d0d3ab0f2d66326c201d9b0e7f5bf43604", + "sha256:a987f840718078212fdf4504d0fd4c6effe34a7e4740378e59d47696e8dfb477", + "sha256:a9bc127cdc4ecf87a5ea22a2556cab6c7eda2923f84e4f3cc588e8470ce4e42e", + "sha256:bd13b5e9b543532453de08bcdc3cc7cebec6f9883e886fd20a92f26940fd3e7a", + "sha256:c65f96dad14f8528a447414125e1fc8feb2ad5a272b8f68477abbcc1ea7d94b9", + "sha256:d8e3098721b84392ee45af2dd554c947c32cc52f862b6a3ae982dbb90f577f14", + "sha256:e6b79d0adb01aae87e8a44c2b64bc3f3fe59515280e00fb6d57a7267a2583cda", + "sha256:e6b8f1881dac458c34778d0a424ae5769de30544fc678eac51c1c8bb2183e9da", + "sha256:e9b2a6309f14c0497f348d08a065d52f3020656f675819fc405fb63bbcd26562", + "sha256:ecbfbc00bf55888edda9868a4cf927205de8499e7fabe6c050322298382953f2", + "sha256:efd0bf5205240182e0f13bcaea41be4fdf5c22c5129fc7ced4a0282ac86998c9" + ], + "version": "==42.0.7" + }, "dataclasses-json": { "hashes": [ "sha256:0c09827d26fffda27f1be2fed7a7a01a29c5ddcd2eb6393ad5ebf9d77e9deae8", @@ -275,23 +370,6 @@ "markers": "python_version >= '3.7' and python_version < '4.0'", "version": "==0.6.6" }, - "datadog": { - "hashes": [ - "sha256:4a56d57490ea699a0dfd9253547485a57b4120e93489defadcf95c66272374d6", - "sha256:4cb7a7991af6cadb868fe450cd456473e65f11fc678b7d7cf61044ff1c6074d8" - ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5, 3.6'", - "version": "==0.49.1" - }, - "datadog-lambda": { - "hashes": [ - "sha256:2005c09351f0c10da63fd29d1f43d035c4c5c6a71492416817741536a6e45896", - "sha256:de8e9a40b4dbee3314bfc1c2c91d071691a78e324a041dcb07bf52754ead3e10" - ], - "index": "pypi", - "markers": "python_version < '4' and python_full_version >= '3.8.0'", - "version": "==5.94.0" - }, "ddsketch": { "hashes": [ "sha256:6d047b455fe2837c43d366ff1ae6ba0c3166e15499de8688437a75cea914224e", @@ -302,69 +380,70 @@ }, "ddtrace": { "hashes": [ - "sha256:00810ecbf1af1e288435f7ac4f6198073229d1b61977b728b4a241c839190b04", - "sha256:03a204b5760e1b377210b0a301beb6b2775dcb82f8e1cdf10a7921d3a6357aa9", - "sha256:05c7587ceab5a91624a687d02a17a17019e867a5ceb08516e6616183e0c089c3", - "sha256:069ff3b9e34d9d3dfd256ba56f3eedc335068386d8e0e7422d7d981017283161", - "sha256:0e9c048d9894a556996a66e039c4d5bcb2f423983ead616cc0741d2fbf6ca57b", - "sha256:1392de7f7e0cfd0071f90438f8eab0cbf8c669d94f4c09426ad3ad80ab55c0ff", - "sha256:15592f3c2a64b3b5ae3eafba4b73a9e7c1724bcabd68b8d507c7a521858c66e0", - "sha256:159321423be46c38687689143dcbe941b00bd771a73011de05f6eff2a3ef732a", - "sha256:29060a48904c8a19f2dcac9dc7b9a84bf773bffa8ae1ba5cccc8a5bb74b238a0", - "sha256:2ac88197694d152ba6434bb89cde352e6356f34cf1a63071f72f7aa278ea7f55", - "sha256:35c9274db4a0b4d9dab79329fb0df2325b47e5f804ab834d3d79864650197d37", - "sha256:36e73afb37bc724793bc3f95c6fe5dfea3e0cfe939caf5ceff9bbbef0e3dac83", - "sha256:455ef7df497cf26cd81740045b4702f94748fe16624409409511a48601b031d3", - "sha256:4aac5b75f03a8872472cd8e7fa7ebe3ff8614458174c619a7d3be8ca8d8d9f5b", - "sha256:4d91b9d0b5d9fef3b1a3c78a98f3dca7708b78dd7cf04f6bb78c898b8d401c5e", - "sha256:51d55d553f848bac57b5b4b8818cc6b1c9a97ab7b77c8809453498699ba92d65", - "sha256:58cf8c5145151ff19f94e8fb99aaf581e5180e87c5cb4e6765806e45dc473077", - "sha256:5d1c74178eee1c4a8508c93819ca996439d13f159f8202190f7705509edafe4b", - "sha256:6580ddf25a9ed4a190e49565d843700710820a9dc5ff1f3ccbf6a159ed5e8521", - "sha256:65f344aac5073e1fd74ff4ba7b404c8400d9165b715d4b40b3f10a62468fc957", - "sha256:6a612efe745abd01cb89c0b49e2f7b7adaaf075975b095426e361bcb8415ca51", - "sha256:6c543750759f5ad91ab280a034b8daca0bf52527b0ee69e4ca5ba2e6d7ce207c", - "sha256:76f853ce8d6b1fcfb5211a5589f3f8610d92d669c6b9975af7b1d09e8197ee92", - "sha256:814bfcc447c7ed3acab0e07f1561b41853466a6a009d94f745bf1df39c085afa", - "sha256:819ea545b9fb8e51c0ab3553aefbb3ad8577454129ce9c9ea538146001ff2a19", - "sha256:85c881c943c4228faea04f1982c0fad722d6e2b5317fed4ceff64b3a9fb849eb", - "sha256:88202985f5d0230536e25d30c206fa883c8be24a01ae6a4caf1085633ba613d1", - "sha256:890348d2c7863dd00207412879b39f9524cbd6053d2a034e653996cb65bcd10a", - "sha256:8ef07d1ae9a29b9639eaadeefead70b8c1d99d9a6cb9b8fe837eb312642b3419", - "sha256:8f4a7cdf28c46e9c697e9dcd7b1787dc657bf1af610e4bab7041f80b4cf9962a", - "sha256:8ff3540e6430812d5178e5d8b42d7712301d4d68412fe3346f2cf6263173464e", - "sha256:948427c85cbbc0d4ebd77a81e07601cd54eba3b2056293e6f35815f3d824f1c3", - "sha256:9fd98f860c7a5821ce905d16635103b21a03302605acc2002dbcc2eb7848bf20", - "sha256:a018fae0a0be1eaa44059dcd87ecfa10bcb95cfa03c89f7bf7a06b7bc869d52c", - "sha256:a022870b41836d85583d5bfad2b9e82b9cae8193b18e18e1341b7bb87f1617b0", - "sha256:a0dfa90232408a30c5618141845fbd32295d2252467349922b7e26d040082f08", - "sha256:a6a80cf6059383823844420609b3c6d286b6b0010d883893a8e8fad4585ead61", - "sha256:aa791627fe1704ef7bc5416d4a4dfc43c95d52888b0db455ae568a75c63c93e2", - "sha256:b09329b8aec540dc3a8e2ee08ed21367977d299d423d21cbafebd8eaeb3eb5a6", - "sha256:b559ac4e790771cf48527bd9344437390a0129846e6087e582e4705a1131d4ec", - "sha256:b88610aa191efbdcda23a87e8a3f2281b5b299894fe884f40248449ef42f77ca", - "sha256:ba2f8d6b160e31b828df499cdb3f4635e538a566c39090557b72ac55779fae34", - "sha256:bfbb3c37b1bcd132619accae98a37a9c7425ed040b0da88c51b64ebbb18f956b", - "sha256:c05cd7675d0a15496cf0de34b4389683bf29e2b14ebde23a18fba656a51cd695", - "sha256:c25b9e5a7c8bf7bf2dd554af1b8dd3cb45d3e9ecb66acac6b73a61e6a19d2355", - "sha256:c507aabca054b6167c3f103dd12bbd594cc283e7272c13d3a8f1098528869cfd", - "sha256:c551b8788d9c9714963d77e54d1f1ff368022169d9ed6d3b1ea7cb0d660cd7d4", - "sha256:c675d91e4dbb4319ae6eb9537141c95d7fa4014f231b55f6240de6c6bc3d5ede", - "sha256:cbcdc34e20a20f18d6deca2099dfc47e30f1ef00b0060c486016b005c88c42b0", - "sha256:d0aeb72b3300bb146421934bf7f44cbf4c5fa436cbb86cb615de995d914e7b22", - "sha256:d56ce1faa85a7456694fa61a0fc4e4853ac5dd1316f7d8315592deddea9ee4e0", - "sha256:dab563872e32fb1e482d69194b7ecd1fdec5261140250218acd91872f8471cc8", - "sha256:dac86f13b32e00249f4743d7fbdfe47ad72c8b958c4a967f7677e85c5258d844", - "sha256:dc58a7079621dc6a6c28d5c64866aa017caca7f64af2b919b5d8304cdec063f7", - "sha256:dee692c1c5d667cbebbfe4aa216f112c74c4dda07c29523e9d9832c141830376", - "sha256:e1649afa71e00869d91f087fbd90dbeb7ed666ed958ef1d37fea8c70c3349c5f", - "sha256:e1a5fdd1ecb6b3f7282917f34f28ec8d81dc629a14a56e643b93faaa5918a4d2", - "sha256:e24b2f5117e7f9fa50c06ea95cafa0bb61dda78fcea7fd2758ef2358ecf2959c", - "sha256:e485740d85d37ee52ec7fa0f10ef9acc4c8580d2bdec14371bf3acddd0b1ba1c", - "sha256:f3d8398e1a1f931d153b457d312d1a5e1a1b56aa15839c3dd03403fa66364007" + "sha256:0674abd6118382f82c7139bab7de106644b260f09ec43f56952ae36b9840f191", + "sha256:0c7f9506fc33448a91d706ee838b712ac76a00dc2213c96ddc62e25ce1a3708e", + "sha256:0cb3bc95ca4dae45531ad59d859712ccd52a465f19b0e03d3703e1c9fcec8614", + "sha256:0ecc3d65adeac43e2ec3f9e4e9ce2a166230a7c7f6e5913d9b80753983a414f1", + "sha256:11e055acbc9de6d0d2fa26d9ca97a0256ede9ff36199419bce23837764c84559", + "sha256:1247855a6465c2fc167977d6bd8497c8b28e177530a12faad16fd4f3c31a4dac", + "sha256:1312acce7049d72a7e63110525b30c1bca192cf8dbffbca5fc626e6268a850ec", + "sha256:1731aa5685161118696a439891bedb31a9c6bfa3233acca076dbed3f3caf239e", + "sha256:1bcedeca9c60e6eb9dfb3654bdce383161322b1261ab6d1cf067f43451cf81c5", + "sha256:2379716ec9ddc0c8f38c48a0b19e2116c167eb0de34436a48d348351dde60dbe", + "sha256:2c08eb99996e79708948d7eb547c02888daa64a7f2e220ba8ff6144e63996a8c", + "sha256:32b5c067e1a907202f25e6376d9e676644edcf16cae7beb9ac1087a067833393", + "sha256:33c5a3e7c3fe82be8316b7264bca45627cc105c9b8f72eef6b6cf9dea6102ca1", + "sha256:376ac38cd9de471842a5b395005deed1d7ca7448b7dd66402bf108598c996e0d", + "sha256:3a1d029713e6fa6887c4fb3e51e1a75c39855270edafce2dbb8678d168d3272a", + "sha256:3e8b797f34a7475f57f251d6f6c53b5c72d6964351e1c1b147efc55844153697", + "sha256:51ef36d884cb508166c3699ec8fd01ac32749bea2e826953e697658697e351f4", + "sha256:521c0d6c972d51a887415cccf6c8ad49761fc7ce7af4f70794648429c9aa6ea8", + "sha256:5893d1b9b5ae0cca73977ad5056abaf7afb9bcd4e8c92d6abbe6208925a29027", + "sha256:5b6ceae3201c77b08801f55a751cf0f1b85d250cfac6dfd4a9f0663c131a916d", + "sha256:5ddebfcef09e84b9c6efa58036576347243946473a6db4f143f4b60d9f5e6b34", + "sha256:6429395c9f380ff668d8ec4914a3eea095bfa8cc243b07939fbf5d3a3c68e919", + "sha256:6c45f818c08eab1c6898ea8263422cf19e2425bf725e1843c094c5628baac589", + "sha256:7360f43edec29f5ded3b9f3eb3b549997fb84484b9f3f33bb97a37dfd13266a0", + "sha256:77be9629d7123636a7559bf49bc3a9255febcea22685214c441ace28d3bf1e6b", + "sha256:7b7eb223d7c000daa69d7d82e36af254f8e49e154b4713d842b60b2d4cbc250c", + "sha256:7be8dfa8104bc95a06b484bf8d04374e6f081c2158b6718fa0a955813fd530ec", + "sha256:7dca42e52f74416b1085664dc3412be01cc27e50218cb8e7f9c46473fdecdb2d", + "sha256:7f77edf558f3548c55cacca1402ee85a371f2522423f54e1e92a880933fbaaf6", + "sha256:8152cb0a87ed63177c0ee2a51a5d43dfe7663e398c86d7c46129a02c21456c9c", + "sha256:88dd0f886fd02c215f6017460a088695b55d945c61171c2e7897728572ca2616", + "sha256:901d47e455cb197ee12604a10b5d10ee74e0b56a5e0a50092e19a654976923dc", + "sha256:9430e3197f4e56cce4a2b0829faddf4c4efde6588fea66fb4aa3adc4946cae0c", + "sha256:982adfd87ed1d45e8f98b6ac37be0e4a6d09f8fa550d6f3c0ee2bd8d270cf55f", + "sha256:9a780f6277c414bc8f004937c0d293b682036905de31aa2edbdf89eaecf1a1d4", + "sha256:9b5cfb81984c74a60cdb3c167203a4bbeaa319b5242e17b5ca0adbee4df54a7f", + "sha256:9df3b59794352d3af6df52278e724999dc3b70e7cf59142e3c6170dd43715a6c", + "sha256:9e430bcc2c806d6dd249ec5281c38c2507d58ca903d56df1cac171fac4bd082a", + "sha256:9ec3c24d1a7ea7e1ac5c8251d074b7c053e60f3060f335fb88e612fe35dbbe54", + "sha256:a6f2cf37088a5e528d1f78d571cec30f9b1cc03cf1097bf609f7f7c13a3aada8", + "sha256:a6f67d1c9dec80469921424272a83b5cad57e4b26ae750fec4b763f1bf6c4ece", + "sha256:adcb83109ac918e233c3c3d825a3bbc82c8fd1e45d632466bd15a21122ed2b81", + "sha256:afd4a0c0c377d4ee99cff78e4d73727850b49434bd2042743468a3244757939c", + "sha256:b40194643b21159910bd52f041597aee8d594d5450feb16ffb1ac0bd51016eb7", + "sha256:b8630a7cc4d037f0ff214f9b65093b0482f3e0efd28043690ac003cbfd9ac56d", + "sha256:b92b8e0dab9444a79c7afa4253e88e1948ac51267deb354251fb2ee64c0659f1", + "sha256:bf61abef38d26e4e892fdf63493541290630038d78f8294dde81596ed7208858", + "sha256:c16a77926332615c018c44591c93215b08aaddfd724bd84bfbee4efa8bfcf877", + "sha256:c3af5070b6eda559555c68a92bdef2d8495208fff0a123601655ebc6a0ec6d02", + "sha256:c58d3e106af514486f65f09fee75a42a3fd83d5be9538befd5d5ceda6995da3b", + "sha256:d2987a57da49557a084d8c5512f3632058d7eeee8e32e49fc5e12c821963cf88", + "sha256:da94a1752032ab805bccb4878fdda671719f13c541cfca004aedd9b99c92ccfd", + "sha256:dac2370ea09f61530f01df454394f8c291a77a08d7c73180b1b5743746e6a390", + "sha256:e03aefcca84e57babf9e74f5ed859b249826eb3213edd5d6e4003c702dbbcd68", + "sha256:e0cd1b698a258afad559344dbe703ed35032558298d024d9fa4c83466dd21892", + "sha256:edc2c7cc433659069e56bf064560a348aad485b53799a5870b7e0d488a83cc1d", + "sha256:f06b1377d45f4664148fddec54180cd888f5abdb41fc31942641cd7797f17848", + "sha256:f15d799e3e7c3055cdb8e4b4bdf5f9c137893e0e04ff6b430d0f663bb48dd176", + "sha256:f46a66d324a0b42e6fd1ba7a9c703ff7fc4e9e00e08f3c283d2e54e24a8e63c2", + "sha256:f992a657bc0b12dee142c43256733500dcb5d1d5667e8282667875edce10e055" ], + "index": "pypi", "markers": "python_version >= '3.7'", - "version": "==2.8.4" + "version": "==2.8.5" }, "deprecated": { "hashes": [ @@ -389,6 +468,22 @@ "markers": "python_version >= '3.6'", "version": "==1.9.0" }, + "dnspython": { + "hashes": [ + "sha256:5ef3b9680161f6fa89daf8ad451b5f1a33b18ae8a1c6778cdf4b43f08c0a6e50", + "sha256:e8f0f9c23a7b7cb99ded64e6c3a6f3e701d78f50c55e002b839dea7225cff7cc" + ], + "markers": "python_version >= '3.8'", + "version": "==2.6.1" + }, + "email-validator": { + "hashes": [ + "sha256:200a70680ba08904be6d1eef729205cc0d687634399a5924d842533efb824b84", + "sha256:97d882d174e2a65732fb43bfce81a3a834cbc1bde8bf419e30ef5ea976370a05" + ], + "markers": "python_version >= '3.8'", + "version": "==2.1.1" + }, "envier": { "hashes": [ "sha256:b45ef6051fea33d0c32a64e186bff2cfb446e2242d6781216c9bc9ce708c5909", @@ -397,6 +492,23 @@ "markers": "python_version >= '3.7'", "version": "==0.5.1" }, + "fastapi": { + "hashes": [ + "sha256:97ecbf994be0bcbdadedf88c3150252bed7b2087075ac99735403b1b76cc8fc0", + "sha256:b9db9dd147c91cb8b769f7183535773d8741dd46f9dc6676cd82eab510228cd7" + ], + "index": "pypi", + "markers": "python_version >= '3.8'", + "version": "==0.111.0" + }, + "fastapi-cli": { + "hashes": [ + "sha256:a2552f3a7ae64058cdbb530be6fa6dbfc975dc165e4fa66d224c3d396e25e809", + "sha256:e2e9ffaffc1f7767f488d6da34b6f5a377751c996f397902eb6abb99a67bde32" + ], + "markers": "python_version >= '3.8'", + "version": "==0.0.4" + }, "frozenlist": { "hashes": [ "sha256:04ced3e6a46b4cfffe20f9ae482818e34eba9b5fb0ce4056e4cc9b6e212d09b7", @@ -482,11 +594,11 @@ }, "fsspec": { "hashes": [ - "sha256:918d18d41bf73f0e2b261824baeb1b124bcf771767e3a26425cd7dec3332f512", - "sha256:f39780e282d7d117ffb42bb96992f8a90795e4d0fb0f661a70ca39fe9c43ded9" + "sha256:1d021b0b0f933e3b3029ed808eb400c08ba101ca2de4b3483fbc9ca23fcee94a", + "sha256:e0fdbc446d67e182f49a70b82cf7889028a63588fde6b222521f10937b2b670c" ], "markers": "python_version >= '3.8'", - "version": "==2024.3.1" + "version": "==2024.5.0" }, "greenlet": { "hashes": [ @@ -568,6 +680,47 @@ "markers": "python_version >= '3.8'", "version": "==1.0.5" }, + "httptools": { + "hashes": [ + "sha256:00d5d4b68a717765b1fabfd9ca755bd12bf44105eeb806c03d1962acd9b8e563", + "sha256:0ac5a0ae3d9f4fe004318d64b8a854edd85ab76cffbf7ef5e32920faef62f142", + "sha256:0cf2372e98406efb42e93bfe10f2948e467edfd792b015f1b4ecd897903d3e8d", + "sha256:1ed99a373e327f0107cb513b61820102ee4f3675656a37a50083eda05dc9541b", + "sha256:3c3b214ce057c54675b00108ac42bacf2ab8f85c58e3f324a4e963bbc46424f4", + "sha256:3e802e0b2378ade99cd666b5bffb8b2a7cc8f3d28988685dc300469ea8dd86cb", + "sha256:3f30d3ce413088a98b9db71c60a6ada2001a08945cb42dd65a9a9fe228627658", + "sha256:405784577ba6540fa7d6ff49e37daf104e04f4b4ff2d1ac0469eaa6a20fde084", + "sha256:48ed8129cd9a0d62cf4d1575fcf90fb37e3ff7d5654d3a5814eb3d55f36478c2", + "sha256:4bd3e488b447046e386a30f07af05f9b38d3d368d1f7b4d8f7e10af85393db97", + "sha256:4f0f8271c0a4db459f9dc807acd0eadd4839934a4b9b892f6f160e94da309837", + "sha256:5cceac09f164bcba55c0500a18fe3c47df29b62353198e4f37bbcc5d591172c3", + "sha256:639dc4f381a870c9ec860ce5c45921db50205a37cc3334e756269736ff0aac58", + "sha256:678fcbae74477a17d103b7cae78b74800d795d702083867ce160fc202104d0da", + "sha256:6a4f5ccead6d18ec072ac0b84420e95d27c1cdf5c9f1bc8fbd8daf86bd94f43d", + "sha256:6f58e335a1402fb5a650e271e8c2d03cfa7cea46ae124649346d17bd30d59c90", + "sha256:75c8022dca7935cba14741a42744eee13ba05db00b27a4b940f0d646bd4d56d0", + "sha256:7a7ea483c1a4485c71cb5f38be9db078f8b0e8b4c4dc0210f531cdd2ddac1ef1", + "sha256:7d9ceb2c957320def533671fc9c715a80c47025139c8d1f3797477decbc6edd2", + "sha256:7ebaec1bf683e4bf5e9fbb49b8cc36da482033596a415b3e4ebab5a4c0d7ec5e", + "sha256:85ed077c995e942b6f1b07583e4eb0a8d324d418954fc6af913d36db7c05a5a0", + "sha256:8ae5b97f690badd2ca27cbf668494ee1b6d34cf1c464271ef7bfa9ca6b83ffaf", + "sha256:8b0bb634338334385351a1600a73e558ce619af390c2b38386206ac6a27fecfc", + "sha256:8e216a038d2d52ea13fdd9b9c9c7459fb80d78302b257828285eca1c773b99b3", + "sha256:93ad80d7176aa5788902f207a4e79885f0576134695dfb0fefc15b7a4648d503", + "sha256:95658c342529bba4e1d3d2b1a874db16c7cca435e8827422154c9da76ac4e13a", + "sha256:95fb92dd3649f9cb139e9c56604cc2d7c7bf0fc2e7c8d7fbd58f96e35eddd2a3", + "sha256:97662ce7fb196c785344d00d638fc9ad69e18ee4bfb4000b35a52efe5adcc949", + "sha256:9bb68d3a085c2174c2477eb3ffe84ae9fb4fde8792edb7bcd09a1d8467e30a84", + "sha256:b512aa728bc02354e5ac086ce76c3ce635b62f5fbc32ab7082b5e582d27867bb", + "sha256:c6e26c30455600b95d94b1b836085138e82f177351454ee841c148f93a9bad5a", + "sha256:d2f6c3c4cb1948d912538217838f6e9960bc4a521d7f9b323b3da579cd14532f", + "sha256:dcbab042cc3ef272adc11220517278519adf8f53fd3056d0e68f0a6f891ba94e", + "sha256:e0b281cf5a125c35f7f6722b65d8542d2e57331be573e9e88bc8b0115c4a7a81", + "sha256:e57997ac7fb7ee43140cc03664de5f268813a481dff6245e0075925adc6aa185", + "sha256:fe467eb086d80217b7584e61313ebadc8d187a4d95bb62031b7bab4b205c3ba3" + ], + "version": "==0.6.1" + }, "httpx": { "hashes": [ "sha256:71d5465162c13681bff01ad59b2cc68dd838ea1f10e51574bac27103f00c91a5", @@ -593,6 +746,14 @@ "markers": "python_version >= '3.8'", "version": "==7.0.0" }, + "jinja2": { + "hashes": [ + "sha256:4a3aee7acbbe7303aede8e9648d13b8bf88a429282aa6122a993f0ac800cb369", + "sha256:bc5dd2abb727a5319567b7a813e6a2e7318c39f4f487cfe6c89c6f9c7d25197d" + ], + "markers": "python_version >= '3.7'", + "version": "==3.1.4" + }, "joblib": { "hashes": [ "sha256:06d478d5674cbc267e7496a410ee875abd68e4340feff4490bcb7afb88060ae6", @@ -628,19 +789,19 @@ }, "llama-index-core": { "hashes": [ - "sha256:02f06bdefb5c6fd11dee1f65007a98decf3b266ad76136b7cfd3bec44efc5493", - "sha256:a6e8ea790e5b3656a254d9b47f8c00044dd46aae1cd43004c5d1303a7502b3e6" + "sha256:72d30aea7a77f87484abe99a341c945021c95ea5e2adcc60199094931a07623a", + "sha256:c08df2a46ebf417ca0f1b68a5d797df0c053796d96c93e1cdc0456d11a89753a" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.10.36" + "version": "==0.10.40" }, "llama-index-embeddings-openai": { "hashes": [ - "sha256:0fd292b2f9a0ad4534a790d6374726bc885853188087eb018167dcf239643924", - "sha256:fbd16d6197b91f4dbdc6d0707e573cc224ac2b0a48d5b370c6232dd8a2282473" + "sha256:1bc1fc9b46773a12870c5d3097d3735d7ca33805f12462a8e35ae8a6e5ce1cf6", + "sha256:c3cfa83b537ded34d035fc172a945dd444c87fb58a89b02dfbf785b675f9f681" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.9" + "version": "==0.1.10" }, "llama-index-indices-managed-llama-cloud": { "hashes": [ @@ -660,11 +821,11 @@ }, "llama-index-llms-openai": { "hashes": [ - "sha256:2bd98ff3abbb4aa0daed1fbe01d8b69f8270ab86c53f8da51fc9f148a672264c", - "sha256:f61b64a997892e424fb3cd547090d279c5b210ef15b614fc39de854d3ccaa7e7" + "sha256:e9d631135160ac87093cd7d5ae3e32c4debf8eeaafa29ee4739675867155d773", + "sha256:fd2fa240238c7dca0170bf55957bb088d9b6acf63c80924eb59498f3a5ac0df3" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.19" + "version": "==0.1.21" }, "llama-index-multi-modal-llms-openai": { "hashes": [ @@ -692,11 +853,11 @@ }, "llama-index-readers-file": { "hashes": [ - "sha256:37de54ad0cfbdc607c195532b9a292417a4714f57773570b87027b8dc381f0e2", - "sha256:a8d4a69a9ea659c14ebb22ca9a5560b9c7ec6f501e7f68f6c52f591374165376" + "sha256:32450d0a3edc6ef6af575f814beec39cd3a3351eaf0e3c97045bdd72a7a7b38d", + "sha256:fde8ecb588e703849e51dc0f075f56d1f5db3bc1479dd00c21b42e93b81b6267" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.22" + "version": "==0.1.23" }, "llama-index-readers-llama-parse": { "hashes": [ @@ -708,11 +869,11 @@ }, "llama-parse": { "hashes": [ - "sha256:01836147b5238873b24a7dd41c5ab942b01b09b92d75570f30cf2861c084a0eb", - "sha256:c48c53a3080daeede293df620dddb1f381e084c31ee2dd44dce3f8615df723e8" + "sha256:b45c2db33a0d6b7a2d5f59e3d0ec7ee7f8227a852eaa56b04aa12b12f2c0d521", + "sha256:bb9724d04fd31ed037000896c7cef7fcb9051325497db4592a15f8144754cd00" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.4.3" + "version": "==0.4.4" }, "llamaindex-py-client": { "hashes": [ @@ -722,6 +883,80 @@ "markers": "python_version >= '3.8' and python_version < '4'", "version": "==0.1.19" }, + "markdown-it-py": { + "hashes": [ + "sha256:355216845c60bd96232cd8d8c40e8f9765cc86f46880e43a8fd22dc1a1a8cab1", + "sha256:e3f60a94fa066dc52ec76661e37c851cb232d92f9886b15cb560aaada2df8feb" + ], + "markers": "python_version >= '3.8'", + "version": "==3.0.0" + }, + "markupsafe": { + "hashes": [ + "sha256:00e046b6dd71aa03a41079792f8473dc494d564611a8f89bbbd7cb93295ebdcf", + "sha256:075202fa5b72c86ad32dc7d0b56024ebdbcf2048c0ba09f1cde31bfdd57bcfff", + "sha256:0e397ac966fdf721b2c528cf028494e86172b4feba51d65f81ffd65c63798f3f", + "sha256:17b950fccb810b3293638215058e432159d2b71005c74371d784862b7e4683f3", + "sha256:1f3fbcb7ef1f16e48246f704ab79d79da8a46891e2da03f8783a5b6fa41a9532", + "sha256:2174c595a0d73a3080ca3257b40096db99799265e1c27cc5a610743acd86d62f", + "sha256:2b7c57a4dfc4f16f7142221afe5ba4e093e09e728ca65c51f5620c9aaeb9a617", + "sha256:2d2d793e36e230fd32babe143b04cec8a8b3eb8a3122d2aceb4a371e6b09b8df", + "sha256:30b600cf0a7ac9234b2638fbc0fb6158ba5bdcdf46aeb631ead21248b9affbc4", + "sha256:397081c1a0bfb5124355710fe79478cdbeb39626492b15d399526ae53422b906", + "sha256:3a57fdd7ce31c7ff06cdfbf31dafa96cc533c21e443d57f5b1ecc6cdc668ec7f", + "sha256:3c6b973f22eb18a789b1460b4b91bf04ae3f0c4234a0a6aa6b0a92f6f7b951d4", + "sha256:3e53af139f8579a6d5f7b76549125f0d94d7e630761a2111bc431fd820e163b8", + "sha256:4096e9de5c6fdf43fb4f04c26fb114f61ef0bf2e5604b6ee3019d51b69e8c371", + "sha256:4275d846e41ecefa46e2015117a9f491e57a71ddd59bbead77e904dc02b1bed2", + "sha256:4c31f53cdae6ecfa91a77820e8b151dba54ab528ba65dfd235c80b086d68a465", + "sha256:4f11aa001c540f62c6166c7726f71f7573b52c68c31f014c25cc7901deea0b52", + "sha256:5049256f536511ee3f7e1b3f87d1d1209d327e818e6ae1365e8653d7e3abb6a6", + "sha256:58c98fee265677f63a4385256a6d7683ab1832f3ddd1e66fe948d5880c21a169", + "sha256:598e3276b64aff0e7b3451b72e94fa3c238d452e7ddcd893c3ab324717456bad", + "sha256:5b7b716f97b52c5a14bffdf688f971b2d5ef4029127f1ad7a513973cfd818df2", + "sha256:5dedb4db619ba5a2787a94d877bc8ffc0566f92a01c0ef214865e54ecc9ee5e0", + "sha256:619bc166c4f2de5caa5a633b8b7326fbe98e0ccbfacabd87268a2b15ff73a029", + "sha256:629ddd2ca402ae6dbedfceeba9c46d5f7b2a61d9749597d4307f943ef198fc1f", + "sha256:656f7526c69fac7f600bd1f400991cc282b417d17539a1b228617081106feb4a", + "sha256:6ec585f69cec0aa07d945b20805be741395e28ac1627333b1c5b0105962ffced", + "sha256:72b6be590cc35924b02c78ef34b467da4ba07e4e0f0454a2c5907f473fc50ce5", + "sha256:7502934a33b54030eaf1194c21c692a534196063db72176b0c4028e140f8f32c", + "sha256:7a68b554d356a91cce1236aa7682dc01df0edba8d043fd1ce607c49dd3c1edcf", + "sha256:7b2e5a267c855eea6b4283940daa6e88a285f5f2a67f2220203786dfa59b37e9", + "sha256:823b65d8706e32ad2df51ed89496147a42a2a6e01c13cfb6ffb8b1e92bc910bb", + "sha256:8590b4ae07a35970728874632fed7bd57b26b0102df2d2b233b6d9d82f6c62ad", + "sha256:8dd717634f5a044f860435c1d8c16a270ddf0ef8588d4887037c5028b859b0c3", + "sha256:8dec4936e9c3100156f8a2dc89c4b88d5c435175ff03413b443469c7c8c5f4d1", + "sha256:97cafb1f3cbcd3fd2b6fbfb99ae11cdb14deea0736fc2b0952ee177f2b813a46", + "sha256:a17a92de5231666cfbe003f0e4b9b3a7ae3afb1ec2845aadc2bacc93ff85febc", + "sha256:a549b9c31bec33820e885335b451286e2969a2d9e24879f83fe904a5ce59d70a", + "sha256:ac07bad82163452a6884fe8fa0963fb98c2346ba78d779ec06bd7a6262132aee", + "sha256:ae2ad8ae6ebee9d2d94b17fb62763125f3f374c25618198f40cbb8b525411900", + "sha256:b91c037585eba9095565a3556f611e3cbfaa42ca1e865f7b8015fe5c7336d5a5", + "sha256:bc1667f8b83f48511b94671e0e441401371dfd0f0a795c7daa4a3cd1dde55bea", + "sha256:bec0a414d016ac1a18862a519e54b2fd0fc8bbfd6890376898a6c0891dd82e9f", + "sha256:bf50cd79a75d181c9181df03572cdce0fbb75cc353bc350712073108cba98de5", + "sha256:bff1b4290a66b490a2f4719358c0cdcd9bafb6b8f061e45c7a2460866bf50c2e", + "sha256:c061bb86a71b42465156a3ee7bd58c8c2ceacdbeb95d05a99893e08b8467359a", + "sha256:c8b29db45f8fe46ad280a7294f5c3ec36dbac9491f2d1c17345be8e69cc5928f", + "sha256:ce409136744f6521e39fd8e2a24c53fa18ad67aa5bc7c2cf83645cce5b5c4e50", + "sha256:d050b3361367a06d752db6ead6e7edeb0009be66bc3bae0ee9d97fb326badc2a", + "sha256:d283d37a890ba4c1ae73ffadf8046435c76e7bc2247bbb63c00bd1a709c6544b", + "sha256:d9fad5155d72433c921b782e58892377c44bd6252b5af2f67f16b194987338a4", + "sha256:daa4ee5a243f0f20d528d939d06670a298dd39b1ad5f8a72a4275124a7819eff", + "sha256:db0b55e0f3cc0be60c1f19efdde9a637c32740486004f20d1cff53c3c0ece4d2", + "sha256:e61659ba32cf2cf1481e575d0462554625196a1f2fc06a1c777d3f48e8865d46", + "sha256:ea3d8a3d18833cf4304cd2fc9cbb1efe188ca9b5efef2bdac7adc20594a0e46b", + "sha256:ec6a563cff360b50eed26f13adc43e61bc0c04d94b8be985e6fb24b81f6dcfdf", + "sha256:f5dfb42c4604dddc8e4305050aa6deb084540643ed5804d7455b5df8fe16f5e5", + "sha256:fa173ec60341d6bb97a89f5ea19c85c5643c1e7dedebc22f5181eb73573142c5", + "sha256:fa9db3f79de01457b03d4f01b34cf91bc0048eb2c3846ff26f66687c2f6d16ab", + "sha256:fce659a462a1be54d2ffcacea5e3ba2d74daa74f30f5f143fe0c58636e355fdd", + "sha256:ffee1f21e5ef0d712f9033568f8344d5da8cc2869dbd08d87c84656e6a2d2f68" + ], + "markers": "python_version >= '3.7'", + "version": "==2.1.5" + }, "marshmallow": { "hashes": [ "sha256:70b54a6282f4704d12c0a41599682c5c5450e843b9ec406308653b47c59648a1", @@ -730,6 +965,14 @@ "markers": "python_version >= '3.8'", "version": "==3.21.2" }, + "mdurl": { + "hashes": [ + "sha256:84008a41e51615a49fc9966191ff91509e3c40b939176e643fd50a5c2196b8f8", + "sha256:bb413d29f5eea38f31dd4754dd7377d4465116fb207585f97bf925588687c1ba" + ], + "markers": "python_version >= '3.7'", + "version": "==0.1.2" + }, "multidict": { "hashes": [ "sha256:01265f5e40f5a17f8241d52656ed27192be03bfa8764d88e8220141d1e4b3556", @@ -902,12 +1145,12 @@ }, "openai": { "hashes": [ - "sha256:aa2f381f476f5fa4df8728a34a3e454c321caa064b7b68ab6e9daa1ed082dbf9", - "sha256:f561ce86f4b4008eb6c78622d641e4b7e1ab8a8cdb15d2f0b2a49942d40d21a8" + "sha256:f3488d9a1c4e0d332b019377d27d7cb4b3d6103fd5d0a416c7ceac780d1d9b88", + "sha256:fb2635efd270efaf9fac2e07558d7948373b940637d3ae3ab624c1a983d4f03f" ], "index": "pypi", "markers": "python_full_version >= '3.7.1'", - "version": "==1.25.1" + "version": "==1.30.4" }, "opentelemetry-api": { "hashes": [ @@ -917,6 +1160,58 @@ "markers": "python_version >= '3.8'", "version": "==1.24.0" }, + "orjson": { + "hashes": [ + "sha256:0943a96b3fa09bee1afdfccc2cb236c9c64715afa375b2af296c73d91c23eab2", + "sha256:0a62f9968bab8a676a164263e485f30a0b748255ee2f4ae49a0224be95f4532b", + "sha256:16bda83b5c61586f6f788333d3cf3ed19015e3b9019188c56983b5a299210eb5", + "sha256:1770e2a0eae728b050705206d84eda8b074b65ee835e7f85c919f5705b006c9b", + "sha256:17e0713fc159abc261eea0f4feda611d32eabc35708b74bef6ad44f6c78d5ea0", + "sha256:18566beb5acd76f3769c1d1a7ec06cdb81edc4d55d2765fb677e3eaa10fa99e0", + "sha256:1952c03439e4dce23482ac846e7961f9d4ec62086eb98ae76d97bd41d72644d7", + "sha256:1bd2218d5a3aa43060efe649ec564ebedec8ce6ae0a43654b81376216d5ebd42", + "sha256:1c23dfa91481de880890d17aa7b91d586a4746a4c2aa9a145bebdbaf233768d5", + "sha256:252124b198662eee80428f1af8c63f7ff077c88723fe206a25df8dc57a57b1fa", + "sha256:2b166507acae7ba2f7c315dcf185a9111ad5e992ac81f2d507aac39193c2c818", + "sha256:2e5e176c994ce4bd434d7aafb9ecc893c15f347d3d2bbd8e7ce0b63071c52e25", + "sha256:3582b34b70543a1ed6944aca75e219e1192661a63da4d039d088a09c67543b08", + "sha256:382e52aa4270a037d41f325e7d1dfa395b7de0c367800b6f337d8157367bf3a7", + "sha256:416b195f78ae461601893f482287cee1e3059ec49b4f99479aedf22a20b1098b", + "sha256:4ad1f26bea425041e0a1adad34630c4825a9e3adec49079b1fb6ac8d36f8b754", + "sha256:4c895383b1ec42b017dd2c75ae8a5b862fc489006afde06f14afbdd0309b2af0", + "sha256:5102f50c5fc46d94f2033fe00d392588564378260d64377aec702f21a7a22912", + "sha256:520de5e2ef0b4ae546bea25129d6c7c74edb43fc6cf5213f511a927f2b28148b", + "sha256:544a12eee96e3ab828dbfcb4d5a0023aa971b27143a1d35dc214c176fdfb29b3", + "sha256:73100d9abbbe730331f2242c1fc0bcb46a3ea3b4ae3348847e5a141265479700", + "sha256:831c6ef73f9aa53c5f40ae8f949ff7681b38eaddb6904aab89dca4d85099cb78", + "sha256:8bc7a4df90da5d535e18157220d7915780d07198b54f4de0110eca6b6c11e290", + "sha256:8d0b84403d287d4bfa9bf7d1dc298d5c1c5d9f444f3737929a66f2fe4fb8f134", + "sha256:8d40c7f7938c9c2b934b297412c067936d0b54e4b8ab916fd1a9eb8f54c02294", + "sha256:9059d15c30e675a58fdcd6f95465c1522b8426e092de9fff20edebfdc15e1cb0", + "sha256:93433b3c1f852660eb5abdc1f4dd0ced2be031ba30900433223b28ee0140cde5", + "sha256:978be58a68ade24f1af7758626806e13cff7748a677faf95fbb298359aa1e20d", + "sha256:99b880d7e34542db89f48d14ddecbd26f06838b12427d5a25d71baceb5ba119d", + "sha256:9a7bc9e8bc11bac40f905640acd41cbeaa87209e7e1f57ade386da658092dc16", + "sha256:9e253498bee561fe85d6325ba55ff2ff08fb5e7184cd6a4d7754133bd19c9195", + "sha256:9f3e87733823089a338ef9bbf363ef4de45e5c599a9bf50a7a9b82e86d0228da", + "sha256:9fb6c3f9f5490a3eb4ddd46fc1b6eadb0d6fc16fb3f07320149c3286a1409dd8", + "sha256:a39aa73e53bec8d410875683bfa3a8edf61e5a1c7bb4014f65f81d36467ea098", + "sha256:b69a58a37dab856491bf2d3bbf259775fdce262b727f96aafbda359cb1d114d8", + "sha256:b8d4d1a6868cde356f1402c8faeb50d62cee765a1f7ffcfd6de732ab0581e063", + "sha256:ba7f67aa7f983c4345eeda16054a4677289011a478ca947cd69c0a86ea45e534", + "sha256:be2719e5041e9fb76c8c2c06b9600fe8e8584e6980061ff88dcbc2691a16d20d", + "sha256:be2aab54313752c04f2cbaab4515291ef5af8c2256ce22abc007f89f42f49109", + "sha256:c0403ed9c706dcd2809f1600ed18f4aae50be263bd7112e54b50e2c2bc3ebd6d", + "sha256:c8334c0d87103bb9fbbe59b78129f1f40d1d1e8355bbed2ca71853af15fa4ed3", + "sha256:cb0175a5798bdc878956099f5c54b9837cb62cfbf5d0b86ba6d77e43861bcec2", + "sha256:ccaa0a401fc02e8828a5bedfd80f8cd389d24f65e5ca3954d72c6582495b4bcf", + "sha256:cf20465e74c6e17a104ecf01bf8cd3b7b252565b4ccee4548f18b012ff2f8069", + "sha256:d4a654ec1de8fdaae1d80d55cee65893cb06494e124681ab335218be6a0691e7", + "sha256:e852baafceff8da3c9defae29414cc8513a1586ad93e45f27b89a639c68e8176" + ], + "markers": "python_version >= '3.8'", + "version": "==3.10.3" + }, "packaging": { "hashes": [ "sha256:2ddfb553fdf02fb784c234c7ba6ccc288296ceabec964ad2eae3777778130bc5", @@ -1037,20 +1332,28 @@ }, "protobuf": { "hashes": [ - "sha256:38aa5f535721d5bb99861166c445c4105c4e285c765fbb2ac10f116e32dcd46d", - "sha256:3c388ea6ddfe735f8cf69e3f7dc7611e73107b60bdfcf5d0f024c3ccd3794e23", - "sha256:7ee014c2c87582e101d6b54260af03b6596728505c79f17c8586e7523aaa8f8c", - "sha256:8ca2a1d97c290ec7b16e4e5dff2e5ae150cc1582f55b5ab300d45cb0dfa90e51", - "sha256:9b557c317ebe6836835ec4ef74ec3e994ad0894ea424314ad3552bc6e8835b4e", - "sha256:b9ba3ca83c2e31219ffbeb9d76b63aad35a3eb1544170c55336993d7a18ae72c", - "sha256:d693d2504ca96750d92d9de8a103102dd648fda04540495535f0fec7577ed8fc", - "sha256:da612f2720c0183417194eeaa2523215c4fcc1a1949772dc65f05047e08d5932", - "sha256:e6039957449cb918f331d32ffafa8eb9255769c96aa0560d9a5bf0b4e00a2a33", - "sha256:f7417703f841167e5a27d48be13389d52ad705ec09eade63dfc3180a959215d7", - "sha256:fbfe61e7ee8c1860855696e3ac6cfd1b01af5498facc6834fcc345c9684fb2ca" + "sha256:07f2b9a15255e3cf3f137d884af7972407b556a7a220912b252f26dc3121e6bf", + "sha256:2f83bf341d925650d550b8932b71763321d782529ac0eaf278f5242f513cc04e", + "sha256:56937f97ae0dcf4e220ff2abb1456c51a334144c9960b23597f044ce99c29c89", + "sha256:587be23f1212da7a14a6c65fd61995f8ef35779d4aea9e36aad81f5f3b80aec5", + "sha256:673ad60f1536b394b4fa0bcd3146a4130fcad85bfe3b60eaa86d6a0ace0fa374", + "sha256:744489f77c29174328d32f8921566fb0f7080a2f064c5137b9d6f4b790f9e0c1", + "sha256:7cb65fc8fba680b27cf7a07678084c6e68ee13cab7cace734954c25a43da6d0f", + "sha256:a17f4d664ea868102feaa30a674542255f9f4bf835d943d588440d1f49a3ed15", + "sha256:aabbbcf794fbb4c692ff14ce06780a66d04758435717107c387f12fb477bf0d8", + "sha256:b276e3f477ea1eebff3c2e1515136cfcff5ac14519c45f9b4aa2f6a87ea627c4", + "sha256:f51f33d305e18646f03acfdb343aac15b8115235af98bc9f844bf9446573827b" ], "markers": "python_version >= '3.8'", - "version": "==5.26.1" + "version": "==5.27.0" + }, + "pycparser": { + "hashes": [ + "sha256:491c8be9c040f5390f5bf44a5b07752bd07f56edf992381b05c701439eec10f6", + "sha256:c3702b6d3dd8c7abc1afa565d7e63d53a1d0bd86cdc24edd75470f4de499cfcc" + ], + "markers": "python_version >= '3.8'", + "version": "==2.22" }, "pydantic": { "hashes": [ @@ -1146,6 +1449,34 @@ "markers": "python_version >= '3.8'", "version": "==2.18.2" }, + "pydantic-settings": { + "hashes": [ + "sha256:00b9f6a5e95553590434c0fa01ead0b216c3e10bc54ae02e37f359948643c5ed", + "sha256:0235391d26db4d2190cb9b31051c4b46882d28a51533f97440867f012d4da091" + ], + "index": "pypi", + "markers": "python_version >= '3.8'", + "version": "==2.2.1" + }, + "pygments": { + "hashes": [ + "sha256:786ff802f32e91311bff3889f6e9a86e81505fe99f2735bb6d60ae0c5004f199", + "sha256:b8e6aca0523f3ab76fee51799c488e38782ac06eafcf95e7ba832985c8e7b13a" + ], + "markers": "python_version >= '3.8'", + "version": "==2.18.0" + }, + "pyjwt": { + "extras": [ + "crypto" + ], + "hashes": [ + "sha256:57e28d156e3d5c10088e0c68abb90bfac3df82b40a71bd0daa20c65ccd5c23de", + "sha256:59127c392cc44c2da5bb3192169a91f429924e17aff6534d70fdc02ab3e04320" + ], + "markers": "python_version >= '3.7'", + "version": "==2.8.0" + }, "pypdf": { "hashes": [ "sha256:dc035581664e0ad717e3492acebc1a5fc23dba759e788e3d4a9fc9b1a32e72c1", @@ -1171,6 +1502,23 @@ "markers": "python_version >= '3.8'", "version": "==1.0.1" }, + "python-json-logger": { + "hashes": [ + "sha256:23e7ec02d34237c5aa1e29a070193a4ea87583bb4e7f8fd06d3de8264c4b2e1c", + "sha256:f380b826a991ebbe3de4d897aeec42760035ac760345e57b812938dc8b35e2bd" + ], + "index": "pypi", + "markers": "python_version >= '3.6'", + "version": "==2.0.7" + }, + "python-multipart": { + "hashes": [ + "sha256:03f54688c663f1b7977105f021043b0793151e4cb1c1a9d4a11fc13d622c4026", + "sha256:97ca7b8ea7b05f977dc3849c3ba99d51689822fab725c3703af7c866a0c2b215" + ], + "markers": "python_version >= '3.8'", + "version": "==0.0.9" + }, "pytz": { "hashes": [ "sha256:2a29735ea9c18baf14b448846bde5a48030ed267578472d8955cd0e7443a9812", @@ -1237,104 +1585,120 @@ }, "regex": { "hashes": [ - "sha256:031219782d97550c2098d9a68ce9e9eaefe67d2d81d8ff84c8354f9c009e720c", - "sha256:0709ba544cf50bd5cb843df4b8bb6701bae2b70a8e88da9add8386cbca5c1385", - "sha256:0a9f89d7db5ef6bdf53e5cc8e6199a493d0f1374b3171796b464a74ebe8e508a", - "sha256:0bc94873ba11e34837bffd7e5006703abeffc4514e2f482022f46ce05bd25e67", - "sha256:0ce56a923f4c01d7568811bfdffe156268c0a7aae8a94c902b92fe34c4bde785", - "sha256:0faecb6d5779753a6066a3c7a0471a8d29fe25d9981ca9e552d6d1b8f8b6a594", - "sha256:1118ba9def608250250f4b3e3f48c62f4562ba16ca58ede491b6e7554bfa09ff", - "sha256:12446827f43c7881decf2c126762e11425de5eb93b3b0d8b581344c16db7047a", - "sha256:14905ed75c7a6edf423eb46c213ed3f4507c38115f1ed3c00f4ec9eafba50e58", - "sha256:15e593386ec6331e0ab4ac0795b7593f02ab2f4b30a698beb89fbdc34f92386a", - "sha256:160ba087232c5c6e2a1e7ad08bd3a3f49b58c815be0504d8c8aacfb064491cd8", - "sha256:161a206c8f3511e2f5fafc9142a2cc25d7fe9a1ec5ad9b4ad2496a7c33e1c5d2", - "sha256:169fd0acd7a259f58f417e492e93d0e15fc87592cd1e971c8c533ad5703b5830", - "sha256:193b7c6834a06f722f0ce1ba685efe80881de7c3de31415513862f601097648c", - "sha256:1a3903128f9e17a500618e80c68165c78c741ebb17dd1a0b44575f92c3c68b02", - "sha256:1d5bd666466c8f00a06886ce1397ba8b12371c1f1c6d1bef11013e9e0a1464a8", - "sha256:224a9269f133564109ce668213ef3cb32bc72ccf040b0b51c72a50e569e9dc9e", - "sha256:236cace6c1903effd647ed46ce6dd5d76d54985fc36dafc5256032886736c85d", - "sha256:249fbcee0a277c32a3ce36d8e36d50c27c968fdf969e0fbe342658d4e010fbc8", - "sha256:29d839829209f3c53f004e1de8c3113efce6d98029f044fa5cfee666253ee7e6", - "sha256:2c8982ee19ccecabbaeac1ba687bfef085a6352a8c64f821ce2f43e6d76a9298", - "sha256:2f30a5ab8902f93930dc6f627c4dd5da2703333287081c85cace0fc6e21c25af", - "sha256:304e7e2418146ae4d0ef0e9ffa28f881f7874b45b4994cc2279b21b6e7ae50c8", - "sha256:32e5f3b8e32918bfbdd12eca62e49ab3031125c454b507127ad6ecbd86e62fca", - "sha256:334b79ce9c08f26b4659a53f42892793948a613c46f1b583e985fd5a6bf1c149", - "sha256:33d19f0cde6838c81acffff25c7708e4adc7dd02896c9ec25c3939b1500a1778", - "sha256:3799e36d60a35162bb35b2246d8bb012192b7437dff807ef79c14e7352706306", - "sha256:42be5de7cc8c1edac55db92d82b68dc8e683b204d6f5414c5a51997a323d7081", - "sha256:44b3267cea873684af022822195298501568ed44d542f9a2d9bebc0212e99069", - "sha256:458d68d34fb74b906709735c927c029e62f7d06437a98af1b5b6258025223210", - "sha256:45cc13d398b6359a7708986386f72bd156ae781c3e83a68a6d4cee5af04b1ce9", - "sha256:4e7eaf9df15423d07b6050fb91f86c66307171b95ea53e2d87a7993b6d02c7f7", - "sha256:4fad420b14ae1970a1f322e8ae84a1d9d89375eb71e1b504060ab2d1bfe68f3c", - "sha256:504b5116e2bd1821efd815941edff7535e93372a098e156bb9dffde30264e798", - "sha256:50e7e96a527488334379e05755b210b7da4a60fc5d6481938c1fa053e0c92184", - "sha256:51d27844763c273a122e08a3e86e7aefa54ee09fb672d96a645ece0454d8425e", - "sha256:5253dcb0bfda7214523de58b002eb0090cb530d7c55993ce5f6d17faf953ece7", - "sha256:534efd2653ebc4f26fc0e47234e53bf0cb4715bb61f98c64d2774a278b58c846", - "sha256:560278c9975694e1f0bc50da187abf2cdc1e4890739ea33df2bc4a85eeef143e", - "sha256:571452362d552de508c37191b6abbbb660028b8b418e2d68c20779e0bc8eaaa8", - "sha256:62b5f7910b639f3c1d122d408421317c351e213ca39c964ad4121f27916631c6", - "sha256:696639a73ca78a380acfaa0a1f6dd8220616a99074c05bba9ba8bb916914b224", - "sha256:6ccdeef4584450b6f0bddd5135354908dacad95425fcb629fe36d13e48b60f32", - "sha256:70364a097437dd0a90b31cd77f09f7387ad9ac60ef57590971f43b7fca3082a5", - "sha256:7117cb7d6ac7f2e985f3d18aa8a1728864097da1a677ffa69e970ca215baebf1", - "sha256:7467ad8b0eac0b28e52679e972b9b234b3de0ea5cee12eb50091d2b68145fe36", - "sha256:7d35d4cc9270944e95f9c88af757b0c9fc43f396917e143a5756608462c5223b", - "sha256:7dda3091838206969c2b286f9832dff41e2da545b99d1cfaea9ebd8584d02708", - "sha256:853cc36e756ff673bf984e9044ccc8fad60b95a748915dddeab9488aea974c73", - "sha256:8722f72068b3e1156a4b2e1afde6810f1fc67155a9fa30a4b9d5b4bc46f18fb0", - "sha256:8c6c71cf92b09e5faa72ea2c68aa1f61c9ce11cb66fdc5069d712f4392ddfd00", - "sha256:903350bf44d7e4116b4d5898b30b15755d61dcd3161e3413a49c7db76f0bee5a", - "sha256:91b53dea84415e8115506cc62e441a2b54537359c63d856d73cb1abe05af4c9a", - "sha256:951be1eae7b47660412dc4938777a975ebc41936d64e28081bf2e584b47ec246", - "sha256:972b49f2fe1047b9249c958ec4fa1bdd2cf8ce305dc19d27546d5a38e57732d8", - "sha256:9a8625849387b9d558d528e263ecc9c0fbde86cfa5c2f0eef43fff480ae24d71", - "sha256:9cdbb1998da94607d5eec02566b9586f0e70d6438abf1b690261aac0edda7ab6", - "sha256:9e6d4d6ae1827b2f8c7200aaf7501c37cf3f3896c86a6aaf2566448397c823dd", - "sha256:aab65121229c2ecdf4a31b793d99a6a0501225bd39b616e653c87b219ed34a49", - "sha256:ab98016541543692a37905871a5ffca59b16e08aacc3d7d10a27297b443f572d", - "sha256:ad45f3bccfcb00868f2871dce02a755529838d2b86163ab8a246115e80cfb7d6", - "sha256:b43b78f9386d3d932a6ce5af4b45f393d2e93693ee18dc4800d30a8909df700e", - "sha256:b66421f8878a0c82fc0c272a43e2121c8d4c67cb37429b764f0d5ad70b82993b", - "sha256:ba034c8db4b264ef1601eb33cd23d87c5013b8fb48b8161debe2e5d3bd9156b0", - "sha256:bbdc5db2c98ac2bf1971ffa1410c87ca7a15800415f788971e8ba8520fc0fda9", - "sha256:bc0db93ad039fc2fe32ccd3dd0e0e70c4f3d6e37ae83f0a487e1aba939bd2fbd", - "sha256:bf7c8ee4861d9ef5b1120abb75846828c811f932d63311596ad25fa168053e00", - "sha256:bf9596cba92ce7b1fd32c7b07c6e3212c7eed0edc271757e48bfcd2b54646452", - "sha256:c43395a3b7cc9862801a65c6994678484f186ce13c929abab44fb8a9e473a55a", - "sha256:c46a76a599fcbf95f98755275c5527304cc4f1bb69919434c1e15544d7052910", - "sha256:ca23b41355ba95929e9505ee04e55495726aa2282003ed9b012d86f857d3e49b", - "sha256:cd832bd9b6120d6074f39bdfbb3c80e416848b07ac72910f1c7f03131a6debc3", - "sha256:cfa6d61a76c77610ba9274c1a90a453062bdf6887858afbe214d18ad41cf6bde", - "sha256:d8a0f0ab5453e409586b11ebe91c672040bc804ca98d03a656825f7890cbdf88", - "sha256:e91b1976358e17197157b405cab408a5f4e33310cda211c49fc6da7cffd0b2f0", - "sha256:ea057306ab469130167014b662643cfaed84651c792948891d003cf0039223a5", - "sha256:eda3dd46df535da787ffb9036b5140f941ecb91701717df91c9daf64cabef953", - "sha256:f03b1dbd4d9596dd84955bb40f7d885204d6aac0d56a919bb1e0ff2fb7e1735a", - "sha256:fa9335674d7c819674467c7b46154196c51efbaf5f5715187fd366814ba3fa39" + "sha256:0721931ad5fe0dda45d07f9820b90b2148ccdd8e45bb9e9b42a146cb4f695649", + "sha256:10002e86e6068d9e1c91eae8295ef690f02f913c57db120b58fdd35a6bb1af35", + "sha256:10e4ce0dca9ae7a66e6089bb29355d4432caed736acae36fef0fdd7879f0b0cb", + "sha256:119af6e56dce35e8dfb5222573b50c89e5508d94d55713c75126b753f834de68", + "sha256:1337b7dbef9b2f71121cdbf1e97e40de33ff114801263b275aafd75303bd62b5", + "sha256:13cdaf31bed30a1e1c2453ef6015aa0983e1366fad2667657dbcac7b02f67133", + "sha256:1595f2d10dff3d805e054ebdc41c124753631b6a471b976963c7b28543cf13b0", + "sha256:16093f563098448ff6b1fa68170e4acbef94e6b6a4e25e10eae8598bb1694b5d", + "sha256:1878b8301ed011704aea4c806a3cadbd76f84dece1ec09cc9e4dc934cfa5d4da", + "sha256:19068a6a79cf99a19ccefa44610491e9ca02c2be3305c7760d3831d38a467a6f", + "sha256:19dfb1c504781a136a80ecd1fff9f16dddf5bb43cec6871778c8a907a085bb3d", + "sha256:1b5269484f6126eee5e687785e83c6b60aad7663dafe842b34691157e5083e53", + "sha256:1c1c174d6ec38d6c8a7504087358ce9213d4332f6293a94fbf5249992ba54efa", + "sha256:2431b9e263af1953c55abbd3e2efca67ca80a3de8a0437cb58e2421f8184717a", + "sha256:287eb7f54fc81546346207c533ad3c2c51a8d61075127d7f6d79aaf96cdee890", + "sha256:2b4c884767504c0e2401babe8b5b7aea9148680d2e157fa28f01529d1f7fcf67", + "sha256:35cb514e137cb3488bce23352af3e12fb0dbedd1ee6e60da053c69fb1b29cc6c", + "sha256:391d7f7f1e409d192dba8bcd42d3e4cf9e598f3979cdaed6ab11288da88cb9f2", + "sha256:3ad070b823ca5890cab606c940522d05d3d22395d432f4aaaf9d5b1653e47ced", + "sha256:3cd7874d57f13bf70078f1ff02b8b0aa48d5b9ed25fc48547516c6aba36f5741", + "sha256:3e507ff1e74373c4d3038195fdd2af30d297b4f0950eeda6f515ae3d84a1770f", + "sha256:455705d34b4154a80ead722f4f185b04c4237e8e8e33f265cd0798d0e44825fa", + "sha256:4a605586358893b483976cffc1723fb0f83e526e8f14c6e6614e75919d9862cf", + "sha256:4babf07ad476aaf7830d77000874d7611704a7fcf68c9c2ad151f5d94ae4bfc4", + "sha256:4eee78a04e6c67e8391edd4dad3279828dd66ac4b79570ec998e2155d2e59fd5", + "sha256:5397de3219a8b08ae9540c48f602996aa6b0b65d5a61683e233af8605c42b0f2", + "sha256:5b5467acbfc153847d5adb21e21e29847bcb5870e65c94c9206d20eb4e99a384", + "sha256:5eaa7ddaf517aa095fa8da0b5015c44d03da83f5bd49c87961e3c997daed0de7", + "sha256:632b01153e5248c134007209b5c6348a544ce96c46005d8456de1d552455b014", + "sha256:64c65783e96e563103d641760664125e91bd85d8e49566ee560ded4da0d3e704", + "sha256:64f18a9a3513a99c4bef0e3efd4c4a5b11228b48aa80743be822b71e132ae4f5", + "sha256:673b5a6da4557b975c6c90198588181029c60793835ce02f497ea817ff647cb2", + "sha256:68811ab14087b2f6e0fc0c2bae9ad689ea3584cad6917fc57be6a48bbd012c49", + "sha256:6e8d717bca3a6e2064fc3a08df5cbe366369f4b052dcd21b7416e6d71620dca1", + "sha256:71a455a3c584a88f654b64feccc1e25876066c4f5ef26cd6dd711308aa538694", + "sha256:72d7a99cd6b8f958e85fc6ca5b37c4303294954eac1376535b03c2a43eb72629", + "sha256:7b59138b219ffa8979013be7bc85bb60c6f7b7575df3d56dc1e403a438c7a3f6", + "sha256:7dbe2467273b875ea2de38ded4eba86cbcbc9a1a6d0aa11dcf7bd2e67859c435", + "sha256:833616ddc75ad595dee848ad984d067f2f31be645d603e4d158bba656bbf516c", + "sha256:87e2a9c29e672fc65523fb47a90d429b70ef72b901b4e4b1bd42387caf0d6835", + "sha256:8fe45aa3f4aa57faabbc9cb46a93363edd6197cbc43523daea044e9ff2fea83e", + "sha256:9e717956dcfd656f5055cc70996ee2cc82ac5149517fc8e1b60261b907740201", + "sha256:9efa1a32ad3a3ea112224897cdaeb6aa00381627f567179c0314f7b65d354c62", + "sha256:9ff11639a8d98969c863d4617595eb5425fd12f7c5ef6621a4b74b71ed8726d5", + "sha256:a094801d379ab20c2135529948cb84d417a2169b9bdceda2a36f5f10977ebc16", + "sha256:a0981022dccabca811e8171f913de05720590c915b033b7e601f35ce4ea7019f", + "sha256:a0bd000c6e266927cb7a1bc39d55be95c4b4f65c5be53e659537537e019232b1", + "sha256:a32b96f15c8ab2e7d27655969a23895eb799de3665fa94349f3b2fbfd547236f", + "sha256:a81e3cfbae20378d75185171587cbf756015ccb14840702944f014e0d93ea09f", + "sha256:ac394ff680fc46b97487941f5e6ae49a9f30ea41c6c6804832063f14b2a5a145", + "sha256:ada150c5adfa8fbcbf321c30c751dc67d2f12f15bd183ffe4ec7cde351d945b3", + "sha256:b2b6f1b3bb6f640c1a92be3bbfbcb18657b125b99ecf141fb3310b5282c7d4ed", + "sha256:b802512f3e1f480f41ab5f2cfc0e2f761f08a1f41092d6718868082fc0d27143", + "sha256:ba68168daedb2c0bab7fd7e00ced5ba90aebf91024dea3c88ad5063c2a562cca", + "sha256:bfc4f82cabe54f1e7f206fd3d30fda143f84a63fe7d64a81558d6e5f2e5aaba9", + "sha256:c0c18345010870e58238790a6779a1219b4d97bd2e77e1140e8ee5d14df071aa", + "sha256:c3bea0ba8b73b71b37ac833a7f3fd53825924165da6a924aec78c13032f20850", + "sha256:c486b4106066d502495b3025a0a7251bf37ea9540433940a23419461ab9f2a80", + "sha256:c49e15eac7c149f3670b3e27f1f28a2c1ddeccd3a2812cba953e01be2ab9b5fe", + "sha256:c6a2b494a76983df8e3d3feea9b9ffdd558b247e60b92f877f93a1ff43d26656", + "sha256:cab12877a9bdafde5500206d1020a584355a97884dfd388af3699e9137bf7388", + "sha256:cac27dcaa821ca271855a32188aa61d12decb6fe45ffe3e722401fe61e323cd1", + "sha256:cdd09d47c0b2efee9378679f8510ee6955d329424c659ab3c5e3a6edea696294", + "sha256:cf2430df4148b08fb4324b848672514b1385ae3807651f3567871f130a728cc3", + "sha256:d0a3d8d6acf0c78a1fff0e210d224b821081330b8524e3e2bc5a68ef6ab5803d", + "sha256:d0c0c0003c10f54a591d220997dd27d953cd9ccc1a7294b40a4be5312be8797b", + "sha256:d1f059a4d795e646e1c37665b9d06062c62d0e8cc3c511fe01315973a6542e40", + "sha256:d347a741ea871c2e278fde6c48f85136c96b8659b632fb57a7d1ce1872547600", + "sha256:d3ee02d9e5f482cc8309134a91eeaacbdd2261ba111b0fef3748eeb4913e6a2c", + "sha256:d99ceffa25ac45d150e30bd9ed14ec6039f2aad0ffa6bb87a5936f5782fc1569", + "sha256:e38a7d4e8f633a33b4c7350fbd8bad3b70bf81439ac67ac38916c4a86b465456", + "sha256:e4682f5ba31f475d58884045c1a97a860a007d44938c4c0895f41d64481edbc9", + "sha256:e5bb9425fe881d578aeca0b2b4b3d314ec88738706f66f219c194d67179337cb", + "sha256:e64198f6b856d48192bf921421fdd8ad8eb35e179086e99e99f711957ffedd6e", + "sha256:e6662686aeb633ad65be2a42b4cb00178b3fbf7b91878f9446075c404ada552f", + "sha256:ec54d5afa89c19c6dd8541a133be51ee1017a38b412b1321ccb8d6ddbeb4cf7d", + "sha256:f5b1dff3ad008dccf18e652283f5e5339d70bf8ba7c98bf848ac33db10f7bc7a", + "sha256:f8ec0c2fea1e886a19c3bee0cd19d862b3aa75dcdfb42ebe8ed30708df64687a", + "sha256:f9ebd0a36102fcad2f03696e8af4ae682793a5d30b46c647eaf280d6cfb32796" ], "markers": "python_version >= '3.8'", - "version": "==2024.5.10" + "version": "==2024.5.15" }, "requests": { "hashes": [ - "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f", - "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1" + "sha256:55365417734eb18255590a9ff9eb97e9e1da868d4ccd6402399eaf68af20a760", + "sha256:70761cfe03c773ceb22aa2f671b4757976145175cdfca038c02654d061d6dcc6" ], - "markers": "python_version >= '3.7'", - "version": "==2.31.0" + "markers": "python_version >= '3.8'", + "version": "==2.32.3" + }, + "rich": { + "hashes": [ + "sha256:4edbae314f59eb482f54e9e30bf00d33350aaa94f4bfcd4e9e3110e64d0d7222", + "sha256:9be308cb1fe2f1f57d67ce99e95af38a1e2bc71ad9813b0e247cf7ffbcc3a432" + ], + "markers": "python_full_version >= '3.7.0'", + "version": "==13.7.1" }, "setuptools": { "hashes": [ - "sha256:6c1fccdac05a97e598fb0ae3bbed5904ccb317337a51139dcd51453611bbb987", - "sha256:c636ac361bc47580504644275c9ad802c50415c7522212252c033bd15f301f32" + "sha256:54faa7f2e8d2d11bcd2c07bed282eef1046b5c080d1c32add737d7b5817b1ad4", + "sha256:f211a66637b8fa059bb28183da127d4e86396c991a942b028c6650d4319c3fd0" ], "markers": "python_version >= '3.12'", - "version": "==69.5.1" + "version": "==70.0.0" + }, + "shellingham": { + "hashes": [ + "sha256:7ecfff8f2fd72616f7481040475a65b2bf8af90a56c89140852d1120324e8686", + "sha256:8dbca0739d487e5bd35ab3ca4b36e11c4078f3a234bfce294b0a0291363404de" + ], + "markers": "python_version >= '3.7'", + "version": "==1.5.4" }, "six": { "hashes": [ @@ -1426,6 +1790,14 @@ "markers": "python_version >= '3.8'", "version": "==0.5.0" }, + "starlette": { + "hashes": [ + "sha256:6fe59f29268538e5d0d182f2791a479a0c64638e6935d1c6989e63fb2699c6ee", + "sha256:9af890290133b79fc3db55474ade20f6220a364a0402e0b556e7cd5e1e093823" + ], + "markers": "python_version >= '3.8'", + "version": "==0.37.2" + }, "striprtf": { "hashes": [ "sha256:8c8f9d32083cdc2e8bfb149455aa1cc5a4e0a035893bedc75db8b73becb3a1bb", @@ -1491,13 +1863,21 @@ "markers": "python_version >= '3.7'", "version": "==4.66.4" }, + "typer": { + "hashes": [ + "sha256:070d7ca53f785acbccba8e7d28b08dcd88f79f1fbda035ade0aecec71ca5c914", + "sha256:49e73131481d804288ef62598d97a1ceef3058905aa536a1134f90891ba35482" + ], + "markers": "python_version >= '3.7'", + "version": "==0.12.3" + }, "typing-extensions": { "hashes": [ - "sha256:83f085bd5ca59c80295fc2a82ab5dac679cbe02b9f33f7d83af68e241bea51b0", - "sha256:c1f94d72897edaf4ce775bb7558d5b79d8126906a14ea5ed1635921406c0387a" + "sha256:8cbcdc8606ebcb0d95453ad7dc5065e6237b6aa230a31e81d0f440c30fed5fd8", + "sha256:b349c66bea9016ac22978d800cfff206d5f9816951f12a7d0ec5578b0a819594" ], "markers": "python_version >= '3.8'", - "version": "==4.11.0" + "version": "==4.12.0" }, "typing-inspect": { "hashes": [ @@ -1600,11 +1980,215 @@ }, "urllib3": { "hashes": [ - "sha256:c97dfde1f7bd43a71c8d2a58e369e9b2bf692d1334ea9f9cae55add7d0dd0f84", - "sha256:fdb6d215c776278489906c2f8916e6e7d4f5a9b602ccbcfdf7f016fc8da0596e" + "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d", + "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19" ], - "markers": "python_version >= '3.11'", - "version": "==2.0.7" + "markers": "python_version >= '3.8'", + "version": "==2.2.1" + }, + "uvicorn": { + "extras": [ + "standard" + ], + "hashes": [ + "sha256:78fa0b5f56abb8562024a59041caeb555c86e48d0efdd23c3fe7de7a4075bdab", + "sha256:f678dec4fa3a39706bbf49b9ec5fc40049d42418716cea52b53f07828a60aa37" + ], + "markers": "python_version >= '3.8'", + "version": "==0.30.0" + }, + "uvloop": { + "hashes": [ + "sha256:0246f4fd1bf2bf702e06b0d45ee91677ee5c31242f39aab4ea6fe0c51aedd0fd", + "sha256:02506dc23a5d90e04d4f65c7791e65cf44bd91b37f24cfc3ef6cf2aff05dc7ec", + "sha256:13dfdf492af0aa0a0edf66807d2b465607d11c4fa48f4a1fd41cbea5b18e8e8b", + "sha256:2693049be9d36fef81741fddb3f441673ba12a34a704e7b4361efb75cf30befc", + "sha256:271718e26b3e17906b28b67314c45d19106112067205119dddbd834c2b7ce797", + "sha256:2df95fca285a9f5bfe730e51945ffe2fa71ccbfdde3b0da5772b4ee4f2e770d5", + "sha256:31e672bb38b45abc4f26e273be83b72a0d28d074d5b370fc4dcf4c4eb15417d2", + "sha256:34175c9fd2a4bc3adc1380e1261f60306344e3407c20a4d684fd5f3be010fa3d", + "sha256:45bf4c24c19fb8a50902ae37c5de50da81de4922af65baf760f7c0c42e1088be", + "sha256:472d61143059c84947aa8bb74eabbace30d577a03a1805b77933d6bd13ddebbd", + "sha256:47bf3e9312f63684efe283f7342afb414eea4d3011542155c7e625cd799c3b12", + "sha256:492e2c32c2af3f971473bc22f086513cedfc66a130756145a931a90c3958cb17", + "sha256:4ce6b0af8f2729a02a5d1575feacb2a94fc7b2e983868b009d51c9a9d2149bef", + "sha256:5138821e40b0c3e6c9478643b4660bd44372ae1e16a322b8fc07478f92684e24", + "sha256:5588bd21cf1fcf06bded085f37e43ce0e00424197e7c10e77afd4bbefffef428", + "sha256:570fc0ed613883d8d30ee40397b79207eedd2624891692471808a95069a007c1", + "sha256:5a05128d315e2912791de6088c34136bfcdd0c7cbc1cf85fd6fd1bb321b7c849", + "sha256:5daa304d2161d2918fa9a17d5635099a2f78ae5b5960e742b2fcfbb7aefaa593", + "sha256:5f17766fb6da94135526273080f3455a112f82570b2ee5daa64d682387fe0dcd", + "sha256:6e3d4e85ac060e2342ff85e90d0c04157acb210b9ce508e784a944f852a40e67", + "sha256:7010271303961c6f0fe37731004335401eb9075a12680738731e9c92ddd96ad6", + "sha256:7207272c9520203fea9b93843bb775d03e1cf88a80a936ce760f60bb5add92f3", + "sha256:78ab247f0b5671cc887c31d33f9b3abfb88d2614b84e4303f1a63b46c046c8bd", + "sha256:7b1fd71c3843327f3bbc3237bedcdb6504fd50368ab3e04d0410e52ec293f5b8", + "sha256:8ca4956c9ab567d87d59d49fa3704cf29e37109ad348f2d5223c9bf761a332e7", + "sha256:91ab01c6cd00e39cde50173ba4ec68a1e578fee9279ba64f5221810a9e786533", + "sha256:cd81bdc2b8219cb4b2556eea39d2e36bfa375a2dd021404f90a62e44efaaf957", + "sha256:da8435a3bd498419ee8c13c34b89b5005130a476bda1d6ca8cfdde3de35cd650", + "sha256:de4313d7f575474c8f5a12e163f6d89c0a878bc49219641d49e6f1444369a90e", + "sha256:e27f100e1ff17f6feeb1f33968bc185bf8ce41ca557deee9d9bbbffeb72030b7", + "sha256:f467a5fd23b4fc43ed86342641f3936a68ded707f4627622fa3f82a120e18256" + ], + "version": "==0.19.0" + }, + "watchfiles": { + "hashes": [ + "sha256:00095dd368f73f8f1c3a7982a9801190cc88a2f3582dd395b289294f8975172b", + "sha256:00ad0bcd399503a84cc688590cdffbe7a991691314dde5b57b3ed50a41319a31", + "sha256:00f39592cdd124b4ec5ed0b1edfae091567c72c7da1487ae645426d1b0ffcad1", + "sha256:030bc4e68d14bcad2294ff68c1ed87215fbd9a10d9dea74e7cfe8a17869785ab", + "sha256:052d668a167e9fc345c24203b104c313c86654dd6c0feb4b8a6dfc2462239249", + "sha256:067dea90c43bf837d41e72e546196e674f68c23702d3ef80e4e816937b0a3ffd", + "sha256:0b04a2cbc30e110303baa6d3ddce8ca3664bc3403be0f0ad513d1843a41c97d1", + "sha256:0bc3b2f93a140df6806c8467c7f51ed5e55a931b031b5c2d7ff6132292e803d6", + "sha256:0c8e0aa0e8cc2a43561e0184c0513e291ca891db13a269d8d47cb9841ced7c71", + "sha256:103622865599f8082f03af4214eaff90e2426edff5e8522c8f9e93dc17caee13", + "sha256:1235c11510ea557fe21be5d0e354bae2c655a8ee6519c94617fe63e05bca4171", + "sha256:1cc0cba54f47c660d9fa3218158b8963c517ed23bd9f45fe463f08262a4adae1", + "sha256:1d9188979a58a096b6f8090e816ccc3f255f137a009dd4bbec628e27696d67c1", + "sha256:213792c2cd3150b903e6e7884d40660e0bcec4465e00563a5fc03f30ea9c166c", + "sha256:25c817ff2a86bc3de3ed2df1703e3d24ce03479b27bb4527c57e722f8554d971", + "sha256:2627a91e8110b8de2406d8b2474427c86f5a62bf7d9ab3654f541f319ef22bcb", + "sha256:280a4afbc607cdfc9571b9904b03a478fc9f08bbeec382d648181c695648202f", + "sha256:28324d6b28bcb8d7c1041648d7b63be07a16db5510bea923fc80b91a2a6cbed6", + "sha256:28585744c931576e535860eaf3f2c0ec7deb68e3b9c5a85ca566d69d36d8dd27", + "sha256:28f393c1194b6eaadcdd8f941307fc9bbd7eb567995232c830f6aef38e8a6e88", + "sha256:2abeb79209630da981f8ebca30a2c84b4c3516a214451bfc5f106723c5f45843", + "sha256:2bdadf6b90c099ca079d468f976fd50062905d61fae183f769637cb0f68ba59a", + "sha256:2f350cbaa4bb812314af5dab0eb8d538481e2e2279472890864547f3fe2281ed", + "sha256:3218a6f908f6a276941422b035b511b6d0d8328edd89a53ae8c65be139073f84", + "sha256:3973145235a38f73c61474d56ad6199124e7488822f3a4fc97c72009751ae3b0", + "sha256:3a0d883351a34c01bd53cfa75cd0292e3f7e268bacf2f9e33af4ecede7e21d1d", + "sha256:425440e55cd735386ec7925f64d5dde392e69979d4c8459f6bb4e920210407f2", + "sha256:4b9f2a128a32a2c273d63eb1fdbf49ad64852fc38d15b34eaa3f7ca2f0d2b797", + "sha256:4cc382083afba7918e32d5ef12321421ef43d685b9a67cc452a6e6e18920890e", + "sha256:52fc9b0dbf54d43301a19b236b4a4614e610605f95e8c3f0f65c3a456ffd7d35", + "sha256:55b7cc10261c2786c41d9207193a85c1db1b725cf87936df40972aab466179b6", + "sha256:581f0a051ba7bafd03e17127735d92f4d286af941dacf94bcf823b101366249e", + "sha256:5834e1f8b71476a26df97d121c0c0ed3549d869124ed2433e02491553cb468c2", + "sha256:5e45fb0d70dda1623a7045bd00c9e036e6f1f6a85e4ef2c8ae602b1dfadf7550", + "sha256:61af9efa0733dc4ca462347becb82e8ef4945aba5135b1638bfc20fad64d4f0e", + "sha256:68fe0c4d22332d7ce53ad094622b27e67440dacefbaedd29e0794d26e247280c", + "sha256:72a44e9481afc7a5ee3291b09c419abab93b7e9c306c9ef9108cb76728ca58d2", + "sha256:7a74436c415843af2a769b36bf043b6ccbc0f8d784814ba3d42fc961cdb0a9dc", + "sha256:8597b6f9dc410bdafc8bb362dac1cbc9b4684a8310e16b1ff5eee8725d13dcd6", + "sha256:8c39987a1397a877217be1ac0fb1d8b9f662c6077b90ff3de2c05f235e6a8f96", + "sha256:8c3e3675e6e39dc59b8fe5c914a19d30029e36e9f99468dddffd432d8a7b1c93", + "sha256:8dc1fc25a1dedf2dd952909c8e5cb210791e5f2d9bc5e0e8ebc28dd42fed7562", + "sha256:8fdebb655bb1ba0122402352b0a4254812717a017d2dc49372a1d47e24073795", + "sha256:9165bcab15f2b6d90eedc5c20a7f8a03156b3773e5fb06a790b54ccecdb73385", + "sha256:94ebe84a035993bb7668f58a0ebf998174fb723a39e4ef9fce95baabb42b787f", + "sha256:9624a68b96c878c10437199d9a8b7d7e542feddda8d5ecff58fdc8e67b460848", + "sha256:96eec15e5ea7c0b6eb5bfffe990fc7c6bd833acf7e26704eb18387fb2f5fd087", + "sha256:97b94e14b88409c58cdf4a8eaf0e67dfd3ece7e9ce7140ea6ff48b0407a593ec", + "sha256:988e981aaab4f3955209e7e28c7794acdb690be1efa7f16f8ea5aba7ffdadacb", + "sha256:a8a31bfd98f846c3c284ba694c6365620b637debdd36e46e1859c897123aa232", + "sha256:a927b3034d0672f62fb2ef7ea3c9fc76d063c4b15ea852d1db2dc75fe2c09696", + "sha256:ace7d060432acde5532e26863e897ee684780337afb775107c0a90ae8dbccfd2", + "sha256:aec83c3ba24c723eac14225194b862af176d52292d271c98820199110e31141e", + "sha256:b44b70850f0073b5fcc0b31ede8b4e736860d70e2dbf55701e05d3227a154a67", + "sha256:b610fb5e27825b570554d01cec427b6620ce9bd21ff8ab775fc3a32f28bba63e", + "sha256:b810a2c7878cbdecca12feae2c2ae8af59bea016a78bc353c184fa1e09f76b68", + "sha256:bbf8a20266136507abf88b0df2328e6a9a7c7309e8daff124dda3803306a9fdb", + "sha256:bd4c06100bce70a20c4b81e599e5886cf504c9532951df65ad1133e508bf20be", + "sha256:c2444dc7cb9d8cc5ab88ebe792a8d75709d96eeef47f4c8fccb6df7c7bc5be71", + "sha256:c49b76a78c156979759d759339fb62eb0549515acfe4fd18bb151cc07366629c", + "sha256:c4a65474fd2b4c63e2c18ac67a0c6c66b82f4e73e2e4d940f837ed3d2fd9d4da", + "sha256:c5af2347d17ab0bd59366db8752d9e037982e259cacb2ba06f2c41c08af02c39", + "sha256:c668228833c5619f6618699a2c12be057711b0ea6396aeaece4ded94184304ea", + "sha256:c7b978c384e29d6c7372209cbf421d82286a807bbcdeb315427687f8371c340a", + "sha256:d048ad5d25b363ba1d19f92dcf29023988524bee6f9d952130b316c5802069cb", + "sha256:d3e1f3cf81f1f823e7874ae563457828e940d75573c8fbf0ee66818c8b6a9099", + "sha256:d47e9ef1a94cc7a536039e46738e17cce058ac1593b2eccdede8bf72e45f372a", + "sha256:da1e0a8caebf17976e2ffd00fa15f258e14749db5e014660f53114b676e68538", + "sha256:dc1b9b56f051209be458b87edb6856a449ad3f803315d87b2da4c93b43a6fe72", + "sha256:dc2e8fe41f3cac0660197d95216c42910c2b7e9c70d48e6d84e22f577d106fc1", + "sha256:dc92d2d2706d2b862ce0568b24987eba51e17e14b79a1abcd2edc39e48e743c8", + "sha256:dd64f3a4db121bc161644c9e10a9acdb836853155a108c2446db2f5ae1778c3d", + "sha256:e0f0a874231e2839abbf473256efffe577d6ee2e3bfa5b540479e892e47c172d", + "sha256:f7e1f9c5d1160d03b93fc4b68a0aeb82fe25563e12fbcdc8507f8434ab6f823c", + "sha256:fe82d13461418ca5e5a808a9e40f79c1879351fcaeddbede094028e74d836e86" + ], + "version": "==0.22.0" + }, + "websockets": { + "hashes": [ + "sha256:00700340c6c7ab788f176d118775202aadea7602c5cc6be6ae127761c16d6b0b", + "sha256:0bee75f400895aef54157b36ed6d3b308fcab62e5260703add87f44cee9c82a6", + "sha256:0e6e2711d5a8e6e482cacb927a49a3d432345dfe7dea8ace7b5790df5932e4df", + "sha256:12743ab88ab2af1d17dd4acb4645677cb7063ef4db93abffbf164218a5d54c6b", + "sha256:1a9d160fd080c6285e202327aba140fc9a0d910b09e423afff4ae5cbbf1c7205", + "sha256:1bf386089178ea69d720f8db6199a0504a406209a0fc23e603b27b300fdd6892", + "sha256:1df2fbd2c8a98d38a66f5238484405b8d1d16f929bb7a33ed73e4801222a6f53", + "sha256:1e4b3f8ea6a9cfa8be8484c9221ec0257508e3a1ec43c36acdefb2a9c3b00aa2", + "sha256:1f38a7b376117ef7aff996e737583172bdf535932c9ca021746573bce40165ed", + "sha256:23509452b3bc38e3a057382c2e941d5ac2e01e251acce7adc74011d7d8de434c", + "sha256:248d8e2446e13c1d4326e0a6a4e9629cb13a11195051a73acf414812700badbd", + "sha256:25eb766c8ad27da0f79420b2af4b85d29914ba0edf69f547cc4f06ca6f1d403b", + "sha256:27a5e9964ef509016759f2ef3f2c1e13f403725a5e6a1775555994966a66e931", + "sha256:2c71bd45a777433dd9113847af751aae36e448bc6b8c361a566cb043eda6ec30", + "sha256:2cb388a5bfb56df4d9a406783b7f9dbefb888c09b71629351cc6b036e9259370", + "sha256:2d225bb6886591b1746b17c0573e29804619c8f755b5598d875bb4235ea639be", + "sha256:2e5fc14ec6ea568200ea4ef46545073da81900a2b67b3e666f04adf53ad452ec", + "sha256:363f57ca8bc8576195d0540c648aa58ac18cf85b76ad5202b9f976918f4219cf", + "sha256:3c6cc1360c10c17463aadd29dd3af332d4a1adaa8796f6b0e9f9df1fdb0bad62", + "sha256:3d829f975fc2e527a3ef2f9c8f25e553eb7bc779c6665e8e1d52aa22800bb38b", + "sha256:3e3aa8c468af01d70332a382350ee95f6986db479ce7af14d5e81ec52aa2b402", + "sha256:3f61726cae9f65b872502ff3c1496abc93ffbe31b278455c418492016e2afc8f", + "sha256:423fc1ed29f7512fceb727e2d2aecb952c46aa34895e9ed96071821309951123", + "sha256:46e71dbbd12850224243f5d2aeec90f0aaa0f2dde5aeeb8fc8df21e04d99eff9", + "sha256:4d87be612cbef86f994178d5186add3d94e9f31cc3cb499a0482b866ec477603", + "sha256:5693ef74233122f8ebab026817b1b37fe25c411ecfca084b29bc7d6efc548f45", + "sha256:5aa9348186d79a5f232115ed3fa9020eab66d6c3437d72f9d2c8ac0c6858c558", + "sha256:5d873c7de42dea355d73f170be0f23788cf3fa9f7bed718fd2830eefedce01b4", + "sha256:5f6ffe2c6598f7f7207eef9a1228b6f5c818f9f4d53ee920aacd35cec8110438", + "sha256:604428d1b87edbf02b233e2c207d7d528460fa978f9e391bd8aaf9c8311de137", + "sha256:6350b14a40c95ddd53e775dbdbbbc59b124a5c8ecd6fbb09c2e52029f7a9f480", + "sha256:6e2df67b8014767d0f785baa98393725739287684b9f8d8a1001eb2839031447", + "sha256:6e96f5ed1b83a8ddb07909b45bd94833b0710f738115751cdaa9da1fb0cb66e8", + "sha256:6e9e7db18b4539a29cc5ad8c8b252738a30e2b13f033c2d6e9d0549b45841c04", + "sha256:70ec754cc2a769bcd218ed8d7209055667b30860ffecb8633a834dde27d6307c", + "sha256:7b645f491f3c48d3f8a00d1fce07445fab7347fec54a3e65f0725d730d5b99cb", + "sha256:7fa3d25e81bfe6a89718e9791128398a50dec6d57faf23770787ff441d851967", + "sha256:81df9cbcbb6c260de1e007e58c011bfebe2dafc8435107b0537f393dd38c8b1b", + "sha256:8572132c7be52632201a35f5e08348137f658e5ffd21f51f94572ca6c05ea81d", + "sha256:87b4aafed34653e465eb77b7c93ef058516cb5acf3eb21e42f33928616172def", + "sha256:8e332c210b14b57904869ca9f9bf4ca32f5427a03eeb625da9b616c85a3a506c", + "sha256:9893d1aa45a7f8b3bc4510f6ccf8db8c3b62120917af15e3de247f0780294b92", + "sha256:9edf3fc590cc2ec20dc9d7a45108b5bbaf21c0d89f9fd3fd1685e223771dc0b2", + "sha256:9fdf06fd06c32205a07e47328ab49c40fc1407cdec801d698a7c41167ea45113", + "sha256:a02413bc474feda2849c59ed2dfb2cddb4cd3d2f03a2fedec51d6e959d9b608b", + "sha256:a1d9697f3337a89691e3bd8dc56dea45a6f6d975f92e7d5f773bc715c15dde28", + "sha256:a571f035a47212288e3b3519944f6bf4ac7bc7553243e41eac50dd48552b6df7", + "sha256:ab3d732ad50a4fbd04a4490ef08acd0517b6ae6b77eb967251f4c263011a990d", + "sha256:ae0a5da8f35a5be197f328d4727dbcfafa53d1824fac3d96cdd3a642fe09394f", + "sha256:b067cb952ce8bf40115f6c19f478dc71c5e719b7fbaa511359795dfd9d1a6468", + "sha256:b2ee7288b85959797970114deae81ab41b731f19ebcd3bd499ae9ca0e3f1d2c8", + "sha256:b81f90dcc6c85a9b7f29873beb56c94c85d6f0dac2ea8b60d995bd18bf3e2aae", + "sha256:ba0cab91b3956dfa9f512147860783a1829a8d905ee218a9837c18f683239611", + "sha256:baa386875b70cbd81798fa9f71be689c1bf484f65fd6fb08d051a0ee4e79924d", + "sha256:bbe6013f9f791944ed31ca08b077e26249309639313fff132bfbf3ba105673b9", + "sha256:bea88d71630c5900690fcb03161ab18f8f244805c59e2e0dc4ffadae0a7ee0ca", + "sha256:befe90632d66caaf72e8b2ed4d7f02b348913813c8b0a32fae1cc5fe3730902f", + "sha256:c3181df4583c4d3994d31fb235dc681d2aaad744fbdbf94c4802485ececdecf2", + "sha256:c4e37d36f0d19f0a4413d3e18c0d03d0c268ada2061868c1e6f5ab1a6d575077", + "sha256:c588f6abc13f78a67044c6b1273a99e1cf31038ad51815b3b016ce699f0d75c2", + "sha256:cbe83a6bbdf207ff0541de01e11904827540aa069293696dd528a6640bd6a5f6", + "sha256:d554236b2a2006e0ce16315c16eaa0d628dab009c33b63ea03f41c6107958374", + "sha256:dbcf72a37f0b3316e993e13ecf32f10c0e1259c28ffd0a85cee26e8549595fbc", + "sha256:dc284bbc8d7c78a6c69e0c7325ab46ee5e40bb4d50e494d8131a07ef47500e9e", + "sha256:dff6cdf35e31d1315790149fee351f9e52978130cef6c87c4b6c9b3baf78bc53", + "sha256:e469d01137942849cff40517c97a30a93ae79917752b34029f0ec72df6b46399", + "sha256:eb809e816916a3b210bed3c82fb88eaf16e8afcf9c115ebb2bacede1797d2547", + "sha256:ed2fcf7a07334c77fc8a230755c2209223a7cc44fc27597729b8ef5425aa61a3", + "sha256:f44069528d45a933997a6fef143030d8ca8042f0dfaad753e2906398290e2870", + "sha256:f764ba54e33daf20e167915edc443b6f88956f37fb606449b4a5b10ba42235a5", + "sha256:fc4e7fa5414512b481a2483775a8e8be7803a35b30ca805afa4998a84f9fd9e8", + "sha256:ffefa1374cd508d633646d51a8e9277763a9b78ae71324183693959cf94635a7" + ], + "version": "==12.0" }, "wrapt": { "hashes": [ @@ -1788,11 +2372,11 @@ }, "zipp": { "hashes": [ - "sha256:206f5a15f2af3dbaee80769fb7dc6f249695e940acca08dfb2a4769fe61e538b", - "sha256:2884ed22e7d8961de1c9a05142eb69a247f120291bc0206a00a7642f09b5b715" + "sha256:952df858fb3164426c976d9338d3961e8e8b3758e2e059e0f754b8c4262625ee", + "sha256:96dc6ad62f1441bcaccef23b274ec471518daf4fbbc580341204936a5a3dddec" ], "markers": "python_version >= '3.8'", - "version": "==3.18.1" + "version": "==3.19.0" } }, "develop": { @@ -1827,29 +2411,29 @@ }, "boto3": { "hashes": [ - "sha256:60e5dda0b29805fb410bfda1d98e898edaebedac0e6983e9c57cb88e44dfa64e", - "sha256:6c8125310005255ea998bccc3e8353b4df81a96ab105c89c118461f6c54c07c8" + "sha256:4460958d2b0c53bd2195b23ed5d45db2350e514486fe8caeb38b285b30742280", + "sha256:eeb11bca9b19d12baf93436fb8a16b8b824f1f7e8b9bcc722607e862c46b1b08" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==1.34.97" + "version": "==1.34.114" }, "boto3-stubs": { "hashes": [ - "sha256:8aebce123ca023c21bfd5b1f7dc96c53358368bb7224021619943ef7dd5b2947", - "sha256:9f55147d67dc4a113a468f89f4a3dc75b9b3fa976f4f0988b8ba68a872e31e6a" + "sha256:2b51ebdaec9a2845c65a0a0962586cd400ab69ab678c13da4e4d191df1d99bb7", + "sha256:7215e9e273e96f22ef3a7ff4164e7c80e0b5d7f4e196020a04a03488d2b3b281" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==1.34.97" + "version": "==1.34.114" }, "botocore": { "hashes": [ - "sha256:727d5d3e800ac8b705fca6e19b6fefa1e728a81d62a712df9bd32ed0117c740b", - "sha256:a459d060b541beecb50681e6e8a39313cca981e146a59ba7c5229d62f631a016" + "sha256:269cae7ba99081519a9f87d7298e238d9e68ba94eb4f8ddfa906224c34cb8b6c", + "sha256:ec4d42c816e9b2d87a2439ad277e7dda16a4a614ef6839cf66f4c1a58afa547c" ], "markers": "python_version >= '3.8'", - "version": "==1.34.105" + "version": "==1.34.116" }, "botocore-stubs": { "hashes": [ @@ -2064,18 +2648,16 @@ "sha256:ecbfbc00bf55888edda9868a4cf927205de8499e7fabe6c050322298382953f2", "sha256:efd0bf5205240182e0f13bcaea41be4fdf5c22c5129fc7ced4a0282ac86998c9" ], - "index": "pypi", - "markers": "python_version >= '3.7'", "version": "==42.0.7" }, "docker": { "hashes": [ - "sha256:12ba681f2777a0ad28ffbcc846a69c31b4dfd9752b47eb425a274ee269c5e14b", - "sha256:323736fb92cd9418fc5e7133bc953e11a9da04f4483f828b527db553f1e7e5a3" + "sha256:ad8c70e6e3f8926cb8a92619b832b4ea5299e2831c14284663184e200546fa6c", + "sha256:c96b93b7f0a746f9e77d325bcfb87422a3d8bd4f03136ae8a85b37f1898d5fc0" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==7.0.0" + "version": "==7.1.0" }, "idna": { "hashes": [ @@ -2177,11 +2759,11 @@ }, "platformdirs": { "hashes": [ - "sha256:031cd18d4ec63ec53e82dceaac0417d218a6863f7745dfcc9efe7793b7039bdf", - "sha256:17d5a1161b3fd67b390023cb2d3b026bbd40abde6fdb052dfbd3a29c3ba22ee1" + "sha256:2d7a1657e36a80ea911db832a8a6ece5ee53d8de21edd5cc5879af6530b1bfee", + "sha256:38b7b51f512eed9e84a22788b4bce1de17c0adb134d6becb09836e37d8654cd3" ], "markers": "python_version >= '3.8'", - "version": "==4.2.1" + "version": "==4.2.2" }, "pluggy": { "hashes": [ @@ -2207,23 +2789,14 @@ "markers": "python_version >= '3.8'", "version": "==2.18.0" }, - "pyjwt": { - "hashes": [ - "sha256:57e28d156e3d5c10088e0c68abb90bfac3df82b40a71bd0daa20c65ccd5c23de", - "sha256:59127c392cc44c2da5bb3192169a91f429924e17aff6534d70fdc02ab3e04320" - ], - "index": "pypi", - "markers": "python_version >= '3.7'", - "version": "==2.8.0" - }, "pytest": { "hashes": [ - "sha256:1733f0620f6cda4095bbf0d9ff8022486e91892245bb9e7d5542c018f612f233", - "sha256:d507d4482197eac0ba2bae2e9babf0672eb333017bcedaa5fb1a3d42c1174b3f" + "sha256:5046e5b46d8e4cac199c373041f26be56fdb81eb4e67dc11d4e10811fc3408fd", + "sha256:faccc5d332b8c3719f40283d0d44aa5cf101cec36f88cde9ed8f2bc0538612b1" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==8.2.0" + "version": "==8.2.1" }, "python-dateutil": { "hashes": [ @@ -2235,44 +2808,43 @@ }, "requests": { "hashes": [ - "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f", - "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1" + "sha256:55365417734eb18255590a9ff9eb97e9e1da868d4ccd6402399eaf68af20a760", + "sha256:70761cfe03c773ceb22aa2f671b4757976145175cdfca038c02654d061d6dcc6" ], - "markers": "python_version >= '3.7'", - "version": "==2.31.0" + "markers": "python_version >= '3.8'", + "version": "==2.32.3" }, "rich": { "hashes": [ "sha256:4edbae314f59eb482f54e9e30bf00d33350aaa94f4bfcd4e9e3110e64d0d7222", "sha256:9be308cb1fe2f1f57d67ce99e95af38a1e2bc71ad9813b0e247cf7ffbcc3a432" ], - "index": "pypi", "markers": "python_full_version >= '3.7.0'", "version": "==13.7.1" }, "ruff": { "hashes": [ - "sha256:0e2e06459042ac841ed510196c350ba35a9b24a643e23db60d79b2db92af0c2b", - "sha256:1f32cadf44c2020e75e0c56c3408ed1d32c024766bd41aedef92aa3ca28eef68", - "sha256:22e306bf15e09af45ca812bc42fa59b628646fa7c26072555f278994890bc7ac", - "sha256:24016ed18db3dc9786af103ff49c03bdf408ea253f3cb9e3638f39ac9cf2d483", - "sha256:33bcc160aee2520664bc0859cfeaebc84bb7323becff3f303b8f1f2d81cb4edc", - "sha256:3afabaf7ba8e9c485a14ad8f4122feff6b2b93cc53cd4dad2fd24ae35112d5c5", - "sha256:5ec481661fb2fd88a5d6cf1f83403d388ec90f9daaa36e40e2c003de66751798", - "sha256:652e4ba553e421a6dc2a6d4868bc3b3881311702633eb3672f9f244ded8908cd", - "sha256:6a2243f8f434e487c2a010c7252150b1fdf019035130f41b77626f5655c9ca22", - "sha256:6ab165ef5d72392b4ebb85a8b0fbd321f69832a632e07a74794c0e598e7a8376", - "sha256:7891ee376770ac094da3ad40c116258a381b86c7352552788377c6eb16d784fe", - "sha256:799eb468ea6bc54b95527143a4ceaf970d5aa3613050c6cff54c85fda3fde480", - "sha256:82986bb77ad83a1719c90b9528a9dd663c9206f7c0ab69282af8223566a0c34e", - "sha256:8772130a063f3eebdf7095da00c0b9898bd1774c43b336272c3e98667d4fb8fa", - "sha256:8d14dc8953f8af7e003a485ef560bbefa5f8cc1ad994eebb5b12136049bbccc5", - "sha256:cbd1e87c71bca14792948c4ccb51ee61c3296e164019d2d484f3eaa2d360dfaf", - "sha256:ec4ba9436a51527fb6931a8839af4c36a5481f8c19e8f5e42c2f7ad3a49f5069" + "sha256:04a80acfc862e0e1630c8b738e70dcca03f350bad9e106968a8108379e12b31f", + "sha256:0cf5cc02d3ae52dfb0c8a946eb7a1d6ffe4d91846ffc8ce388baa8f627e3bd50", + "sha256:1fa8561489fadf483ffbb091ea94b9c39a00ed63efacd426aae2f197a45e67fc", + "sha256:1ff930d6e05f444090a0139e4e13e1e2e1f02bd51bb4547734823c760c621e79", + "sha256:3a6a0a4f4b5f54fff7c860010ab3dd81425445e37d35701a965c0248819dde7a", + "sha256:3f9ced5cbb7510fd7525448eeb204e0a22cabb6e99a3cb160272262817d49786", + "sha256:4d5b914818d8047270308fe3e85d9d7f4a31ec86c6475c9f418fbd1624d198e0", + "sha256:4f02284335c766678778475e7698b7ab83abaf2f9ff0554a07b6f28df3b5c259", + "sha256:602ebd7ad909eab6e7da65d3c091547781bb06f5f826974a53dbe563d357e53c", + "sha256:735a16407a1a8f58e4c5b913ad6102722e80b562dd17acb88887685ff6f20cf6", + "sha256:9018bf59b3aa8ad4fba2b1dc0299a6e4e60a4c3bc62bbeaea222679865453062", + "sha256:a769ae07ac74ff1a019d6bd529426427c3e30d75bdf1e08bb3d46ac8f417326a", + "sha256:a797a87da50603f71e6d0765282098245aca6e3b94b7c17473115167d8dfb0b7", + "sha256:be47700ecb004dfa3fd4dcdddf7322d4e632de3c06cd05329d69c45c0280e618", + "sha256:ea3424793c29906407e3cf417f28fc33f689dacbbadfb52b7e9a809dd535dcef", + "sha256:ef995583a038cd4a7edf1422c9e19118e2511b8ba0b015861b4abd26ec5367c5", + "sha256:f13410aabd3b5776f9c5699f42b37a3a348d65498c4310589bc6e5c548dc8a2f" ], "index": "pypi", "markers": "python_version >= '3.7'", - "version": "==0.4.2" + "version": "==0.4.6" }, "s3transfer": { "hashes": [ @@ -2300,21 +2872,21 @@ }, "types-docker": { "hashes": [ - "sha256:9a1245de5163c775665af3d473453aff490749c16720c1b086b2fd7de794c5e1", - "sha256:d2fe85da198ed5dbb988b6c1223e0893fd93b9316ed04fd79435fd0f12a59e29" + "sha256:c0bbbcbb487a66b3fa3b6fde5771ce9e7ccfc427723c1afea0463c671f3c58bc", + "sha256:fca00cb7d6da11401018d4dbf06232c8f3a9e98323971476fe3d4481374e1ebf" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==7.0.0.20240513" + "version": "==7.0.0.20240528" }, "types-requests": { "hashes": [ - "sha256:4428df33c5503945c74b3f42e82b181e86ec7b724620419a2966e2de604ce1a1", - "sha256:6216cdac377c6b9a040ac1c0404f7284bd13199c0e1bb235f4324627e8898cf5" + "sha256:26b8a6de32d9f561192b9942b41c0ab2d8010df5677ca8aa146289d11d505f57", + "sha256:f19ed0e2daa74302069bbbbf9e82902854ffa780bc790742a810a9aaa52f65ec" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==2.31.0.20240406" + "version": "==2.32.0.20240523" }, "types-s3transfer": { "hashes": [ @@ -2326,19 +2898,19 @@ }, "typing-extensions": { "hashes": [ - "sha256:83f085bd5ca59c80295fc2a82ab5dac679cbe02b9f33f7d83af68e241bea51b0", - "sha256:c1f94d72897edaf4ce775bb7558d5b79d8126906a14ea5ed1635921406c0387a" + "sha256:8cbcdc8606ebcb0d95453ad7dc5065e6237b6aa230a31e81d0f440c30fed5fd8", + "sha256:b349c66bea9016ac22978d800cfff206d5f9816951f12a7d0ec5578b0a819594" ], "markers": "python_version >= '3.8'", - "version": "==4.11.0" + "version": "==4.12.0" }, "urllib3": { "hashes": [ - "sha256:c97dfde1f7bd43a71c8d2a58e369e9b2bf692d1334ea9f9cae55add7d0dd0f84", - "sha256:fdb6d215c776278489906c2f8916e6e7d4f5a9b602ccbcfdf7f016fc8da0596e" + "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d", + "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19" ], - "markers": "python_version >= '3.11'", - "version": "==2.0.7" + "markers": "python_version >= '3.8'", + "version": "==2.2.1" } } } diff --git a/opentrons-ai-server/README.md b/opentrons-ai-server/README.md index 27a8855b44d..9221d8ff590 100644 --- a/opentrons-ai-server/README.md +++ b/opentrons-ai-server/README.md @@ -20,9 +20,11 @@ The Opentrons AI application's server. 1. This will create a `.python-version` file in this directory 1. select the node version with `nvs` or `nvm` currently 18.19\* 1. Install pipenv and python dependencies `make setup` -1. to build and deploy you must have -1. AWS credentials and the right roles -1. docker installed + +## For building and deploying + +1. AWS credentials and config +1. docker ## Install a dev dependency @@ -30,32 +32,29 @@ The Opentrons AI application's server. ## Install a production dependency -`python -m pipenv install openai==1.25.1` +`python -m pipenv install openai==1.30.4` -## Lambda Code Organizations and Separation of Concerns +## FastAPI Code Organization and Separation of Concerns - handler - - the lambda handler + - the router and request/response handling - domain - - the business logic + - business logic - integration - - the integration with other services + - integration with other services ## Dev process 1. Make your changes -1. Fix what can be automatically then lent and unit test like CI will `make pre-commit` +1. Fix what can be automatically then lint and unit test like CI will `make pre-commit` 1. `make pre-commit` passes -1. deploy to sandbox `make deploy test-live ENV=sandbox AWS_PROFILE=the-profile` - -## Custom runtime - -- Due to the size requirements of `llama-index` and our data we switched to a custom runtime -- This also allows us to use HTTP streaming -- The runtime is defined in the `Dockerfile` -- deploy.py contains the steps to - 1. build the container image - 1. tag the container image (currently uses the epoch until versioning in place) - 1. log into and push to the correct ECR - 1. create a new lambda version against the new image - 1. await the function to be ready +1. run locally `make run` this runs the FastAPI server directly at localhost:8000 + 1. this watches for changes and restarts the server +1. test locally `make live-test` (ENV=local is the default in the Makefile) +1. use the live client `make live-client` + +## ECS Fargate + +- Our first version of this service is a long running POST that may take from 1-3 minutes to complete +- This forces us to use CloudFront(Max 180) + Load Balancer + ECS Fargate FastAPI container +- An AWS service ticket is needed to increase the max CloudFront response time from 60 to 180 seconds diff --git a/opentrons-ai-server/api/domain/openai_predict.py b/opentrons-ai-server/api/domain/openai_predict.py index d6a0d7706c9..37de011d579 100644 --- a/opentrons-ai-server/api/domain/openai_predict.py +++ b/opentrons-ai-server/api/domain/openai_predict.py @@ -1,3 +1,4 @@ +import logging from pathlib import Path from typing import List, Tuple @@ -22,7 +23,9 @@ tools, ) from api.domain.utils import refine_characters -from api.settings import Settings, is_running_on_lambda +from api.settings import Settings + +logger = logging.getLogger(__name__) ROOT_PATH: Path = Path(Path(__file__)).parent.parent.parent @@ -37,7 +40,7 @@ def __init__(self, settings: Settings) -> None: def get_docs_all(self, query: str) -> Tuple[str, str, str]: commands = self.extract_atomic_description(query) - print(f"commands: {commands}") + logger.info("Commands", extra={"commands": commands}) # define file paths for storage example_command_path = str(ROOT_PATH / "api" / "storage" / "index" / "commands") @@ -84,28 +87,27 @@ class atomic_descr(BaseModel): output_cls=atomic_descr, prompt_template_str=prompt_template_str.format(protocol_description=protocol_description), verbose=False, - llm=li_OpenAI(model=self.settings.OPENAI_MODEL_NAME), + llm=li_OpenAI(model=self.settings.openai_model_name, api_key=self.settings.openai_api_key.get_secret_value()), ) details = program(protocol_description=protocol_description) descriptions = [] - print("=" * 50) for x in details.desc: if x not in ["Modules:", "Adapter:", "Labware:", "Pipette mount:", "Commands:", "Well Allocation:", "No modules"]: descriptions.append(x) return descriptions - def refine_response(self, assitant_message: str) -> str: - if assitant_message is None: + def refine_response(self, assistant_message: str) -> str: + if assistant_message is None: return "" system_message: ChatCompletionMessageParam = { "role": "system", "content": f"{general_rules_1}\n Please leave useful comments for each command.", } - user_message: ChatCompletionMessageParam = {"role": "user", "content": assitant_message} + user_message: ChatCompletionMessageParam = {"role": "user", "content": assistant_message} response = self.client.chat.completions.create( - model=self.settings.OPENAI_MODEL_NAME, + model=self.settings.openai_model_name, messages=[system_message, user_message], stream=False, temperature=0.005, @@ -137,7 +139,7 @@ def predict(self, prompt: str, chat_completion_message_params: List[ChatCompleti messages.append(user_message) response: ChatCompletion = self.client.chat.completions.create( - model=self.settings.OPENAI_MODEL_NAME, + model=self.settings.openai_model_name, messages=messages, stream=False, temperature=0.005, @@ -155,7 +157,7 @@ def predict(self, prompt: str, chat_completion_message_params: List[ChatCompleti assistant_message.content = str(self.refine_response(assistant_message.content)) if assistant_message.tool_calls and assistant_message.tool_calls[0]: - print("Simulation is started.") + logger.info("Simulation has started") if assistant_message.tool_calls[0]: assistant_message.content = str(assistant_message.tool_calls[0].function) messages.append({"role": assistant_message.role, "content": assistant_message.content}) @@ -167,7 +169,7 @@ def predict(self, prompt: str, chat_completion_message_params: List[ChatCompleti ChatCompletionFunctionMessageParam(role="function", name=tool_call.function.name, content=str(function_response)) ) response2: ChatCompletion = self.client.chat.completions.create( - model=self.settings.OPENAI_MODEL_NAME, + model=self.settings.openai_model_name, messages=messages, stream=False, temperature=0, @@ -183,12 +185,10 @@ def predict(self, prompt: str, chat_completion_message_params: List[ChatCompleti def main() -> None: """Intended for testing this class locally.""" - if is_running_on_lambda(): - return from rich import print from rich.prompt import Prompt - settings = Settings.build() + settings = Settings() openai = OpenAIPredict(settings) prompt = Prompt.ask("Type a prompt to send to the OpenAI API:") completion = openai.predict(prompt) diff --git a/opentrons-ai-server/api/domain/prompts.py b/opentrons-ai-server/api/domain/prompts.py index 001cc40c290..e7a8a3c448d 100644 --- a/opentrons-ai-server/api/domain/prompts.py +++ b/opentrons-ai-server/api/domain/prompts.py @@ -1,4 +1,5 @@ import json +import logging import uuid from typing import Any, Dict, Iterable @@ -7,7 +8,8 @@ from api.settings import Settings -settings = Settings.build() +settings: Settings = Settings() +logger = logging.getLogger(__name__) def generate_unique_name() -> str: @@ -25,23 +27,23 @@ def send_post_request(payload: str) -> str: response = requests.post(url, json=data, headers=headers) if response.status_code != 200: - print("Error: " + response.text) + logger.error("Error: " + response.text) return "Error: " + response.text # Check the response before returning it # ToDo clean up code response_data: Dict[str, Any] = response.json() if "error_message" in response_data: - print("Error in response:", response_data["error_message"]) + logger.error("Error in response:", response_data["error_message"]) return str(response_data["error_message"]) elif "protocol_name" in response_data: - # print("Protocol executed successfully. Run log:", response_data["run_log"]) + logger.debug("Protocol executed successfully", extra={"response_data": response_data["run_log"]}) return str(response_data["run_status"]) # ToDo if run_log option is on # return response_data["run_log"] else: - print("Unexpected response:", response_data) + logger.info("Unexpected response:", extra={"response_data": response_data}) return "Unexpected response" diff --git a/opentrons-ai-server/api/handler/fast.py b/opentrons-ai-server/api/handler/fast.py new file mode 100644 index 00000000000..dba406e8767 --- /dev/null +++ b/opentrons-ai-server/api/handler/fast.py @@ -0,0 +1,252 @@ +import asyncio +import os +from typing import Any, Awaitable, Callable, Literal, Union + +import ddtrace +from ddtrace import tracer +from fastapi import FastAPI, HTTPException, Query, Request, Response, Security, status +from fastapi.exceptions import RequestValidationError +from fastapi.middleware.cors import CORSMiddleware +from fastapi.openapi.docs import get_redoc_html, get_swagger_ui_html +from fastapi.responses import HTMLResponse, JSONResponse +from pydantic import BaseModel, Field, conint +from starlette.middleware.base import BaseHTTPMiddleware + +from api.domain.openai_predict import OpenAIPredict +from api.handler.logging_config import get_logger, setup_logging +from api.integration.auth import VerifyToken +from api.models.chat_request import ChatRequest +from api.models.chat_response import ChatResponse +from api.models.empty_request_error import EmptyRequestError +from api.models.internal_server_error import InternalServerError +from api.settings import Settings + +setup_logging() +logger = get_logger(__name__) +ddtrace.patch(logging=True) +settings: Settings = Settings() +auth: VerifyToken = VerifyToken() +openai: OpenAIPredict = OpenAIPredict(settings) + + +# Initialize FastAPI app with metadata +app = FastAPI( + title="Opentrons AI API", + description="An API for generating chat responses.", + version=os.getenv("DD_VERSION", "local"), + openapi_url="/api/openapi.json", +) + +# CORS and PREFLIGHT settings +# ALLOWED_ORIGINS is now an environment variable +ALLOWED_CREDENTIALS: bool = True +ALLOWED_METHODS: str = "GET,POST,OPTIONS" +ALLOWED_HEADERS: str = "content-type,authorization,origin,accept" +ALLOWED_ACCESS_CONTROL_EXPOSE_HEADERS: str = "content-type" +ALLOWED_ACCESS_CONTROL_MAX_AGE: str = "600" + +# Add CORS middleware +app.add_middleware( + CORSMiddleware, + allow_origins=settings.allowed_origins, + allow_credentials=ALLOWED_CREDENTIALS, + allow_methods=ALLOWED_METHODS, + allow_headers=ALLOWED_HEADERS, +) + + +# Add Timeout middleware +class TimeoutMiddleware(BaseHTTPMiddleware): + def __init__(self, app: FastAPI, timeout_s: int) -> None: + super().__init__(app) + self.timeout_s = timeout_s + + async def dispatch(self, request: Request, call_next: Any) -> JSONResponse | Any: + try: + return await asyncio.wait_for(call_next(request), timeout=self.timeout_s) + except asyncio.TimeoutError: + return JSONResponse({"detail": "API Request timed out"}, status_code=504) + + +# Control the timeout message by timing out before cloudfront would +# 2 seconds before the CloudFront timeout (180 seconds) +# 12 second before the uvicorn timeout (190 seconds) +# 22 seconds before the ALB timeout (200 seconds) +app.add_middleware(TimeoutMiddleware, timeout_s=178) + + +# Models +class Status(BaseModel): + status: Literal["ok", "error"] + version: str + + +class ErrorResponse(BaseModel): + message: str + + +class HealthResponse(BaseModel): + status: Status + + +class TimeoutResponse(BaseModel): + message: str + + +class CorsHeadersResponse(BaseModel): + Access_Control_Allow_Origin: str = Field(alias="Access-Control-Allow-Origin") + Access_Control_Allow_Methods: str = Field(alias="Access-Control-Allow-Methods") + Access_Control_Allow_Headers: str = Field(alias="Access-Control-Allow-Headers") + Access_Control_Expose_Headers: str = Field(alias="Access-Control-Expose-Headers") + Access_Control_Max_Age: str = Field(alias="Access-Control-Max-Age") + + +@tracer.wrap() +@app.post( + "/api/chat/completion", + response_model=Union[ChatResponse, ErrorResponse], + summary="Create Chat Completion", + description="Generate a chat response based on the provided prompt.", +) +async def create_chat_completion( + body: ChatRequest, auth_result: Any = Security(auth.verify) # noqa: B008 +) -> Union[ChatResponse, ErrorResponse]: # noqa: B008 + """ + Generate a chat completion response using OpenAI. + + - **request**: The HTTP request containing the chat message. + - **returns**: A chat response or an error message. + """ + logger.info("POST /api/chat/completion", extra={"body": body.model_dump(), "auth_result": auth_result}) + try: + if not body.message or body.message == "": + raise HTTPException( + status_code=status.HTTP_400_BAD_REQUEST, detail=EmptyRequestError(message="Request body is empty").model_dump() + ) + + if body.fake: + return ChatResponse(reply="Fake response", fake=body.fake) + response: Union[str, None] = openai.predict(prompt=body.message, chat_completion_message_params=body.history) + + if response is None or response == "": + return ChatResponse(reply="No response was generated", fake=body.fake) + + return ChatResponse(reply=response, fake=body.fake) + + except Exception as e: + logger.exception(e) + raise HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, detail=InternalServerError(exception_object=e).model_dump() + ) from e + + +@app.get( + "/health", + response_model=Status, + summary="LB Health Check", + description="Check the health and version of the API.", + include_in_schema=False, +) +@app.get("/api/health", response_model=Status, summary="Health Check", description="Check the health and version of the API.") +async def get_health(request: Request) -> Status: + """ + Perform a health check of the API. + + - **returns**: A Status containing the version of the API. + """ + logger.debug(f"{request.method} {request.url.path}") + return Status(status="ok", version=settings.dd_version) + + +@app.get("/api/timeout", response_model=TimeoutResponse) +async def timeout_endpoint(request: Request, seconds: conint(ge=1, le=300) = Query(..., description="Number of seconds to wait")): # type: ignore # noqa: B008 + """ + Wait for the specified number of seconds and then respond. + + - **seconds**: The number of seconds to wait (between 1 and 300). + """ + # call me with http://localhost:8000/api/timeout?seconds=180 + logger.info(f"{request.method} {request.url.path}") + try: + await asyncio.sleep(seconds) # Asynchronously wait for the specified time + return TimeoutResponse(message=f"Waited for {seconds} seconds") + except Exception as e: + raise HTTPException(status_code=500, detail=str(e)) from e + + +@app.get("/api/redoc", include_in_schema=False) +async def redoc_html() -> HTMLResponse: + return get_redoc_html(openapi_url="/api/openapi.json", title="Opentrons API Documentation") + + +@app.get("/api/doc", include_in_schema=False) +async def swagger_html() -> HTMLResponse: + return get_swagger_ui_html(openapi_url="/api/openapi.json", title="Opentrons API Documentation") + + +@app.options( + "/{path:path}", response_model=CorsHeadersResponse, summary="CORS Preflight Request", description="Handle CORS preflight requests." +) +async def handle_options(request: Request) -> JSONResponse: + """ + Handle CORS preflight requests. + + This endpoint responds to CORS preflight requests with the appropriate headers. + + - **returns**: CORS headers. + """ + logger.info(f"{request.method} {request.url.path}") + response = CorsHeadersResponse.model_validate( + { + "Access-Control-Allow-Origin": settings.allowed_origins, + "Access-Control-Allow-Methods": ALLOWED_METHODS, + "Access-Control-Allow-Headers": ALLOWED_HEADERS, + "Access-Control-Expose-Headers": ALLOWED_ACCESS_CONTROL_EXPOSE_HEADERS, + "Access-Control-Max-Age": ALLOWED_ACCESS_CONTROL_MAX_AGE, + } + ) + return JSONResponse(response.model_dump(by_alias=True)) + + +# General exception handler for validation errors +@app.exception_handler(RequestValidationError) +async def validation_exception_handler(request: Request, exc: RequestValidationError) -> JSONResponse: + """ + Handle validation errors. + + - **request**: The HTTP request that caused the error. + - **exc**: The validation exception that was raised. + - **returns**: A JSON response with a 422 status code and error details. + """ + logger.error(f"Validation error for route {request.url.path}: {exc}") + return JSONResponse(status_code=status.HTTP_422_UNPROCESSABLE_ENTITY, content={"message": "Validation error", "details": exc.errors()}) + + +@app.middleware("http") +async def custom_404_middleware(request: Request, call_next: Callable[[Request], Awaitable[Response]]) -> Response: + try: + response = await call_next(request) + if response.status_code in (status.HTTP_404_NOT_FOUND, status.HTTP_405_METHOD_NOT_ALLOWED): + logger.info(f"Route not found: {request.url.path}") + return JSONResponse(status_code=status.HTTP_404_NOT_FOUND, content={"detail": f"Route '{request.url.path}' not found"}) + return response + except Exception as exc: + logger.error(f"Error processing request: {exc}") + raise exc + + +# Catch-all handler for any other uncaught exceptions +@app.middleware("http") +async def catch_all_exceptions(request: Request, call_next: Any) -> JSONResponse | Any: + """ + Catch all uncaught exceptions. + + - **request**: The HTTP request that caused the error. + - **call_next**: The next middleware or route handler. + - **returns**: A JSON response with a 500 status code if an exception is raised. + """ + try: + return await call_next(request) + except Exception as exc: + logger.error(f"Unhandled error for route {request.url.path}: {exc}") + return JSONResponse(status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, content={"message": "Internal server error"}) diff --git a/opentrons-ai-server/api/handler/function.py b/opentrons-ai-server/api/handler/function.py deleted file mode 100644 index 8bc0113c698..00000000000 --- a/opentrons-ai-server/api/handler/function.py +++ /dev/null @@ -1,82 +0,0 @@ -import json -import logging -from http import HTTPStatus -from typing import Any, Dict, Union - -from api.domain.openai_predict import OpenAIPredict -from api.models.chat_request import ChatRequest -from api.models.chat_response import ChatResponse -from api.models.empty_request_error import EmptyRequestError -from api.models.internal_server_error import InternalServerError -from api.settings import Settings - -logger = logging.getLogger() - - -def create_response(status_code: int, body: Any, content_type: str = "application/json") -> Dict[str, Any]: - return {"statusCode": status_code, "headers": {"Content-Type": content_type}, "body": json.dumps(body)} - - -def create_chat_completion(event: Dict[str, Any]) -> Dict[str, Any]: - logger.info("POST /chat/completion", extra={"event": event}) - try: - if not event.get("body"): - return create_response(HTTPStatus.BAD_REQUEST, EmptyRequestError(message="Request body is empty").model_dump()) - - body: ChatRequest = ChatRequest.model_validate_json(event["body"]) - if body.fake: - return create_response(HTTPStatus.OK, ChatResponse(reply="Fake response", fake=body.fake).model_dump()) - - settings: Settings = Settings.build() - openai: OpenAIPredict = OpenAIPredict(settings=settings) - response: Union[str, None] = openai.predict(prompt=body.message) - - if response is None or response == "": - return create_response(HTTPStatus.NO_CONTENT, ChatResponse(reply="No response was generated", fake=body.fake).model_dump()) - - return create_response(HTTPStatus.OK, ChatResponse(reply=response, fake=body.fake).model_dump()) - - except Exception as e: - logger.exception("Error processing request", extra={"error": str(e)}) - return create_response(HTTPStatus.INTERNAL_SERVER_ERROR, InternalServerError(exception_object=e).model_dump()) - - -def get_health(event: Dict[str, Any]) -> Dict[str, Any]: - logger.info("GET /health", extra={"event": event}) - return create_response(HTTPStatus.OK, {"version": "0.0.1"}) - - -def get_options(event: Dict[str, Any]) -> Dict[str, Any]: - """These are the CORS headers that are returned when an OPTIONS request is made""" - # These match the settings in terraform - allowed_origins = ",".join(["*"]) - allowed_methods = ",".join(["GET", "POST", "OPTIONS"]) - allowed_headers = ",".join(["content-type", "authorization", "origin", "accept"]) - expose_headers = ",".join(["content-type"]) - - cors_headers = { - "Access-Control-Allow-Origin": allowed_origins, - "Access-Control-Allow-Methods": allowed_methods, - "Access-Control-Allow-Headers": allowed_headers, - "Access-Control-Expose-Headers": expose_headers, - "Access-Control-Max-Age": "3600", - } - - return create_response(HTTPStatus.OK, cors_headers) - - -def handler(event: Dict[str, Any], context: Any) -> Dict[str, Any]: - settings: Settings = Settings.build() - raw_path: str = event.get("rawPath", "") - method: str = event.get("requestContext", {}).get("http", {}).get("method", "") - logger.info(f"path: {raw_path}, http_method: {method}") - # the below is not robust, this is to get things working - # TODO: this should be its own method with unit tests - if raw_path.lower() == f"/{settings.ENVIRONMENT}/chat/completion" and method.upper() == "POST": - return create_chat_completion(event) - elif raw_path == f"/{settings.ENVIRONMENT}/health".lower() and method.upper() == "GET": - return get_health(event) - elif method.upper() == "OPTIONS": - return get_options(event) - else: - return create_response(HTTPStatus.NOT_FOUND, {"message": f"path {raw_path} method {method} not found"}) diff --git a/opentrons-ai-server/api/handler/local_run.py b/opentrons-ai-server/api/handler/local_run.py new file mode 100644 index 00000000000..0b82fae7e41 --- /dev/null +++ b/opentrons-ai-server/api/handler/local_run.py @@ -0,0 +1,9 @@ +# run.py +import uvicorn + +from api.handler.logging_config import setup_logging + +setup_logging() + +if __name__ == "__main__": + uvicorn.run("api.handler.fast:app", host="localhost", port=8000, timeout_keep_alive=190, reload=True) diff --git a/opentrons-ai-server/api/handler/logging_config.py b/opentrons-ai-server/api/handler/logging_config.py new file mode 100644 index 00000000000..a7efdadeef8 --- /dev/null +++ b/opentrons-ai-server/api/handler/logging_config.py @@ -0,0 +1,31 @@ +# logging_config.py +import logging + +from pythonjsonlogger import jsonlogger + +from api.settings import Settings + +FORMAT = ( + "%(asctime)s %(levelname)s [%(name)s] [%(filename)s:%(lineno)d] " + "[dd.service=%(dd.service)s dd.env=%(dd.env)s dd.version=%(dd.version)s dd.trace_id=%(dd.trace_id)s dd.span_id=%(dd.span_id)s] " + "- %(message)s" +) + + +def setup_logging() -> None: + log_handler = logging.StreamHandler() + formatter = jsonlogger.JsonFormatter(FORMAT) # type: ignore + log_handler.setFormatter(formatter) + + logging.basicConfig( + level=Settings().log_level.upper(), + handlers=[log_handler], + ) + + +# Call this function to initialize logging +setup_logging() + + +def get_logger(name: str) -> logging.Logger: + return logging.getLogger(name) diff --git a/opentrons-ai-server/api/integration/auth.py b/opentrons-ai-server/api/integration/auth.py new file mode 100644 index 00000000000..c3cf4b8d163 --- /dev/null +++ b/opentrons-ai-server/api/integration/auth.py @@ -0,0 +1,60 @@ +import logging +from typing import Any, Optional + +import jwt +from fastapi import HTTPException, Security, status +from fastapi.security import HTTPAuthorizationCredentials, HTTPBearer, SecurityScopes + +from api.settings import Settings + +logger = logging.getLogger(__name__) + + +class UnauthenticatedException(HTTPException): + def __init__(self) -> None: + super().__init__(status_code=status.HTTP_401_UNAUTHORIZED, detail="This request was not authorized correctly.") + + +class VerifyToken: + """Does all the token verification using PyJWT""" + + def __init__(self) -> None: + self.config = Settings() + + # This gets the JWKS from a given URL and does processing so you can + # use any of the keys available + jwks_url = f"https://{self.config.auth0_domain}/.well-known/jwks.json" + self.jwks_client = jwt.PyJWKClient(jwks_url) + + async def verify( + self, security_scopes: SecurityScopes, credentials: Optional[HTTPAuthorizationCredentials] = Security(HTTPBearer()) # noqa: B008 + ) -> Any: + if credentials is None: + raise UnauthenticatedException() + + try: + signing_key = self.jwks_client.get_signing_key_from_jwt(credentials.credentials).key + except jwt.PyJWKClientError as error: + logger.error(error, extra={"credentials": credentials}) + raise UnauthenticatedException() from error + except jwt.exceptions.DecodeError as error: + logger.error(error, extra={"credentials": credentials}) + raise UnauthenticatedException() from error + + try: + payload = jwt.decode( + credentials.credentials, + signing_key, + algorithms=[self.config.auth0_algorithms], + audience=self.config.auth0_api_audience, + issuer=self.config.auth0_issuer, + ) + logger.info("Decoded token", extra={"token": payload}) + return payload + except jwt.ExpiredSignatureError as error: + logger.error(error, extra={"credentials": credentials}) + # Handle token expiration, e.g., refresh token, re-authenticate, etc. + except jwt.PyJWTError as error: + logger.error(error, extra={"credentials": credentials}) + # Handle other JWT errors + raise UnauthenticatedException() diff --git a/opentrons-ai-server/api/models/chat_request.py b/opentrons-ai-server/api/models/chat_request.py index 77d714b234d..b3b9cd5b7ee 100644 --- a/opentrons-ai-server/api/models/chat_request.py +++ b/opentrons-ai-server/api/models/chat_request.py @@ -1,6 +1,17 @@ -from pydantic import BaseModel +from typing import List, Optional + +from openai.types.chat import ChatCompletionMessageParam +from pydantic import BaseModel, Field + + +class Chat(BaseModel): + role: str + content: str class ChatRequest(BaseModel): - message: str - fake: bool + message: str = Field(..., description="The latest message to be processed.") + history: Optional[List[ChatCompletionMessageParam]] = Field( + None, description="Chat history in the form of a list of messages. Type is from OpenAI's ChatCompletionMessageParam" + ) + fake: bool = Field(True, description="When set to true, the response will be a fake. OpenAI API is not used.") diff --git a/opentrons-ai-server/api/settings.py b/opentrons-ai-server/api/settings.py index ae6a28eb797..b110a4b72e0 100644 --- a/opentrons-ai-server/api/settings.py +++ b/opentrons-ai-server/api/settings.py @@ -1,78 +1,60 @@ -import os -from dataclasses import asdict, dataclass from pathlib import Path -from typing import Type -from dotenv import load_dotenv from pydantic import SecretStr - -from api.integration.aws_secrets_manager import fetch_secret +from pydantic_settings import BaseSettings, SettingsConfigDict ENV_PATH: Path = Path(Path(__file__).parent.parent, ".env") -def is_running_on_lambda() -> bool: - """Check if the script is running on AWS Lambda.""" - return "AWS_LAMBDA_FUNCTION_NAME" in os.environ - - -@dataclass(frozen=True) -class Settings: - HUGGINGFACE_SIMULATE_ENDPOINT: str - LOG_LEVEL: str - SERVICE_NAME: str - ENVIRONMENT: str - OPENAI_MODEL_NAME: str - openai_api_key: SecretStr - huggingface_api_key: SecretStr - - @classmethod - def build(cls: Type["Settings"]) -> "Settings": - # Load environment variables from .env file if it exists - load_dotenv(ENV_PATH) - - environment = os.getenv("ENVIRONMENT", "local") - service_name = os.getenv("SERVICE_NAME", "local-ai-api") - openai_model_name = os.getenv("OPENAI_MODEL_NAME", "gpt-4-1106-preview") - huggingface_simulate_endpoint = os.getenv("HUGGINGFACE_SIMULATE_ENDPOINT", "https://Opentrons-simulator.hf.space/protocol") - log_level = os.getenv("LOG_LEVEL", "debug") - - if is_running_on_lambda(): - openai_api_key = fetch_secret(f"{environment}-openai-api-key") - huggingface_api_key = fetch_secret(f"{environment}-huggingface-api-key") - else: - openai_api_key = SecretStr(os.getenv("OPENAI_API_KEY", "")) - huggingface_api_key = SecretStr(os.getenv("HUGGINGFACE_API_KEY", "")) - - return cls( - HUGGINGFACE_SIMULATE_ENDPOINT=huggingface_simulate_endpoint, - LOG_LEVEL=log_level, - SERVICE_NAME=service_name, - ENVIRONMENT=environment, - OPENAI_MODEL_NAME=openai_model_name, - openai_api_key=openai_api_key, - huggingface_api_key=huggingface_api_key, - ) +class Settings(BaseSettings): + """ + If the env_file file exists: It will read the configurations from the env_file file (local execution) + If the env_file file does not exist: + It will read the configurations from the environment variables set in the operating system (deployed execution) + If the variable is not set in the OS the default value is used (this is just for creating the .env file with default values) + """ - @staticmethod - def get_service_name() -> str: - return os.getenv("SERVICE_NAME", "local-ai-api") + model_config = SettingsConfigDict(env_file=ENV_PATH, env_file_encoding="utf-8") + environment: str = "local" + huggingface_simulate_endpoint: str = "https://Opentrons-simulator.hf.space/protocol" + log_level: str = "info" + service_name: str = "local-ai-api" + openai_model_name: str = "gpt-4-1106-preview" + auth0_domain: str = "opentrons-dev.us.auth0.com" + auth0_api_audience: str = "sandbox-ai-api" + auth0_issuer: str = "https://identity.auth-dev.opentrons.com/" + auth0_algorithms: str = "RS256" + dd_version: str = "hardcoded_default_from_settings" + allowed_origins: str = "*" + dd_logs_injection: str = "true" + + # Secrets + # These come from environment variables in the local and deployed execution environments + openai_api_key: SecretStr = SecretStr("default_openai_api_key") + huggingface_api_key: SecretStr = SecretStr("default_huggingface_api_key") + + +def get_settings_from_json(json_str: str) -> Settings: + """ + Validates the settings from a json string. + """ + return Settings.model_validate_json(json_str) def generate_env_file(settings: Settings) -> None: """ Generates a .env file from the current settings including defaults. """ - with open(ENV_PATH, "w") as file: - for field, value in asdict(settings).items(): + for field, value in settings.model_dump().items(): if value is not None: + if isinstance(value, SecretStr): + value = value.get_secret_value() file.write(f"{field.upper()}={value}\n") - print(f".env file generated at {str(ENV_PATH)}") # Example usage if __name__ == "__main__": - config: Settings = Settings.build() + config: Settings = Settings() generate_env_file(config) diff --git a/opentrons-ai-server/deploy.py b/opentrons-ai-server/deploy.py index 21190a85e90..8bb01717012 100644 --- a/opentrons-ai-server/deploy.py +++ b/opentrons-ai-server/deploy.py @@ -2,15 +2,17 @@ import base64 import datetime import subprocess -import time from dataclasses import dataclass +from typing import Dict, List import boto3 import docker +from api.settings import Settings, get_settings_from_json +from pydantic import SecretStr from rich import print from rich.prompt import Prompt -ENVIRONMENTS = ["crt", "dev", "sandbox"] +ENVIRONMENTS = ["crt", "dev", "sandbox", "staging", "prod"] def get_aws_account_id() -> str: @@ -26,28 +28,38 @@ def get_aws_region() -> str: @dataclass(frozen=True) class BaseDeploymentConfig: - IMAGE_NAME: str - FUNCTION_NAME: str + IMAGE_NAME: str # local image name ECR_URL: str ECR_REPOSITORY: str - TAG: str = str(int(datetime.datetime.now().timestamp())) - DEPLOYMENT_TIMEOUT_S: int = 60 + CLUSTER_NAME: str + SERVICE_NAME: str + CONTAINER_NAME: str + ENV_VARIABLES_SECRET_NAME: str # key/value secret as the source of truth for environment variables in the deployed environment + TAG: str = "not set" + DEPLOYMENT_TIMEOUT_S: int = 600 + DEPLOYMENT_POLL_INTERVAL_S: int = 20 @dataclass(frozen=True) class CrtDeploymentConfig(BaseDeploymentConfig): ECR_REPOSITORY: str = "crt-ecr-repo" ECR_URL: str = f"{get_aws_account_id()}.dkr.ecr.{get_aws_region()}.amazonaws.com" - FUNCTION_NAME: str = "crt-api-function" IMAGE_NAME: str = "crt-ai-server" + CLUSTER_NAME: str = "crt-ai-cluster" + SERVICE_NAME: str = "crt-ai-service" + CONTAINER_NAME: str = "crt-ai-api" + ENV_VARIABLES_SECRET_NAME: str = "crt-environment-variables" @dataclass(frozen=True) class SandboxDeploymentConfig(BaseDeploymentConfig): ECR_REPOSITORY: str = "sandbox-ecr-repo" ECR_URL: str = f"{get_aws_account_id()}.dkr.ecr.{get_aws_region()}.amazonaws.com" - FUNCTION_NAME: str = "sandbox-api-function" IMAGE_NAME: str = "sandbox-ai-server" + CLUSTER_NAME: str = "sandbox-ai-cluster" + SERVICE_NAME: str = "sandbox-ai-service" + CONTAINER_NAME: str = "sandbox-ai-api" + ENV_VARIABLES_SECRET_NAME: str = "sandbox-environment-variables" @dataclass(frozen=True) @@ -56,15 +68,43 @@ class DevDeploymentConfig(BaseDeploymentConfig): ECR_URL: str = f"{get_aws_account_id()}.dkr.ecr.{get_aws_region()}.amazonaws.com" FUNCTION_NAME: str = "dev-api-function" IMAGE_NAME: str = "dev-ai-server" + CLUSTER_NAME: str = "dev-ai-cluster" + SERVICE_NAME: str = "dev-ai-service" + CONTAINER_NAME: str = "dev-ai-api" + ENV_VARIABLES_SECRET_NAME: str = "dev-environment-variables" + + +@dataclass(frozen=True) +class StagingDeploymentConfig(BaseDeploymentConfig): + ECR_REPOSITORY: str = "staging-ecr-repo" + ECR_URL: str = f"{get_aws_account_id()}.dkr.ecr.{get_aws_region()}.amazonaws.com" + IMAGE_NAME: str = "staging-ai-server" + CLUSTER_NAME: str = "staging-ai-cluster" + SERVICE_NAME: str = "staging-ai-service" + CONTAINER_NAME: str = "staging-ai-api" + ENV_VARIABLES_SECRET_NAME: str = "staging-environment-variables" + + +@dataclass(frozen=True) +class ProdDeploymentConfig(BaseDeploymentConfig): + ECR_REPOSITORY: str = "prod-ecr-repo" + ECR_URL: str = f"{get_aws_account_id()}.dkr.ecr.{get_aws_region()}.amazonaws.com" + IMAGE_NAME: str = "prod-ai-server" + CLUSTER_NAME: str = "prod-ai-cluster" + SERVICE_NAME: str = "prod-ai-service" + CONTAINER_NAME: str = "prod-ai-api" + ENV_VARIABLES_SECRET_NAME: str = "prod-environment-variables" class Deploy: def __init__(self, config: BaseDeploymentConfig) -> None: self.config: BaseDeploymentConfig = config self.ecr_client = boto3.client("ecr") - self.lambda_client = boto3.client("lambda") + self.ecs_client = boto3.client("ecs") + self.secret_manager_client = boto3.client("secretsmanager") self.docker_client = docker.from_env() self.full_image_name = f"{self.config.ECR_URL}/{self.config.ECR_REPOSITORY}:{self.config.TAG}" + self.env_variables: Settings = self.retrieve_environment_variables() def build_docker_image(self) -> None: print(f"Building Docker image {self.config.IMAGE_NAME}:{self.config.TAG}") @@ -91,48 +131,85 @@ def push_docker_image_to_ecr(self) -> None: subprocess.run(["docker", "push", self.full_image_name], check=True) print(f"Image pushed to ECR: {self.full_image_name}") - def update_lambda(self) -> None | str: - """Update a Lambda function using the ECR image.""" + def retrieve_environment_variables(self) -> Settings: + # Retrieve the environment variables from the secret manager. + # They are not secrets, but are stored in the secret manager as a json string. + # This gives us a source of truth for the environment variables. + # We push the json string from secrets manager through the settings model to validate. + secret_value = self.secret_manager_client.get_secret_value(SecretId=self.config.ENV_VARIABLES_SECRET_NAME) + return get_settings_from_json(secret_value.get("SecretString")) - print(f"Updating Lambda function: {self.config.FUNCTION_NAME}") - response = self.lambda_client.update_function_code( - FunctionName=self.config.FUNCTION_NAME, ImageUri=self.full_image_name, Publish=True - ) - print("Updated Lambda function:") - print(response) - version = str(response["Version"]) - print(f"New version: {version}") - return version - - def wait_for_lambda_status(self, version: str) -> None: - """Wait until the Lambda function's version status is no longer 'Pending', or until timeout.""" - timeout = self.config.DEPLOYMENT_TIMEOUT_S - status = "Pending" # Start with 'Pending' as the initial assumed status - start_time = time.time() - - while status == "Pending": - if time.time() - start_time > timeout: - print(f"Timeout reached after {timeout} seconds. Exiting without status change.") - raise TimeoutError("Timeout reached while waiting for Lambda function status to change.") - function_with_version = f"{self.config.FUNCTION_NAME}:{version}" - # Get the function configuration, including the version - response = self.lambda_client.get_function(FunctionName=function_with_version) - # Extract the state of the function version - status = response["Configuration"]["State"] - print(f"Current status of '{function_with_version}': {status}") - - if status != "Pending": - print(f"Status of '{function_with_version}' is now '{status}'. Exiting loop.") + def update_environment_variables(self, environment_variables: List[Dict[str, str]], key: str, value: str) -> List[Dict[str, str]]: + """ + Updates the environment variables list with the given key and value. + Ensures that the list has unique keys. + If the key exists, its value is updated. If it does not exist, a new dictionary is appended. + """ + # Create a dictionary from the list to ensure unique keys + env_vars_dict = {env_var["name"].upper(): env_var["value"] for env_var in environment_variables} + + # Update the dictionary with the new key-value pair + env_vars_dict[key.upper()] = value + + # Convert the dictionary back to a list of dictionaries + updated_environment_variables = [{"name": k, "value": v} for k, v in env_vars_dict.items()] + + return updated_environment_variables + + def update_ecs_task(self) -> None: + print(f"Updating ECS task with new image: {self.full_image_name}") + response = self.ecs_client.describe_services(cluster=self.config.CLUSTER_NAME, services=[self.config.SERVICE_NAME]) + task_definition_arn = response["services"][0]["taskDefinition"] + + task_definition = self.ecs_client.describe_task_definition(taskDefinition=task_definition_arn)["taskDefinition"] + container_definitions = task_definition["containerDefinitions"] + for container_definition in container_definitions: + if container_definition["name"] == self.config.CONTAINER_NAME: + container_definition["image"] = self.full_image_name + environment_variables = container_definition.get("environment", []) + # set the environment variables for the environment + for key, value in self.env_variables.model_dump().items(): + if not isinstance(value, SecretStr): + # Secrets are not set here + # They are set in the secrets key of ECS task definition + environment_variables = self.update_environment_variables(environment_variables, key, value) + # Overwrite the DD_VERSION environment variable + # with the current deployment tag + # this is what we are using for version currently + environment_variables = self.update_environment_variables(environment_variables, "DD_VERSION", self.config.TAG) + container_definition["environment"] = environment_variables + print("Updated container definition:") + print(container_definition) break - else: - sleep_time = 5 - print(f"Status still 'Pending'. Checking again in {sleep_time} seconds...") - time.sleep(sleep_time) + # https://docs.aws.amazon.com/AmazonECS/latest/developerguide/fargate-tasks-services.html#fargate-tasks-size + new_task_definition = { + "family": task_definition["family"], + "containerDefinitions": container_definitions, + "volumes": task_definition["volumes"], + "taskRoleArn": task_definition["taskRoleArn"], + "executionRoleArn": task_definition["executionRoleArn"], + "networkMode": task_definition["networkMode"], + "requiresCompatibilities": task_definition["requiresCompatibilities"], + "cpu": "1024", + "memory": "2048", + } + print("New task definition:") + print(new_task_definition) + register_response = self.ecs_client.register_task_definition(**new_task_definition) + new_task_definition_arn = register_response["taskDefinition"]["taskDefinitionArn"] + + self.ecs_client.update_service( + cluster=self.config.CLUSTER_NAME, + service=self.config.SERVICE_NAME, + taskDefinition=new_task_definition_arn, + forceNewDeployment=True, + ) def main() -> None: - parser = argparse.ArgumentParser(description="Manage Lambda deployment.") + parser = argparse.ArgumentParser(description="Manage ECS Fargate deployment.") parser.add_argument("--env", type=str, help=f"Deployment environment {ENVIRONMENTS}") + parser.add_argument("--tag", type=str, help="The tag and therefore version of the container to use") args = parser.parse_args() # Determine if the script was called with command-line arguments if args.env: @@ -140,27 +217,44 @@ def main() -> None: print(f"[red]Invalid environment specified: {args.env}[/red]") exit(1) env = args.env.lower() + if args.tag: + tag = args.tag + else: + if args.env: + # Passing --env alone generates a tag and does not prompt! + tag = str(int(datetime.datetime.now().timestamp())) else: - # Interactive prompts if no command-line arguments + # Interactive prompts if env not set env = Prompt.ask("[bold magenta]Enter the deployment environment[/]", choices=ENVIRONMENTS, default="crt") + tag = Prompt.ask( + "[bold magenta]Enter tag for the container that also becomes the version.[/]", + default=str(int(datetime.datetime.now().timestamp())), + ) # Validate environment config: BaseDeploymentConfig - if env == "crt": - config = CrtDeploymentConfig() + if env == "prod": + config = ProdDeploymentConfig(TAG=tag) + elif env == "staging": + config = StagingDeploymentConfig(TAG=tag) + elif env == "crt": + config = CrtDeploymentConfig(TAG=tag) elif env == "dev": - config = DevDeploymentConfig() + config = DevDeploymentConfig(TAG=tag) elif env == "sandbox": - config = SandboxDeploymentConfig() + config = SandboxDeploymentConfig(TAG=tag) else: print(f"[red]Invalid environment specified: {env}[/red]") exit(1) aws = Deploy(config) aws.build_docker_image() aws.push_docker_image_to_ecr() - version = aws.update_lambda() - if version: - aws.wait_for_lambda_status(version) + aws.update_ecs_task() + print(f"Deployment to {env} started.") + print(f"A new image was built and pushed to ECR with tag: {tag}") + print("A new Task definition was defined and registered.") + print("Then we told the ECS service to deploy the new definition.") + print("Monitor the deployment in the ECS console.") if __name__ == "__main__": diff --git a/opentrons-ai-server/pytest.ini b/opentrons-ai-server/pytest.ini index bca574b8aff..7f5d39f9a86 100644 --- a/opentrons-ai-server/pytest.ini +++ b/opentrons-ai-server/pytest.ini @@ -1,5 +1,5 @@ [pytest] -addopts = -s -vv --log-cli-level info +addopts = -vv markers = unit: marks tests as unit tests (select with '-m unit') live: marks tests as live tests (select with '-m live') diff --git a/opentrons-ai-server/tests/conftest.py b/opentrons-ai-server/tests/conftest.py index e9f2ac342fc..aa274fda6fe 100644 --- a/opentrons-ai-server/tests/conftest.py +++ b/opentrons-ai-server/tests/conftest.py @@ -9,7 +9,7 @@ def pytest_addoption(parser: pytest.Parser) -> None: """Add an option to pytest command-line parser to specify the environment.""" parser.addoption( - "--env", action="store", default="sandbox", help="Set the environment for the tests (dev, sandbox, crt, staging, prod)" + "--env", action="store", default="local", help="Set the environment for the tests (local, dev, sandbox, crt, staging, prod)" ) diff --git a/opentrons-ai-server/tests/helpers/client.py b/opentrons-ai-server/tests/helpers/client.py index 3b112fb4b40..32944f9f426 100644 --- a/opentrons-ai-server/tests/helpers/client.py +++ b/opentrons-ai-server/tests/helpers/client.py @@ -1,14 +1,38 @@ +import time +from functools import wraps +from typing import Any, Callable, TypeVar + from api.models.chat_request import ChatRequest from httpx import Client as HttpxClient from httpx import Response, Timeout -from rich import inspect -from rich.console import Console +from rich.console import Console, Group from rich.panel import Panel +from rich.pretty import Pretty from rich.prompt import Prompt +from rich.rule import Rule +from rich.text import Text from tests.helpers.settings import Settings, get_settings from tests.helpers.token import Token +F = TypeVar("F", bound=Callable[..., Any]) + + +def timeit(func: F) -> F: + @wraps(func) + def wrapper(*args: Any, **kwargs: Any) -> Any: + start_time = time.time() + result = func(*args, **kwargs) + end_time = time.time() + elapsed_time = end_time - start_time + console.print(f"[bold green]{func.__name__} completed in {elapsed_time:.4f} seconds[/bold green]") + return result + + return wrapper # type: ignore + + +console = Console() + class Client: def __init__(self, settings: Settings): @@ -21,7 +45,7 @@ def __init__(self, settings: Settings): **self.type_headers, **self.auth_headers, } - self.timeout = Timeout(connect=5.0, read=120.0, write=120.0, pool=5.0) + self.timeout = Timeout(connect=5.0, read=180.0, write=180.0, pool=5.0) self.httpx = HttpxClient(base_url=self.settings.BASE_URL, timeout=self.timeout) def close(self) -> None: @@ -37,6 +61,7 @@ def get_health(self) -> Response: """Call the /health endpoint and return the response.""" return self.httpx.get("/health", headers=self.type_headers) + @timeit def get_chat_completion(self, message: str, fake: bool = True, bad_auth: bool = False) -> Response: """Call the /chat/completion endpoint and return the response.""" request = ChatRequest(message=message, fake=fake) @@ -58,33 +83,44 @@ def get_options(self) -> Response: def print_response(response: Response) -> None: """Prints the HTTP response using rich.""" - console = Console() - console.print(Panel("Response", expand=False)) - inspect(response) + status_code_text = Text(f"Status code: {response.status_code}", style="bold green") + try: + json = response.json() + except Exception: + json = None + if json: + text = Pretty(json) + else: + text = Pretty(response.text) + url = Pretty(response.request.url) + # Group the text elements + panel_content = Group(url, status_code_text, text) + # Print the panel with grouped content + console.print(Panel(panel_content, title="Response", expand=False)) def main() -> None: - console = Console() - env = Prompt.ask("Select environment", choices=["dev", "sandbox", "crt"], default="sandbox") + env = Prompt.ask("Select environment", choices=["local", "dev", "sandbox", "crt", "staging"], default="local") settings = get_settings(env=env) client = Client(settings) try: - console.print(Panel("Getting health endpoint", expand=False)) + console.print(Rule("Getting health endpoint", style="bold")) response = client.get_health() print_response(response) - console.print(Panel("Getting chat completion with fake=True and good auth (won't call OpenAI)", expand=False)) + console.print(Rule("Getting chat completion with fake=True and good auth (won't call OpenAI)", style="bold")) response = client.get_chat_completion("How do I load a pipette?") print_response(response) - console.print(Panel("Getting chat completion with fake=True and bad auth to show 401 error (won't call OpenAI)", expand=False)) + console.print(Rule("Getting chat completion with fake=True and bad auth to show 401 error (won't call OpenAI)", style="bold")) response = client.get_chat_completion("How do I load a pipette?", bad_auth=True) print_response(response) - console.print(Panel("Getting OPTIONS", expand=False)) + console.print(Rule("Getting OPTIONS", style="bold")) response = client.get_options() print_response(response) + console.print(Rule("Now interact", style="bold")) real = Prompt.ask("Actually call OpenAI API?", choices=["y", "n"], default="n") if real == "y": message = Prompt.ask("Enter a message") diff --git a/opentrons-ai-server/tests/helpers/settings.py b/opentrons-ai-server/tests/helpers/settings.py index 493606b2cba..188434fd84c 100644 --- a/opentrons-ai-server/tests/helpers/settings.py +++ b/opentrons-ai-server/tests/helpers/settings.py @@ -34,6 +34,23 @@ def _get_required_env(self, var_name: str) -> str: raise EnvironmentError(f"Required environment variable '{var_name}' is not set.") from err +class LocalSettings(Settings): + ENV_VARIABLE_MAP = { + "TOKEN_URL": "LOCAL_TOKEN_URL", + "BASE_URL": "LOCAL_BASE_URL", + "CLIENT_ID": "LOCAL_CLIENT_ID", + "SECRET": "LOCAL_SECRET", + "AUDIENCE": "LOCAL_AUDIENCE", + "GRANT_TYPE": "LOCAL_GRANT_TYPE", + "CACHED_TOKEN_PATH": str(Path(Path(__file__).parent, "cached_token.txt")), + } + + def __init__(self) -> None: + super().__init__() + load_dotenv(self.ENV_PATH) + self._set_properties() + + class DevSettings(Settings): ENV_VARIABLE_MAP = { "TOKEN_URL": "DEV_TOKEN_URL", @@ -85,20 +102,53 @@ def __init__(self) -> None: self._set_properties() -# TODO:y3rsh:2024-05-11: Add staging and prod +class StagingSettings(Settings): + ENV_VARIABLE_MAP = { + "TOKEN_URL": "STAGING_TOKEN_URL", + "BASE_URL": "STAGING_BASE_URL", + "CLIENT_ID": "STAGING_CLIENT_ID", + "SECRET": "STAGING_SECRET", + "AUDIENCE": "STAGING_AUDIENCE", + "GRANT_TYPE": "STAGING_GRANT_TYPE", + "CACHED_TOKEN_PATH": str(Path(Path(__file__).parent, "staging_cached_token.txt")), + } + + def __init__(self) -> None: + super().__init__() + load_dotenv(self.ENV_PATH) + self._set_properties() + + +class ProdSettings(Settings): + ENV_VARIABLE_MAP = { + "TOKEN_URL": "PROD_TOKEN_URL", + "BASE_URL": "PROD_BASE_URL", + "CLIENT_ID": "PROD_CLIENT_ID", + "SECRET": "PROD_SECRET", + "AUDIENCE": "PROD_AUDIENCE", + "GRANT_TYPE": "PROD_GRANT_TYPE", + "CACHED_TOKEN_PATH": str(Path(Path(__file__).parent, "prod_cached_token.txt")), + } + + def __init__(self) -> None: + super().__init__() + load_dotenv(self.ENV_PATH) + self._set_properties() def get_settings(env: str) -> Settings: - if env.lower() == "dev": + if env.lower() == "local": + return LocalSettings() + elif env.lower() == "dev": return DevSettings() elif env.lower() == "sandbox": return SandboxSettings() elif env.lower() == "crt": return CrtSettings() elif env.lower() == "staging": - raise NotImplementedError("Staging environment not implemented.") + return StagingSettings() elif env.lower() == "prod": - raise NotImplementedError("Production environment not implemented.") + return ProdSettings() else: raise ValueError(f"Unsupported environment: {env}") diff --git a/opentrons-ai-server/tests/helpers/token_verifier.py b/opentrons-ai-server/tests/helpers/token_verifier.py index c1b4e4aac54..0c73e249fa1 100644 --- a/opentrons-ai-server/tests/helpers/token_verifier.py +++ b/opentrons-ai-server/tests/helpers/token_verifier.py @@ -8,7 +8,7 @@ from cryptography.hazmat.primitives import serialization from cryptography.hazmat.primitives.asymmetric.rsa import RSAPublicNumbers from jwt.exceptions import DecodeError, ExpiredSignatureError, InvalidTokenError -from rich import inspect, print +from rich import print from tests.helpers.settings import Settings @@ -68,7 +68,7 @@ def is_valid_token(self, token: str) -> bool: options={"verify_signature": True}, ) print("Decoded token:") - inspect(decoded_token) + print(decoded_token) return True except (DecodeError, ExpiredSignatureError, InvalidTokenError) as e: print(f"JWT validation error: {str(e)}") diff --git a/opentrons-ai-server/tests/test_chat_models.py b/opentrons-ai-server/tests/test_chat_models.py index 4c5141cf13e..aa18f113a49 100644 --- a/opentrons-ai-server/tests/test_chat_models.py +++ b/opentrons-ai-server/tests/test_chat_models.py @@ -7,7 +7,7 @@ @pytest.mark.unit def test_chat_request_model() -> None: # Test valid data - request_data = {"message": "Hello", "fake": False} + request_data = {"message": "Hello", "chat_history": [], "fake": False} request = ChatRequest(**request_data) assert request.message == "Hello" assert request.fake is False @@ -28,5 +28,5 @@ def test_chat_response_model() -> None: # Test invalid data with pytest.raises(ValidationError): - invalid_response_data = {"reply": 123, "fake": "false"} + invalid_response_data = {"reply": 123, "history": "history", "fake": "false"} ChatResponse(**invalid_response_data) diff --git a/opentrons-ai-server/tests/test_live.py b/opentrons-ai-server/tests/test_live.py index 51a0859cec3..e03c01add13 100644 --- a/opentrons-ai-server/tests/test_live.py +++ b/opentrons-ai-server/tests/test_live.py @@ -1,4 +1,5 @@ import pytest +from api.models.chat_response import ChatResponse from tests.helpers.client import Client @@ -15,12 +16,12 @@ def test_get_chat_completion_good_auth(client: Client) -> None: """Test the chat completion endpoint with good authentication.""" response = client.get_chat_completion("How do I load tipracks for my 8 channel pipette on an OT2?", fake=True) assert response.status_code == 200, "Chat completion with good auth should return HTTP 200" + ChatResponse.model_validate(response.json()) @pytest.mark.live def test_get_chat_completion_bad_auth(client: Client) -> None: """Test the chat completion endpoint with bad authentication.""" - # This call never reaches the lambda function, the API Gateway rejects it response = client.get_chat_completion("How do I load a pipette?", bad_auth=True) assert response.status_code == 401, "Chat completion with bad auth should return HTTP 401" @@ -36,7 +37,7 @@ def test_get_bad_endpoint_with_good_auth(client: Client) -> None: def test_get_bad_endpoint_with_bad_auth(client: Client) -> None: """Test a nonexistent endpoint with bad authentication.""" response = client.get_bad_endpoint(bad_auth=True) - assert response.status_code == 401, "nonexistent endpoint with bad auth should return HTTP 401" + assert response.status_code == 404, "nonexistent endpoint with bad auth should return HTTP 404" @pytest.mark.live @@ -44,12 +45,15 @@ def test_get_options(client: Client) -> None: """Test the OPTIONS endpoint.""" response = client.get_options() assert response.status_code == 200, "OPTIONS endpoint should return HTTP 200" + # This is the shape that makes pre-flight from the client happy. expected_headers = { "Access-Control-Allow-Origin": "*", "Access-Control-Allow-Methods": "GET,POST,OPTIONS", "Access-Control-Allow-Headers": "content-type,authorization,origin,accept", "Access-Control-Expose-Headers": "content-type", - "Access-Control-Max-Age": "3600", + "Access-Control-Max-Age": "600", } + body = response.json() + print(response) for header, expected_value in expected_headers.items(): - assert response.json().get(header) == expected_value, f"{header} should be {expected_value}" + assert body.get(header) == expected_value, f"{header} should be {expected_value}"