From e4896d598a259298b206fc616c018fa88dc26193 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Wed, 4 Feb 2026 18:28:21 +0530 Subject: [PATCH 01/13] agent framework v2 changes --- src/backend/pyproject.toml | 26 +- src/backend/uv.lock | 1026 ++--------------- src/backend/v4/callbacks/response_handlers.py | 30 +- .../v4/magentic_agents/common/lifecycle.py | 53 +- .../v4/magentic_agents/foundry_agent.py | 14 +- src/backend/v4/magentic_agents/proxy_agent.py | 26 +- .../orchestration/human_approval_manager.py | 6 +- .../v4/orchestration/orchestration_manager.py | 154 ++- 8 files changed, 276 insertions(+), 1059 deletions(-) diff --git a/src/backend/pyproject.toml b/src/backend/pyproject.toml index d176467f5..7750638dd 100644 --- a/src/backend/pyproject.toml +++ b/src/backend/pyproject.toml @@ -7,21 +7,20 @@ requires-python = ">=3.11" dependencies = [ "azure-ai-evaluation==1.11.0", "azure-ai-inference==1.0.0b9", - "azure-ai-projects==1.0.0", - "azure-ai-agents==1.2.0b5", + "azure-ai-projects==2.0.0b3", "azure-cosmos==4.9.0", "azure-identity==1.24.0", "azure-monitor-events-extension==0.1.0", - "azure-monitor-opentelemetry==1.7.0", + "azure-monitor-opentelemetry>=1.8.0", "azure-search-documents==11.5.3", "fastapi==0.116.1", - "openai==1.105.0", - "opentelemetry-api==1.36.0", - "opentelemetry-exporter-otlp-proto-grpc==1.36.0", - "opentelemetry-exporter-otlp-proto-http==1.36.0", - "opentelemetry-instrumentation-fastapi==0.57b0", - "opentelemetry-instrumentation-openai==0.46.2", - "opentelemetry-sdk==1.36.0", + "openai>=2.8.0", + "opentelemetry-api>=1.39.0", + "opentelemetry-exporter-otlp-proto-grpc>=1.39.0", + "opentelemetry-exporter-otlp-proto-http>=1.39.0", + "opentelemetry-instrumentation-fastapi>=0.57b0", + "opentelemetry-instrumentation-openai>=0.46.2", + "opentelemetry-sdk>=1.39.0", "pytest==8.4.1", "pytest-asyncio==0.24.0", "pytest-cov==5.0.0", @@ -31,6 +30,7 @@ dependencies = [ "uvicorn==0.35.0", "pylint-pydantic==0.3.5", "pexpect==4.9.0", - "mcp==1.13.1", - "agent-framework>=1.0.0b251105", -] + "mcp>=1.24.0,<2", + "agent-framework-azure-ai==1.0.0b260130", + "agent-framework-core==1.0.0b260130" +] \ No newline at end of file diff --git a/src/backend/uv.lock b/src/backend/uv.lock index 73a5e2e5c..526fe789a 100644 --- a/src/backend/uv.lock +++ b/src/backend/uv.lock @@ -1,5 +1,5 @@ version = 1 -revision = 2 +revision = 3 requires-python = ">=3.11" resolution-markers = [ "python_full_version >= '3.13'", @@ -7,101 +7,9 @@ resolution-markers = [ "python_full_version < '3.12'", ] -[[package]] -name = "a2a-sdk" -version = "0.3.11" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "google-api-core" }, - { name = "httpx" }, - { name = "httpx-sse" }, - { name = "protobuf" }, - { name = "pydantic" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/11/2c/6eff205080a4fb3937745f0bab4ff58716cdcc524acd077a493612d34336/a2a_sdk-0.3.11.tar.gz", hash = "sha256:194a6184d3e5c1c5d8941eb64fb33c346df3ebbec754effed8403f253bedb085", size = 226923, upload-time = "2025-11-07T11:05:38.496Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/00/f9/3e633485a3f23f5b3e04a7f0d3e690ae918fd1252941e8107c7593d882f1/a2a_sdk-0.3.11-py3-none-any.whl", hash = "sha256:f57673d5f38b3e0eb7c5b57e7dc126404d02c54c90692395ab4fd06aaa80cc8f", size = 140381, upload-time = "2025-11-07T11:05:37.093Z" }, -] - -[[package]] -name = "ag-ui-protocol" -version = "0.1.10" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pydantic" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/67/bb/5a5ec893eea5805fb9a3db76a9888c3429710dfb6f24bbb37568f2cf7320/ag_ui_protocol-0.1.10.tar.gz", hash = "sha256:3213991c6b2eb24bb1a8c362ee270c16705a07a4c5962267a083d0959ed894f4", size = 6945, upload-time = "2025-11-06T15:17:17.068Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/8f/78/eb55fabaab41abc53f52c0918a9a8c0f747807e5306273f51120fd695957/ag_ui_protocol-0.1.10-py3-none-any.whl", hash = "sha256:c81e6981f30aabdf97a7ee312bfd4df0cd38e718d9fc10019c7d438128b93ab5", size = 7889, upload-time = "2025-11-06T15:17:15.325Z" }, -] - -[[package]] -name = "agent-framework" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-a2a" }, - { name = "agent-framework-ag-ui" }, - { name = "agent-framework-anthropic" }, - { name = "agent-framework-azure-ai" }, - { name = "agent-framework-chatkit" }, - { name = "agent-framework-copilotstudio" }, - { name = "agent-framework-core" }, - { name = "agent-framework-devui" }, - { name = "agent-framework-lab" }, - { name = "agent-framework-mem0" }, - { name = "agent-framework-purview" }, - { name = "agent-framework-redis" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/36/6a/8e467a13b06471f300236d3caa29370be355cd9cbc6169f2bc93e780d24e/agent_framework-1.0.0b251108.tar.gz", hash = "sha256:456c5aa6b03ad0c3545eca3f0460d94eb51eb2f7a3827530ac7cb6203ff2adc8", size = 2408664, upload-time = "2025-11-08T18:17:30.388Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c9/35/4d384facf5a8af7a3a0830ab38cbbfa0140c1abdb494a4ec4ed4dc1b3092/agent_framework-1.0.0b251108-py3-none-any.whl", hash = "sha256:faaacbb7af156084847df39a7a7e4151198fa4f00271c742672e202466d796cf", size = 5613, upload-time = "2025-11-08T18:17:28.547Z" }, -] - -[[package]] -name = "agent-framework-a2a" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "a2a-sdk" }, - { name = "agent-framework-core" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/e7/3c/ede80d6f004888c6b1d2f014215a78e9e9325ec6789c73ceb32e63e196f9/agent_framework_a2a-1.0.0b251108.tar.gz", hash = "sha256:4799cbf6be6314e4c8c1e1b6b4ab58dad771af3af555afb508ee1b485ae92896", size = 11023, upload-time = "2025-11-08T18:17:32.634Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/a3/e2/4a8e7cef6cb32a752543f2c4cde39ece089ce758bd12659b6404bef88732/agent_framework_a2a-1.0.0b251108-py3-none-any.whl", hash = "sha256:0804719a7341a9f5caa90a3e83b8fd907e80166bfa52a2a82ab061ab58362a1b", size = 7035, upload-time = "2025-11-08T18:17:31.459Z" }, -] - -[[package]] -name = "agent-framework-ag-ui" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "ag-ui-protocol" }, - { name = "agent-framework-core" }, - { name = "fastapi" }, - { name = "uvicorn" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/63/3e/f32ec6e059d3878cfd045a5e31a502f9016d3ec3b7a7633911d92640f134/agent_framework_ag_ui-1.0.0b251108.tar.gz", hash = "sha256:ff0b3471ce7c56a908dfed42e0484cbba034a471e9ae187f8d852a6677e34734", size = 57026, upload-time = "2025-11-08T18:17:34.422Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/42/d4/b2f31a8196b11d7af95ec7f33674877055e9868399caddf29afe184772f0/agent_framework_ag_ui-1.0.0b251108-py3-none-any.whl", hash = "sha256:974435f1c22d914f2e032603e7a50d5d8a02f960742dcd4cdaf1a69f0e08a3b3", size = 23387, upload-time = "2025-11-08T18:17:33.234Z" }, -] - -[[package]] -name = "agent-framework-anthropic" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "anthropic" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/d8/0e/2db7dc5be7aeab0cdafc00d90c2cb85eae17e7c0607ac312a02b42b424eb/agent_framework_anthropic-1.0.0b251108.tar.gz", hash = "sha256:b7b46bd735627587c58e429cc8f15cd7175c1aebb4a6b02128a2423fd07a7948", size = 13464, upload-time = "2025-11-08T18:17:36.18Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/a1/9f/b852b795aa0a735577fb3823fca98027a260c68bd7d1f8e3820ff467b286/agent_framework_anthropic-1.0.0b251108-py3-none-any.whl", hash = "sha256:d7ce9d10338fea0ddfb7a20aa9b977f270ae63a5565287ecd89c3b0cc10d1c41", size = 8719, upload-time = "2025-11-08T18:17:35.083Z" }, -] - [[package]] name = "agent-framework-azure-ai" -version = "1.0.0b251108" +version = "1.0.0b260130" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "agent-framework-core" }, @@ -109,43 +17,20 @@ dependencies = [ { name = "azure-ai-agents" }, { name = "azure-ai-projects" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/d8/18/29016d35185e51ad29b31f2089d66d0dbae11ecf02fe7707eab798dc8a48/agent_framework_azure_ai-1.0.0b251108.tar.gz", hash = "sha256:75fd77959f8e770338dacd41e6fc6698151a3c85abb5941e97d6581d8a7fd9e6", size = 25686, upload-time = "2025-11-08T18:17:38.174Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/48/db/725da2ad1467b54edc5d31ac64a3a46e0277cf9b6b7508aef1a2dc9b7360/agent_framework_azure_ai-1.0.0b251108-py3-none-any.whl", hash = "sha256:5957e90eb0ce3d4fde2d54cde53d01a44f4a3b242faedaca304a359a34d18f7b", size = 13655, upload-time = "2025-11-08T18:17:37.061Z" }, -] - -[[package]] -name = "agent-framework-chatkit" -version = "0.0.1a0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/d7/0c/e7a14bb04393e65d04016ebf8dd61d4560f794a124df28923ee5afebbaa4/agent_framework_chatkit-0.0.1a0.tar.gz", hash = "sha256:7687daaab3f48be7f72dcd08cf60383afa488e42bc6ecb3825d1978bb72da28a", size = 1862, upload-time = "2025-10-07T18:31:22.111Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/36/ab/b7c1ca3ee7d88688d1cb7ff66957d319aa9056291a34eb39ebb1206d9985/agent_framework_chatkit-0.0.1a0-py3-none-any.whl", hash = "sha256:a9ab2dd40aa0e243119eec37f78f5d429bc3f08b835eb66725c2440360ff31de", size = 2240, upload-time = "2025-10-07T18:31:20.834Z" }, -] - -[[package]] -name = "agent-framework-copilotstudio" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "microsoft-agents-copilotstudio-client" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/e2/d5/ffca3932aea23b79ebcdb51a9537971dc0344a36e0ae0c1605f87d51fc3e/agent_framework_copilotstudio-1.0.0b251108.tar.gz", hash = "sha256:464d3d36a9138372f463efc9e7dce24094162c5211eab45644bedaceb19c486a", size = 11985, upload-time = "2025-11-08T18:17:41.223Z" } +sdist = { url = "https://files.pythonhosted.org/packages/ac/ef/69ead4fcd2c21608ce35353a507df23df51872552747f803c43d1d81f612/agent_framework_azure_ai-1.0.0b260130.tar.gz", hash = "sha256:c571275089a801f961370ba824568c8b02143b1a6bb5b1d78b97c6debdf4906f", size = 32723, upload-time = "2026-01-30T18:56:41.07Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/77/37/5d83bad31a2f15db1864eb6b10d18d5647c511f8a6214ce56423718b60b9/agent_framework_copilotstudio-1.0.0b251108-py3-none-any.whl", hash = "sha256:8f799a9b6fef126893ebbbefa5fc63676498093ccf415e025f30ca3bf83b2593", size = 8710, upload-time = "2025-11-08T18:17:40.433Z" }, + { url = "https://files.pythonhosted.org/packages/72/8f/a1467c352fed5eb6ebb9567109251cc39b5b3ebb5137a2d14c71fea51bc8/agent_framework_azure_ai-1.0.0b260130-py3-none-any.whl", hash = "sha256:87f0248fe6d4f2f4146f0a56a53527af6365d4a377dc2e3d56c37cbb9deae098", size = 38542, upload-time = "2026-01-30T19:01:12.102Z" }, ] [[package]] name = "agent-framework-core" -version = "1.0.0b251108" +version = "1.0.0b260130" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "azure-identity" }, { name = "mcp", extra = ["ws"] }, { name = "openai" }, { name = "opentelemetry-api" }, - { name = "opentelemetry-exporter-otlp-proto-grpc" }, { name = "opentelemetry-sdk" }, { name = "opentelemetry-semantic-conventions-ai" }, { name = "packaging" }, @@ -153,78 +38,9 @@ dependencies = [ { name = "pydantic-settings" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/8b/75/bb7fad403146236ca70a6c5ebac500763381c457b1834cadd1eb0c864c9a/agent_framework_core-1.0.0b251108.tar.gz", hash = "sha256:4d7b0b301e46abdcce469d015194d9359dd10ae15ebe98014064ce00a08b5c2a", size = 463832, upload-time = "2025-11-08T18:17:43.486Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/bb/65/64367c292402cf95ef9a91cd5007734d141b3b96eb8d3146f896efccfc72/agent_framework_core-1.0.0b251108-py3-none-any.whl", hash = "sha256:04392835292ab66c19f873ea3bd78c612ca3bc206792a8c272be250f53cff42b", size = 318092, upload-time = "2025-11-08T18:17:41.949Z" }, -] - -[[package]] -name = "agent-framework-devui" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "fastapi" }, - { name = "python-dotenv" }, - { name = "uvicorn", extra = ["standard"] }, -] -sdist = { url = "https://files.pythonhosted.org/packages/dc/b1/9e66dfcdd4405bc85aa2ab402b8b735460e182160579e23284c3e7308c01/agent_framework_devui-1.0.0b251108.tar.gz", hash = "sha256:b064165b499a8ff23ebd3956970251295a1a22857fbc5a918d7988a0fd428c89", size = 759911, upload-time = "2025-11-08T18:17:46.196Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/5c/47/40601d0e349fbfbf80bedddcc6c556e53fee5555bd7b587ced9f20a004f0/agent_framework_devui-1.0.0b251108-py3-none-any.whl", hash = "sha256:d5564d969bab4dfaf96ad161abbdf456aed1efd7d8ff0953048724cf237c7e8f", size = 337466, upload-time = "2025-11-08T18:17:44.628Z" }, -] - -[[package]] -name = "agent-framework-lab" -version = "1.0.0b251024" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/05/c5/be86273cb3545651d0c8112ff9f38ae8fe13b740ce9b65b9be83ff2d70ee/agent_framework_lab-1.0.0b251024.tar.gz", hash = "sha256:4261cb595b6edfd4f30db613c1885c71b3dcfa2088cf29224d4f17b3ff956b2a", size = 23397, upload-time = "2025-10-24T18:13:48.58Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/26/0f/3974b2b1f6bf523ee3ced0886b6afd5ca8bbebd24aa5278ef77db0d3d765/agent_framework_lab-1.0.0b251024-py3-none-any.whl", hash = "sha256:1596408991a92fcacef4bb939305d2b59159517b707f48114105fc0dd46bfee7", size = 26589, upload-time = "2025-10-24T18:13:47.229Z" }, -] - -[[package]] -name = "agent-framework-mem0" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "mem0ai" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/db/4f/a56c83fbadb42be5edada9fcbb00f460dbb9ddf4b38c4956e16a3ed45b00/agent_framework_mem0-1.0.0b251108.tar.gz", hash = "sha256:742206230ffc780410c145e26f2cd6ccf9e1ac1b616db9ca3327b6be73d81ccc", size = 8045, upload-time = "2025-11-08T18:17:47.8Z" } +sdist = { url = "https://files.pythonhosted.org/packages/4d/39/e508e778219bd6d20e023a6f48235861a639e3cf888776f9e873bbad3c6b/agent_framework_core-1.0.0b260130.tar.gz", hash = "sha256:030a5b2ced796eec6839c2dabad90b4bd1ea33d1026f3ed1813050a56ccfa4ec", size = 301823, upload-time = "2026-01-30T19:01:09.629Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/df/57/5203bc38cf2e7913c6ea10cfb9d6723884e071e4bc7e18e35d6d94c43c6a/agent_framework_mem0-1.0.0b251108-py3-none-any.whl", hash = "sha256:68a7277cc174886288d0cc98fc5459bc57a61332468aa516501d6a217150b023", size = 5302, upload-time = "2025-11-08T18:17:47.033Z" }, -] - -[[package]] -name = "agent-framework-purview" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "azure-core" }, - { name = "httpx" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/11/a8/fc24a829db5d3267633e7c8f4d0b9b03b7a15d4eca9e7e8402c59b5e7f22/agent_framework_purview-1.0.0b251108.tar.gz", hash = "sha256:42620e76614d52e7fd43cba4207a2d29844fd968338e3bf66da98a848eb51b2d", size = 39712, upload-time = "2025-11-08T18:17:49.849Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/01/48/31e0746054e7d1b257f096b111d02e330771647d3583c2011500e2c716e7/agent_framework_purview-1.0.0b251108-py3-none-any.whl", hash = "sha256:bc37ca695a3243f614a522a4d6428604919c99c62e029940cd47a10be725cbfb", size = 26271, upload-time = "2025-11-08T18:17:48.667Z" }, -] - -[[package]] -name = "agent-framework-redis" -version = "1.0.0b251108" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "agent-framework-core" }, - { name = "numpy" }, - { name = "redis" }, - { name = "redisvl" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/86/51/2234704e16833cd8fda9b8d810fea8b0ecf8e66f84f7b41ae42b063af788/agent_framework_redis-1.0.0b251108.tar.gz", hash = "sha256:e6a26b23982a888580e7a92011a3882ec76d523d49c9917ea16df950db70950e", size = 22726, upload-time = "2025-11-08T18:17:51.346Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/4e/42/0eb6a4051b10654de1fbe63e15784e0682427b0227132faca661e1821cb0/agent_framework_redis-1.0.0b251108-py3-none-any.whl", hash = "sha256:a3186b7ea987aa072b9f1ef4663ed3012ed575bb276e201a41db967de2c58c1a", size = 15563, upload-time = "2025-11-08T18:17:50.49Z" }, + { url = "https://files.pythonhosted.org/packages/36/68/afe66c72951a279e0fe048fd5af1e775528cde40dbdab8ec03b42c545df4/agent_framework_core-1.0.0b260130-py3-none-any.whl", hash = "sha256:75b4dd0ca2ae52574d406cf5c9ed7adf63e187379f72fce891743254d83dfd56", size = 348724, upload-time = "2026-01-30T18:56:47.15Z" }, ] [[package]] @@ -391,25 +207,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/78/b6/6307fbef88d9b5ee7421e68d78a9f162e0da4900bc5f5793f6d3d0e34fb8/annotated_types-0.7.0-py3-none-any.whl", hash = "sha256:1f02e8b43a8fbbc3f3e0d4f0f4bfc8131bcb4eebe8849b8e5c773f3a1c582a53", size = 13643, upload-time = "2024-05-20T21:33:24.1Z" }, ] -[[package]] -name = "anthropic" -version = "0.72.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "anyio" }, - { name = "distro" }, - { name = "docstring-parser" }, - { name = "httpx" }, - { name = "jiter" }, - { name = "pydantic" }, - { name = "sniffio" }, - { name = "typing-extensions" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/49/07/61f3ca8e69c5dcdaec31b36b79a53ea21c5b4ca5e93c7df58c71f43bf8d8/anthropic-0.72.0.tar.gz", hash = "sha256:8971fe76dcffc644f74ac3883069beb1527641115ae0d6eb8fa21c1ce4082f7a", size = 493721, upload-time = "2025-10-28T19:13:01.755Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/7b/b7/160d4fb30080395b4143f1d1a4f6c646ba9105561108d2a434b606c03579/anthropic-0.72.0-py3-none-any.whl", hash = "sha256:0e9f5a7582f038cab8efbb4c959e49ef654a56bfc7ba2da51b5a7b8a84de2e4d", size = 357464, upload-time = "2025-10-28T19:13:00.215Z" }, -] - [[package]] name = "anyio" version = "4.11.0" @@ -442,15 +239,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/af/0f/3b8fdc946b4d9cc8cc1e8af42c4e409468c84441b933d037e101b3d72d86/astroid-3.3.11-py3-none-any.whl", hash = "sha256:54c760ae8322ece1abd213057c4b5bba7c49818853fc901ef09719a60dbf9dec", size = 275612, upload-time = "2025-07-13T18:04:21.07Z" }, ] -[[package]] -name = "async-timeout" -version = "5.0.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/a5/ae/136395dfbfe00dfc94da3f3e136d0b13f394cba8f4841120e34226265780/async_timeout-5.0.1.tar.gz", hash = "sha256:d9321a7a3d5a6a5e187e824d2fa0793ce379a202935782d555d6e9d2735677d3", size = 9274, upload-time = "2024-11-06T16:41:39.6Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/fe/ba/e2081de779ca30d473f21f5b30e0e737c438205440784c7dfc81efc2b029/async_timeout-5.0.1-py3-none-any.whl", hash = "sha256:39e3809566ff85354557ec2398b55e096c8364bacac9405a7a1fa429e77fe76c", size = 6233, upload-time = "2024-11-06T16:41:37.9Z" }, -] - [[package]] name = "attrs" version = "25.4.0" @@ -563,18 +351,18 @@ wheels = [ [[package]] name = "azure-ai-projects" -version = "1.0.0" +version = "2.0.0b3" source = { registry = "https://pypi.org/simple" } dependencies = [ - { name = "azure-ai-agents" }, { name = "azure-core" }, + { name = "azure-identity" }, { name = "azure-storage-blob" }, { name = "isodate" }, - { name = "typing-extensions" }, + { name = "openai" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/dd/95/9c04cb5f658c7f856026aa18432e0f0fa254ead2983a3574a0f5558a7234/azure_ai_projects-1.0.0.tar.gz", hash = "sha256:b5f03024ccf0fd543fbe0f5abcc74e45b15eccc1c71ab87fc71c63061d9fd63c", size = 130798, upload-time = "2025-07-31T02:09:27.912Z" } +sdist = { url = "https://files.pythonhosted.org/packages/24/e0/3512d3f07e9dd2eb4af684387c31598c435bd87833b6a81850972963cb9c/azure_ai_projects-2.0.0b3.tar.gz", hash = "sha256:6d09ad110086e450a47b991ee8a3644f1be97fa3085d5981d543f900d78f4505", size = 431749, upload-time = "2026-01-06T05:31:25.849Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b5/db/7149cdf71e12d9737f186656176efc94943ead4f205671768c1549593efe/azure_ai_projects-1.0.0-py3-none-any.whl", hash = "sha256:81369ed7a2f84a65864f57d3fa153e16c30f411a1504d334e184fb070165a3fa", size = 115188, upload-time = "2025-07-31T02:09:29.362Z" }, + { url = "https://files.pythonhosted.org/packages/4e/b6/8fbd4786bb5c0dd19eaff86ddce0fbfb53a6f90d712038272161067a076a/azure_ai_projects-2.0.0b3-py3-none-any.whl", hash = "sha256:3b3048a3ba3904d556ba392b7bd20b6e84c93bb39df6d43a6470cdb0ad08af8c", size = 240717, upload-time = "2026-01-06T05:31:27.716Z" }, ] [[package]] @@ -656,7 +444,7 @@ wheels = [ [[package]] name = "azure-monitor-opentelemetry" -version = "1.7.0" +version = "1.8.5" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "azure-core" }, @@ -672,27 +460,26 @@ dependencies = [ { name = "opentelemetry-resource-detector-azure" }, { name = "opentelemetry-sdk" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/4d/77/be4ae57398fe54fdd97af90df32173f68f37593dc56610c7b04c1643da96/azure_monitor_opentelemetry-1.7.0.tar.gz", hash = "sha256:eba75e793a95d50f6e5bc35dd2781744e2c1a5cc801b530b688f649423f2ee00", size = 51735, upload-time = "2025-08-21T15:52:58.563Z" } +sdist = { url = "https://files.pythonhosted.org/packages/e6/55/b5a48ec320be030eab11126b4636b45ed4ea145f96ddaba6e45974a87add/azure_monitor_opentelemetry-1.8.5.tar.gz", hash = "sha256:7962083a4d650e37e70063edc6315b832b4d6f94d0013ba8428799b36e26a8ce", size = 59683, upload-time = "2026-01-27T21:43:25.657Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d0/bd/b898a883f379d2b4f9bcb9473d4daac24160854d947f17219a7b9211ab34/azure_monitor_opentelemetry-1.7.0-py3-none-any.whl", hash = "sha256:937c60e9706f75c77b221979a273a27e811cc6529d6887099f53916719c66dd3", size = 26316, upload-time = "2025-08-21T15:53:00.153Z" }, + { url = "https://files.pythonhosted.org/packages/6a/18/1df078fce133237f04b9c1018d03bc043a793f1965063d5863aaf1b9947e/azure_monitor_opentelemetry-1.8.5-py3-none-any.whl", hash = "sha256:0f98db1de166ff6bd37ee8d69e657f604cc1785d30607f8daad9bcfdcf3e2111", size = 28986, upload-time = "2026-01-27T21:43:27.231Z" }, ] [[package]] name = "azure-monitor-opentelemetry-exporter" -version = "1.0.0b44" +version = "1.0.0b47" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "azure-core" }, { name = "azure-identity" }, - { name = "fixedint" }, { name = "msrest" }, { name = "opentelemetry-api" }, { name = "opentelemetry-sdk" }, { name = "psutil" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/3e/9a/acb253869ef59482c628f4dc7e049323d0026a9374adf7b398d0b04b6094/azure_monitor_opentelemetry_exporter-1.0.0b44.tar.gz", hash = "sha256:9b0f430a6a46a78bf757ae301488c10c1996f1bd6c5c01a07b9d33583cc4fa4b", size = 271712, upload-time = "2025-10-14T00:27:20.869Z" } +sdist = { url = "https://files.pythonhosted.org/packages/c7/3b/d0e4d8e0f61cb82fd3e94e52291036a7415321f9f7d5386ddb1277d31faa/azure_monitor_opentelemetry_exporter-1.0.0b47.tar.gz", hash = "sha256:c1207bd1c356aa77255e256f1af8eb2ac40a3bf51f90735f456056def7ac38c0", size = 279165, upload-time = "2026-02-03T15:41:17.604Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/4a/46/31809698a0d50559fde108a4f4cb2d9532967ae514a113dba39763e048b7/azure_monitor_opentelemetry_exporter-1.0.0b44-py2.py3-none-any.whl", hash = "sha256:82d23081bf007acab8d4861229ab482e4666307a29492fbf0bf19981b4d37024", size = 198516, upload-time = "2025-10-14T00:27:22.379Z" }, + { url = "https://files.pythonhosted.org/packages/8d/b1/67361bdb9047591f84b2bbd1e03c3161cf85f718a7532b78b4e48f6eaa38/azure_monitor_opentelemetry_exporter-1.0.0b47-py2.py3-none-any.whl", hash = "sha256:be1eca7ddfc07436793981313a68662e14713902f7e7fa7cf81736f1cf6d8bf8", size = 201193, upload-time = "2026-02-03T15:41:18.892Z" }, ] [[package]] @@ -730,8 +517,8 @@ name = "backend" version = "0.1.0" source = { virtual = "." } dependencies = [ - { name = "agent-framework" }, - { name = "azure-ai-agents" }, + { name = "agent-framework-azure-ai" }, + { name = "agent-framework-core" }, { name = "azure-ai-evaluation" }, { name = "azure-ai-inference" }, { name = "azure-ai-projects" }, @@ -762,25 +549,25 @@ dependencies = [ [package.metadata] requires-dist = [ - { name = "agent-framework", specifier = ">=1.0.0b251105" }, - { name = "azure-ai-agents", specifier = "==1.2.0b5" }, + { name = "agent-framework-azure-ai", specifier = "==1.0.0b260130" }, + { name = "agent-framework-core", specifier = "==1.0.0b260130" }, { name = "azure-ai-evaluation", specifier = "==1.11.0" }, { name = "azure-ai-inference", specifier = "==1.0.0b9" }, - { name = "azure-ai-projects", specifier = "==1.0.0" }, + { name = "azure-ai-projects", specifier = "==2.0.0b3" }, { name = "azure-cosmos", specifier = "==4.9.0" }, { name = "azure-identity", specifier = "==1.24.0" }, { name = "azure-monitor-events-extension", specifier = "==0.1.0" }, - { name = "azure-monitor-opentelemetry", specifier = "==1.7.0" }, + { name = "azure-monitor-opentelemetry", specifier = ">=1.8.0" }, { name = "azure-search-documents", specifier = "==11.5.3" }, { name = "fastapi", specifier = "==0.116.1" }, - { name = "mcp", specifier = "==1.13.1" }, - { name = "openai", specifier = "==1.105.0" }, - { name = "opentelemetry-api", specifier = "==1.36.0" }, - { name = "opentelemetry-exporter-otlp-proto-grpc", specifier = "==1.36.0" }, - { name = "opentelemetry-exporter-otlp-proto-http", specifier = "==1.36.0" }, - { name = "opentelemetry-instrumentation-fastapi", specifier = "==0.57b0" }, - { name = "opentelemetry-instrumentation-openai", specifier = "==0.46.2" }, - { name = "opentelemetry-sdk", specifier = "==1.36.0" }, + { name = "mcp", specifier = ">=1.24.0,<2" }, + { name = "openai", specifier = ">=2.8.0" }, + { name = "opentelemetry-api", specifier = ">=1.39.0" }, + { name = "opentelemetry-exporter-otlp-proto-grpc", specifier = ">=1.39.0" }, + { name = "opentelemetry-exporter-otlp-proto-http", specifier = ">=1.39.0" }, + { name = "opentelemetry-instrumentation-fastapi", specifier = ">=0.57b0" }, + { name = "opentelemetry-instrumentation-openai", specifier = ">=0.46.2" }, + { name = "opentelemetry-sdk", specifier = ">=1.39.0" }, { name = "pexpect", specifier = "==4.9.0" }, { name = "pylint-pydantic", specifier = "==0.3.5" }, { name = "pytest", specifier = "==8.4.1" }, @@ -792,24 +579,6 @@ requires-dist = [ { name = "uvicorn", specifier = "==0.35.0" }, ] -[[package]] -name = "backoff" -version = "2.2.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/47/d7/5bbeb12c44d7c4f2fb5b56abce497eb5ed9f34d85701de869acedd602619/backoff-2.2.1.tar.gz", hash = "sha256:03f829f5bb1923180821643f8753b0502c3b682293992485b0eef2807afa5cba", size = 17001, upload-time = "2022-10-05T19:19:32.061Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/df/73/b6e24bd22e6720ca8ee9a85a0c4a2971af8497d8f3193fa05390cbd46e09/backoff-2.2.1-py3-none-any.whl", hash = "sha256:63579f9a0628e06278f7e47b7d7d5b6ce20dc65c5e96a6f3ca99a6adca0396e8", size = 15148, upload-time = "2022-10-05T19:19:30.546Z" }, -] - -[[package]] -name = "cachetools" -version = "6.2.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/cc/7e/b975b5814bd36faf009faebe22c1072a1fa1168db34d285ef0ba071ad78c/cachetools-6.2.1.tar.gz", hash = "sha256:3f391e4bd8f8bf0931169baf7456cc822705f4e2a31f840d218f445b9a854201", size = 31325, upload-time = "2025-10-12T14:55:30.139Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/96/c5/1e741d26306c42e2bf6ab740b2202872727e0f606033c9dd713f8b93f5a8/cachetools-6.2.1-py3-none-any.whl", hash = "sha256:09868944b6dde876dfd44e1d47e18484541eaf12f26f29b7af91b26cc892d701", size = 11280, upload-time = "2025-10-12T14:55:28.382Z" }, -] - [[package]] name = "certifi" version = "2025.10.5" @@ -1206,15 +975,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/ba/5a/18ad964b0086c6e62e2e7500f7edc89e3faa45033c71c1893d34eed2b2de/dnspython-2.8.0-py3-none-any.whl", hash = "sha256:01d9bbc4a2d76bf0db7c1f729812ded6d912bd318d3b1cf81d30c0f845dbf3af", size = 331094, upload-time = "2025-09-07T18:57:58.071Z" }, ] -[[package]] -name = "docstring-parser" -version = "0.17.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/b2/9d/c3b43da9515bd270df0f80548d9944e389870713cc1fe2b8fb35fe2bcefd/docstring_parser-0.17.0.tar.gz", hash = "sha256:583de4a309722b3315439bb31d64ba3eebada841f2e2cee23b99df001434c912", size = 27442, upload-time = "2025-07-21T07:35:01.868Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/55/e2/2537ebcff11c1ee1ff17d8d0b6f4db75873e3b0fb32c2d4a2ee31ecb310a/docstring_parser-0.17.0-py3-none-any.whl", hash = "sha256:cf2569abd23dce8099b300f9b4fa8191e9582dda731fd533daf54c4551658708", size = 36896, upload-time = "2025-07-21T07:35:00.684Z" }, -] - [[package]] name = "fastapi" version = "0.116.1" @@ -1229,15 +989,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/e5/47/d63c60f59a59467fda0f93f46335c9d18526d7071f025cb5b89d5353ea42/fastapi-0.116.1-py3-none-any.whl", hash = "sha256:c46ac7c312df840f0c9e220f7964bada936781bc4e2e6eb71f1c4d7553786565", size = 95631, upload-time = "2025-07-11T16:22:30.485Z" }, ] -[[package]] -name = "fixedint" -version = "0.1.6" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/32/c6/b1b9b3f69915d51909ef6ebe6352e286ec3d6f2077278af83ec6e3cc569c/fixedint-0.1.6.tar.gz", hash = "sha256:703005d090499d41ce7ce2ee7eae8f7a5589a81acdc6b79f1728a56495f2c799", size = 12750, upload-time = "2020-06-20T22:14:16.544Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c8/6d/8f5307d26ce700a89e5a67d1e1ad15eff977211f9ed3ae90d7b0d67f4e66/fixedint-0.1.6-py3-none-any.whl", hash = "sha256:b8cf9f913735d2904deadda7a6daa9f57100599da1de57a7448ea1be75ae8c9c", size = 12702, upload-time = "2020-06-20T22:14:15.454Z" }, -] - [[package]] name = "frozenlist" version = "1.8.0" @@ -1343,36 +1094,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/9a/9a/e35b4a917281c0b8419d4207f4334c8e8c5dbf4f3f5f9ada73958d937dcc/frozenlist-1.8.0-py3-none-any.whl", hash = "sha256:0c18a16eab41e82c295618a77502e17b195883241c563b00f0aa5106fc4eaa0d", size = 13409, upload-time = "2025-10-06T05:38:16.721Z" }, ] -[[package]] -name = "google-api-core" -version = "2.28.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "google-auth" }, - { name = "googleapis-common-protos" }, - { name = "proto-plus" }, - { name = "protobuf" }, - { name = "requests" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/61/da/83d7043169ac2c8c7469f0e375610d78ae2160134bf1b80634c482fa079c/google_api_core-2.28.1.tar.gz", hash = "sha256:2b405df02d68e68ce0fbc138559e6036559e685159d148ae5861013dc201baf8", size = 176759, upload-time = "2025-10-28T21:34:51.529Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/ed/d4/90197b416cb61cefd316964fd9e7bd8324bcbafabf40eef14a9f20b81974/google_api_core-2.28.1-py3-none-any.whl", hash = "sha256:4021b0f8ceb77a6fb4de6fde4502cecab45062e66ff4f2895169e0b35bc9466c", size = 173706, upload-time = "2025-10-28T21:34:50.151Z" }, -] - -[[package]] -name = "google-auth" -version = "2.43.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "cachetools" }, - { name = "pyasn1-modules" }, - { name = "rsa" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/ff/ef/66d14cf0e01b08d2d51ffc3c20410c4e134a1548fc246a6081eae585a4fe/google_auth-2.43.0.tar.gz", hash = "sha256:88228eee5fc21b62a1b5fe773ca15e67778cb07dc8363adcb4a8827b52d81483", size = 296359, upload-time = "2025-11-06T00:13:36.587Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/6f/d1/385110a9ae86d91cc14c5282c61fe9f4dc41c0b9f7d423c6ad77038c4448/google_auth-2.43.0-py2.py3-none-any.whl", hash = "sha256:af628ba6fa493f75c7e9dbe9373d148ca9f4399b5ea29976519e0a3848eddd16", size = 223114, upload-time = "2025-11-06T00:13:35.209Z" }, -] - [[package]] name = "google-crc32c" version = "1.7.1" @@ -1412,56 +1133,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/c4/ab/09169d5a4612a5f92490806649ac8d41e3ec9129c636754575b3553f4ea4/googleapis_common_protos-1.72.0-py3-none-any.whl", hash = "sha256:4299c5a82d5ae1a9702ada957347726b167f9f8d1fc352477702a1e851ff4038", size = 297515, upload-time = "2025-11-06T18:29:13.14Z" }, ] -[[package]] -name = "greenlet" -version = "3.2.4" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/03/b8/704d753a5a45507a7aab61f18db9509302ed3d0a27ac7e0359ec2905b1a6/greenlet-3.2.4.tar.gz", hash = "sha256:0dca0d95ff849f9a364385f36ab49f50065d76964944638be9691e1832e9f86d", size = 188260, upload-time = "2025-08-07T13:24:33.51Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/a4/de/f28ced0a67749cac23fecb02b694f6473f47686dff6afaa211d186e2ef9c/greenlet-3.2.4-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:96378df1de302bc38e99c3a9aa311967b7dc80ced1dcc6f171e99842987882a2", size = 272305, upload-time = "2025-08-07T13:15:41.288Z" }, - { url = "https://files.pythonhosted.org/packages/09/16/2c3792cba130000bf2a31c5272999113f4764fd9d874fb257ff588ac779a/greenlet-3.2.4-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:1ee8fae0519a337f2329cb78bd7a8e128ec0f881073d43f023c7b8d4831d5246", size = 632472, upload-time = "2025-08-07T13:42:55.044Z" }, - { url = "https://files.pythonhosted.org/packages/ae/8f/95d48d7e3d433e6dae5b1682e4292242a53f22df82e6d3dda81b1701a960/greenlet-3.2.4-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:94abf90142c2a18151632371140b3dba4dee031633fe614cb592dbb6c9e17bc3", size = 644646, upload-time = "2025-08-07T13:45:26.523Z" }, - { url = "https://files.pythonhosted.org/packages/d5/5e/405965351aef8c76b8ef7ad370e5da58d57ef6068df197548b015464001a/greenlet-3.2.4-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.whl", hash = "sha256:4d1378601b85e2e5171b99be8d2dc85f594c79967599328f95c1dc1a40f1c633", size = 640519, upload-time = "2025-08-07T13:53:13.928Z" }, - { url = "https://files.pythonhosted.org/packages/25/5d/382753b52006ce0218297ec1b628e048c4e64b155379331f25a7316eb749/greenlet-3.2.4-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:0db5594dce18db94f7d1650d7489909b57afde4c580806b8d9203b6e79cdc079", size = 639707, upload-time = "2025-08-07T13:18:27.146Z" }, - { url = "https://files.pythonhosted.org/packages/1f/8e/abdd3f14d735b2929290a018ecf133c901be4874b858dd1c604b9319f064/greenlet-3.2.4-cp311-cp311-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2523e5246274f54fdadbce8494458a2ebdcdbc7b802318466ac5606d3cded1f8", size = 587684, upload-time = "2025-08-07T13:18:25.164Z" }, - { url = "https://files.pythonhosted.org/packages/5d/65/deb2a69c3e5996439b0176f6651e0052542bb6c8f8ec2e3fba97c9768805/greenlet-3.2.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:1987de92fec508535687fb807a5cea1560f6196285a4cde35c100b8cd632cc52", size = 1116647, upload-time = "2025-08-07T13:42:38.655Z" }, - { url = "https://files.pythonhosted.org/packages/3f/cc/b07000438a29ac5cfb2194bfc128151d52f333cee74dd7dfe3fb733fc16c/greenlet-3.2.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:55e9c5affaa6775e2c6b67659f3a71684de4c549b3dd9afca3bc773533d284fa", size = 1142073, upload-time = "2025-08-07T13:18:21.737Z" }, - { url = "https://files.pythonhosted.org/packages/67/24/28a5b2fa42d12b3d7e5614145f0bd89714c34c08be6aabe39c14dd52db34/greenlet-3.2.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:c9c6de1940a7d828635fbd254d69db79e54619f165ee7ce32fda763a9cb6a58c", size = 1548385, upload-time = "2025-11-04T12:42:11.067Z" }, - { url = "https://files.pythonhosted.org/packages/6a/05/03f2f0bdd0b0ff9a4f7b99333d57b53a7709c27723ec8123056b084e69cd/greenlet-3.2.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:03c5136e7be905045160b1b9fdca93dd6727b180feeafda6818e6496434ed8c5", size = 1613329, upload-time = "2025-11-04T12:42:12.928Z" }, - { url = "https://files.pythonhosted.org/packages/d8/0f/30aef242fcab550b0b3520b8e3561156857c94288f0332a79928c31a52cf/greenlet-3.2.4-cp311-cp311-win_amd64.whl", hash = "sha256:9c40adce87eaa9ddb593ccb0fa6a07caf34015a29bf8d344811665b573138db9", size = 299100, upload-time = "2025-08-07T13:44:12.287Z" }, - { url = "https://files.pythonhosted.org/packages/44/69/9b804adb5fd0671f367781560eb5eb586c4d495277c93bde4307b9e28068/greenlet-3.2.4-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:3b67ca49f54cede0186854a008109d6ee71f66bd57bb36abd6d0a0267b540cdd", size = 274079, upload-time = "2025-08-07T13:15:45.033Z" }, - { url = "https://files.pythonhosted.org/packages/46/e9/d2a80c99f19a153eff70bc451ab78615583b8dac0754cfb942223d2c1a0d/greenlet-3.2.4-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:ddf9164e7a5b08e9d22511526865780a576f19ddd00d62f8a665949327fde8bb", size = 640997, upload-time = "2025-08-07T13:42:56.234Z" }, - { url = "https://files.pythonhosted.org/packages/3b/16/035dcfcc48715ccd345f3a93183267167cdd162ad123cd93067d86f27ce4/greenlet-3.2.4-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:f28588772bb5fb869a8eb331374ec06f24a83a9c25bfa1f38b6993afe9c1e968", size = 655185, upload-time = "2025-08-07T13:45:27.624Z" }, - { url = "https://files.pythonhosted.org/packages/31/da/0386695eef69ffae1ad726881571dfe28b41970173947e7c558d9998de0f/greenlet-3.2.4-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.whl", hash = "sha256:5c9320971821a7cb77cfab8d956fa8e39cd07ca44b6070db358ceb7f8797c8c9", size = 649926, upload-time = "2025-08-07T13:53:15.251Z" }, - { url = "https://files.pythonhosted.org/packages/68/88/69bf19fd4dc19981928ceacbc5fd4bb6bc2215d53199e367832e98d1d8fe/greenlet-3.2.4-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:c60a6d84229b271d44b70fb6e5fa23781abb5d742af7b808ae3f6efd7c9c60f6", size = 651839, upload-time = "2025-08-07T13:18:30.281Z" }, - { url = "https://files.pythonhosted.org/packages/19/0d/6660d55f7373b2ff8152401a83e02084956da23ae58cddbfb0b330978fe9/greenlet-3.2.4-cp312-cp312-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3b3812d8d0c9579967815af437d96623f45c0f2ae5f04e366de62a12d83a8fb0", size = 607586, upload-time = "2025-08-07T13:18:28.544Z" }, - { url = "https://files.pythonhosted.org/packages/8e/1a/c953fdedd22d81ee4629afbb38d2f9d71e37d23caace44775a3a969147d4/greenlet-3.2.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:abbf57b5a870d30c4675928c37278493044d7c14378350b3aa5d484fa65575f0", size = 1123281, upload-time = "2025-08-07T13:42:39.858Z" }, - { url = "https://files.pythonhosted.org/packages/3f/c7/12381b18e21aef2c6bd3a636da1088b888b97b7a0362fac2e4de92405f97/greenlet-3.2.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:20fb936b4652b6e307b8f347665e2c615540d4b42b3b4c8a321d8286da7e520f", size = 1151142, upload-time = "2025-08-07T13:18:22.981Z" }, - { url = "https://files.pythonhosted.org/packages/27/45/80935968b53cfd3f33cf99ea5f08227f2646e044568c9b1555b58ffd61c2/greenlet-3.2.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:ee7a6ec486883397d70eec05059353b8e83eca9168b9f3f9a361971e77e0bcd0", size = 1564846, upload-time = "2025-11-04T12:42:15.191Z" }, - { url = "https://files.pythonhosted.org/packages/69/02/b7c30e5e04752cb4db6202a3858b149c0710e5453b71a3b2aec5d78a1aab/greenlet-3.2.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:326d234cbf337c9c3def0676412eb7040a35a768efc92504b947b3e9cfc7543d", size = 1633814, upload-time = "2025-11-04T12:42:17.175Z" }, - { url = "https://files.pythonhosted.org/packages/e9/08/b0814846b79399e585f974bbeebf5580fbe59e258ea7be64d9dfb253c84f/greenlet-3.2.4-cp312-cp312-win_amd64.whl", hash = "sha256:a7d4e128405eea3814a12cc2605e0e6aedb4035bf32697f72deca74de4105e02", size = 299899, upload-time = "2025-08-07T13:38:53.448Z" }, - { url = "https://files.pythonhosted.org/packages/49/e8/58c7f85958bda41dafea50497cbd59738c5c43dbbea5ee83d651234398f4/greenlet-3.2.4-cp313-cp313-macosx_11_0_universal2.whl", hash = "sha256:1a921e542453fe531144e91e1feedf12e07351b1cf6c9e8a3325ea600a715a31", size = 272814, upload-time = "2025-08-07T13:15:50.011Z" }, - { url = "https://files.pythonhosted.org/packages/62/dd/b9f59862e9e257a16e4e610480cfffd29e3fae018a68c2332090b53aac3d/greenlet-3.2.4-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:cd3c8e693bff0fff6ba55f140bf390fa92c994083f838fece0f63be121334945", size = 641073, upload-time = "2025-08-07T13:42:57.23Z" }, - { url = "https://files.pythonhosted.org/packages/f7/0b/bc13f787394920b23073ca3b6c4a7a21396301ed75a655bcb47196b50e6e/greenlet-3.2.4-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:710638eb93b1fa52823aa91bf75326f9ecdfd5e0466f00789246a5280f4ba0fc", size = 655191, upload-time = "2025-08-07T13:45:29.752Z" }, - { url = "https://files.pythonhosted.org/packages/f2/d6/6adde57d1345a8d0f14d31e4ab9c23cfe8e2cd39c3baf7674b4b0338d266/greenlet-3.2.4-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.whl", hash = "sha256:c5111ccdc9c88f423426df3fd1811bfc40ed66264d35aa373420a34377efc98a", size = 649516, upload-time = "2025-08-07T13:53:16.314Z" }, - { url = "https://files.pythonhosted.org/packages/7f/3b/3a3328a788d4a473889a2d403199932be55b1b0060f4ddd96ee7cdfcad10/greenlet-3.2.4-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:d76383238584e9711e20ebe14db6c88ddcedc1829a9ad31a584389463b5aa504", size = 652169, upload-time = "2025-08-07T13:18:32.861Z" }, - { url = "https://files.pythonhosted.org/packages/ee/43/3cecdc0349359e1a527cbf2e3e28e5f8f06d3343aaf82ca13437a9aa290f/greenlet-3.2.4-cp313-cp313-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:23768528f2911bcd7e475210822ffb5254ed10d71f4028387e5a99b4c6699671", size = 610497, upload-time = "2025-08-07T13:18:31.636Z" }, - { url = "https://files.pythonhosted.org/packages/b8/19/06b6cf5d604e2c382a6f31cafafd6f33d5dea706f4db7bdab184bad2b21d/greenlet-3.2.4-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:00fadb3fedccc447f517ee0d3fd8fe49eae949e1cd0f6a611818f4f6fb7dc83b", size = 1121662, upload-time = "2025-08-07T13:42:41.117Z" }, - { url = "https://files.pythonhosted.org/packages/a2/15/0d5e4e1a66fab130d98168fe984c509249c833c1a3c16806b90f253ce7b9/greenlet-3.2.4-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:d25c5091190f2dc0eaa3f950252122edbbadbb682aa7b1ef2f8af0f8c0afefae", size = 1149210, upload-time = "2025-08-07T13:18:24.072Z" }, - { url = "https://files.pythonhosted.org/packages/1c/53/f9c440463b3057485b8594d7a638bed53ba531165ef0ca0e6c364b5cc807/greenlet-3.2.4-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:6e343822feb58ac4d0a1211bd9399de2b3a04963ddeec21530fc426cc121f19b", size = 1564759, upload-time = "2025-11-04T12:42:19.395Z" }, - { url = "https://files.pythonhosted.org/packages/47/e4/3bb4240abdd0a8d23f4f88adec746a3099f0d86bfedb623f063b2e3b4df0/greenlet-3.2.4-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ca7f6f1f2649b89ce02f6f229d7c19f680a6238af656f61e0115b24857917929", size = 1634288, upload-time = "2025-11-04T12:42:21.174Z" }, - { url = "https://files.pythonhosted.org/packages/0b/55/2321e43595e6801e105fcfdee02b34c0f996eb71e6ddffca6b10b7e1d771/greenlet-3.2.4-cp313-cp313-win_amd64.whl", hash = "sha256:554b03b6e73aaabec3745364d6239e9e012d64c68ccd0b8430c64ccc14939a8b", size = 299685, upload-time = "2025-08-07T13:24:38.824Z" }, - { url = "https://files.pythonhosted.org/packages/22/5c/85273fd7cc388285632b0498dbbab97596e04b154933dfe0f3e68156c68c/greenlet-3.2.4-cp314-cp314-macosx_11_0_universal2.whl", hash = "sha256:49a30d5fda2507ae77be16479bdb62a660fa51b1eb4928b524975b3bde77b3c0", size = 273586, upload-time = "2025-08-07T13:16:08.004Z" }, - { url = "https://files.pythonhosted.org/packages/d1/75/10aeeaa3da9332c2e761e4c50d4c3556c21113ee3f0afa2cf5769946f7a3/greenlet-3.2.4-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:299fd615cd8fc86267b47597123e3f43ad79c9d8a22bebdce535e53550763e2f", size = 686346, upload-time = "2025-08-07T13:42:59.944Z" }, - { url = "https://files.pythonhosted.org/packages/c0/aa/687d6b12ffb505a4447567d1f3abea23bd20e73a5bed63871178e0831b7a/greenlet-3.2.4-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.whl", hash = "sha256:c17b6b34111ea72fc5a4e4beec9711d2226285f0386ea83477cbb97c30a3f3a5", size = 699218, upload-time = "2025-08-07T13:45:30.969Z" }, - { url = "https://files.pythonhosted.org/packages/dc/8b/29aae55436521f1d6f8ff4e12fb676f3400de7fcf27fccd1d4d17fd8fecd/greenlet-3.2.4-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.whl", hash = "sha256:b4a1870c51720687af7fa3e7cda6d08d801dae660f75a76f3845b642b4da6ee1", size = 694659, upload-time = "2025-08-07T13:53:17.759Z" }, - { url = "https://files.pythonhosted.org/packages/92/2e/ea25914b1ebfde93b6fc4ff46d6864564fba59024e928bdc7de475affc25/greenlet-3.2.4-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:061dc4cf2c34852b052a8620d40f36324554bc192be474b9e9770e8c042fd735", size = 695355, upload-time = "2025-08-07T13:18:34.517Z" }, - { url = "https://files.pythonhosted.org/packages/72/60/fc56c62046ec17f6b0d3060564562c64c862948c9d4bc8aa807cf5bd74f4/greenlet-3.2.4-cp314-cp314-manylinux_2_24_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:44358b9bf66c8576a9f57a590d5f5d6e72fa4228b763d0e43fee6d3b06d3a337", size = 657512, upload-time = "2025-08-07T13:18:33.969Z" }, - { url = "https://files.pythonhosted.org/packages/23/6e/74407aed965a4ab6ddd93a7ded3180b730d281c77b765788419484cdfeef/greenlet-3.2.4-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:2917bdf657f5859fbf3386b12d68ede4cf1f04c90c3a6bc1f013dd68a22e2269", size = 1612508, upload-time = "2025-11-04T12:42:23.427Z" }, - { url = "https://files.pythonhosted.org/packages/0d/da/343cd760ab2f92bac1845ca07ee3faea9fe52bee65f7bcb19f16ad7de08b/greenlet-3.2.4-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:015d48959d4add5d6c9f6c5210ee3803a830dce46356e3bc326d6776bde54681", size = 1680760, upload-time = "2025-11-04T12:42:25.341Z" }, - { url = "https://files.pythonhosted.org/packages/e3/a5/6ddab2b4c112be95601c13428db1d8b6608a8b6039816f2ba09c346c08fc/greenlet-3.2.4-cp314-cp314-win_amd64.whl", hash = "sha256:e37ab26028f12dbb0ff65f29a8d3d44a765c61e729647bf2ddfbbed621726f01", size = 303425, upload-time = "2025-08-07T13:32:27.59Z" }, -] - [[package]] name = "grpcio" version = "1.76.0" @@ -1522,28 +1193,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/04/4b/29cac41a4d98d144bf5f6d33995617b185d14b22401f75ca86f384e87ff1/h11-0.16.0-py3-none-any.whl", hash = "sha256:63cf8bbe7522de3bf65932fda1d9c2772064ffb3dae62d55932da54b31cb6c86", size = 37515, upload-time = "2025-04-24T03:35:24.344Z" }, ] -[[package]] -name = "h2" -version = "4.3.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "hpack" }, - { name = "hyperframe" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/1d/17/afa56379f94ad0fe8defd37d6eb3f89a25404ffc71d4d848893d270325fc/h2-4.3.0.tar.gz", hash = "sha256:6c59efe4323fa18b47a632221a1888bd7fde6249819beda254aeca909f221bf1", size = 2152026, upload-time = "2025-08-23T18:12:19.778Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/69/b2/119f6e6dcbd96f9069ce9a2665e0146588dc9f88f29549711853645e736a/h2-4.3.0-py3-none-any.whl", hash = "sha256:c438f029a25f7945c69e0ccf0fb951dc3f73a5f6412981daee861431b70e2bdd", size = 61779, upload-time = "2025-08-23T18:12:17.779Z" }, -] - -[[package]] -name = "hpack" -version = "4.1.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/2c/48/71de9ed269fdae9c8057e5a4c0aa7402e8bb16f2c6e90b3aa53327b113f8/hpack-4.1.0.tar.gz", hash = "sha256:ec5eca154f7056aa06f196a557655c5b009b382873ac8d1e66e79e87535f1dca", size = 51276, upload-time = "2025-01-22T21:44:58.347Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/07/c6/80c95b1b2b94682a72cbdbfb85b81ae2daffa4291fbfa1b1464502ede10d/hpack-4.1.0-py3-none-any.whl", hash = "sha256:157ac792668d995c657d93111f46b4535ed114f0c9c8d672271bbec7eae1b496", size = 34357, upload-time = "2025-01-22T21:44:56.92Z" }, -] - [[package]] name = "httpcore" version = "1.0.9" @@ -1557,42 +1206,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/7e/f5/f66802a942d491edb555dd61e3a9961140fd64c90bce1eafd741609d334d/httpcore-1.0.9-py3-none-any.whl", hash = "sha256:2d400746a40668fc9dec9810239072b40b4484b640a8c38fd654a024c7a1bf55", size = 78784, upload-time = "2025-04-24T22:06:20.566Z" }, ] -[[package]] -name = "httptools" -version = "0.7.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/b5/46/120a669232c7bdedb9d52d4aeae7e6c7dfe151e99dc70802e2fc7a5e1993/httptools-0.7.1.tar.gz", hash = "sha256:abd72556974f8e7c74a259655924a717a2365b236c882c3f6f8a45fe94703ac9", size = 258961, upload-time = "2025-10-10T03:55:08.559Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/9c/08/17e07e8d89ab8f343c134616d72eebfe03798835058e2ab579dcc8353c06/httptools-0.7.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:474d3b7ab469fefcca3697a10d11a32ee2b9573250206ba1e50d5980910da657", size = 206521, upload-time = "2025-10-10T03:54:31.002Z" }, - { url = "https://files.pythonhosted.org/packages/aa/06/c9c1b41ff52f16aee526fd10fbda99fa4787938aa776858ddc4a1ea825ec/httptools-0.7.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a3c3b7366bb6c7b96bd72d0dbe7f7d5eead261361f013be5f6d9590465ea1c70", size = 110375, upload-time = "2025-10-10T03:54:31.941Z" }, - { url = "https://files.pythonhosted.org/packages/cc/cc/10935db22fda0ee34c76f047590ca0a8bd9de531406a3ccb10a90e12ea21/httptools-0.7.1-cp311-cp311-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:379b479408b8747f47f3b253326183d7c009a3936518cdb70db58cffd369d9df", size = 456621, upload-time = "2025-10-10T03:54:33.176Z" }, - { url = "https://files.pythonhosted.org/packages/0e/84/875382b10d271b0c11aa5d414b44f92f8dd53e9b658aec338a79164fa548/httptools-0.7.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:cad6b591a682dcc6cf1397c3900527f9affef1e55a06c4547264796bbd17cf5e", size = 454954, upload-time = "2025-10-10T03:54:34.226Z" }, - { url = "https://files.pythonhosted.org/packages/30/e1/44f89b280f7e46c0b1b2ccee5737d46b3bb13136383958f20b580a821ca0/httptools-0.7.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:eb844698d11433d2139bbeeb56499102143beb582bd6c194e3ba69c22f25c274", size = 440175, upload-time = "2025-10-10T03:54:35.942Z" }, - { url = "https://files.pythonhosted.org/packages/6f/7e/b9287763159e700e335028bc1824359dc736fa9b829dacedace91a39b37e/httptools-0.7.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f65744d7a8bdb4bda5e1fa23e4ba16832860606fcc09d674d56e425e991539ec", size = 440310, upload-time = "2025-10-10T03:54:37.1Z" }, - { url = "https://files.pythonhosted.org/packages/b3/07/5b614f592868e07f5c94b1f301b5e14a21df4e8076215a3bccb830a687d8/httptools-0.7.1-cp311-cp311-win_amd64.whl", hash = "sha256:135fbe974b3718eada677229312e97f3b31f8a9c8ffa3ae6f565bf808d5b6bcb", size = 86875, upload-time = "2025-10-10T03:54:38.421Z" }, - { url = "https://files.pythonhosted.org/packages/53/7f/403e5d787dc4942316e515e949b0c8a013d84078a915910e9f391ba9b3ed/httptools-0.7.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:38e0c83a2ea9746ebbd643bdfb521b9aa4a91703e2cd705c20443405d2fd16a5", size = 206280, upload-time = "2025-10-10T03:54:39.274Z" }, - { url = "https://files.pythonhosted.org/packages/2a/0d/7f3fd28e2ce311ccc998c388dd1c53b18120fda3b70ebb022b135dc9839b/httptools-0.7.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f25bbaf1235e27704f1a7b86cd3304eabc04f569c828101d94a0e605ef7205a5", size = 110004, upload-time = "2025-10-10T03:54:40.403Z" }, - { url = "https://files.pythonhosted.org/packages/84/a6/b3965e1e146ef5762870bbe76117876ceba51a201e18cc31f5703e454596/httptools-0.7.1-cp312-cp312-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:2c15f37ef679ab9ecc06bfc4e6e8628c32a8e4b305459de7cf6785acd57e4d03", size = 517655, upload-time = "2025-10-10T03:54:41.347Z" }, - { url = "https://files.pythonhosted.org/packages/11/7d/71fee6f1844e6fa378f2eddde6c3e41ce3a1fb4b2d81118dd544e3441ec0/httptools-0.7.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:7fe6e96090df46b36ccfaf746f03034e5ab723162bc51b0a4cf58305324036f2", size = 511440, upload-time = "2025-10-10T03:54:42.452Z" }, - { url = "https://files.pythonhosted.org/packages/22/a5/079d216712a4f3ffa24af4a0381b108aa9c45b7a5cc6eb141f81726b1823/httptools-0.7.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:f72fdbae2dbc6e68b8239defb48e6a5937b12218e6ffc2c7846cc37befa84362", size = 495186, upload-time = "2025-10-10T03:54:43.937Z" }, - { url = "https://files.pythonhosted.org/packages/e9/9e/025ad7b65278745dee3bd0ebf9314934c4592560878308a6121f7f812084/httptools-0.7.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:e99c7b90a29fd82fea9ef57943d501a16f3404d7b9ee81799d41639bdaae412c", size = 499192, upload-time = "2025-10-10T03:54:45.003Z" }, - { url = "https://files.pythonhosted.org/packages/6d/de/40a8f202b987d43afc4d54689600ff03ce65680ede2f31df348d7f368b8f/httptools-0.7.1-cp312-cp312-win_amd64.whl", hash = "sha256:3e14f530fefa7499334a79b0cf7e7cd2992870eb893526fb097d51b4f2d0f321", size = 86694, upload-time = "2025-10-10T03:54:45.923Z" }, - { url = "https://files.pythonhosted.org/packages/09/8f/c77b1fcbfd262d422f12da02feb0d218fa228d52485b77b953832105bb90/httptools-0.7.1-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:6babce6cfa2a99545c60bfef8bee0cc0545413cb0018f617c8059a30ad985de3", size = 202889, upload-time = "2025-10-10T03:54:47.089Z" }, - { url = "https://files.pythonhosted.org/packages/0a/1a/22887f53602feaa066354867bc49a68fc295c2293433177ee90870a7d517/httptools-0.7.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:601b7628de7504077dd3dcb3791c6b8694bbd967148a6d1f01806509254fb1ca", size = 108180, upload-time = "2025-10-10T03:54:48.052Z" }, - { url = "https://files.pythonhosted.org/packages/32/6a/6aaa91937f0010d288d3d124ca2946d48d60c3a5ee7ca62afe870e3ea011/httptools-0.7.1-cp313-cp313-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:04c6c0e6c5fb0739c5b8a9eb046d298650a0ff38cf42537fc372b28dc7e4472c", size = 478596, upload-time = "2025-10-10T03:54:48.919Z" }, - { url = "https://files.pythonhosted.org/packages/6d/70/023d7ce117993107be88d2cbca566a7c1323ccbaf0af7eabf2064fe356f6/httptools-0.7.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:69d4f9705c405ae3ee83d6a12283dc9feba8cc6aaec671b412917e644ab4fa66", size = 473268, upload-time = "2025-10-10T03:54:49.993Z" }, - { url = "https://files.pythonhosted.org/packages/32/4d/9dd616c38da088e3f436e9a616e1d0cc66544b8cdac405cc4e81c8679fc7/httptools-0.7.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:44c8f4347d4b31269c8a9205d8a5ee2df5322b09bbbd30f8f862185bb6b05346", size = 455517, upload-time = "2025-10-10T03:54:51.066Z" }, - { url = "https://files.pythonhosted.org/packages/1d/3a/a6c595c310b7df958e739aae88724e24f9246a514d909547778d776799be/httptools-0.7.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:465275d76db4d554918aba40bf1cbebe324670f3dfc979eaffaa5d108e2ed650", size = 458337, upload-time = "2025-10-10T03:54:52.196Z" }, - { url = "https://files.pythonhosted.org/packages/fd/82/88e8d6d2c51edc1cc391b6e044c6c435b6aebe97b1abc33db1b0b24cd582/httptools-0.7.1-cp313-cp313-win_amd64.whl", hash = "sha256:322d00c2068d125bd570f7bf78b2d367dad02b919d8581d7476d8b75b294e3e6", size = 85743, upload-time = "2025-10-10T03:54:53.448Z" }, - { url = "https://files.pythonhosted.org/packages/34/50/9d095fcbb6de2d523e027a2f304d4551855c2f46e0b82befd718b8b20056/httptools-0.7.1-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:c08fe65728b8d70b6923ce31e3956f859d5e1e8548e6f22ec520a962c6757270", size = 203619, upload-time = "2025-10-10T03:54:54.321Z" }, - { url = "https://files.pythonhosted.org/packages/07/f0/89720dc5139ae54b03f861b5e2c55a37dba9a5da7d51e1e824a1f343627f/httptools-0.7.1-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:7aea2e3c3953521c3c51106ee11487a910d45586e351202474d45472db7d72d3", size = 108714, upload-time = "2025-10-10T03:54:55.163Z" }, - { url = "https://files.pythonhosted.org/packages/b3/cb/eea88506f191fb552c11787c23f9a405f4c7b0c5799bf73f2249cd4f5228/httptools-0.7.1-cp314-cp314-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:0e68b8582f4ea9166be62926077a3334064d422cf08ab87d8b74664f8e9058e1", size = 472909, upload-time = "2025-10-10T03:54:56.056Z" }, - { url = "https://files.pythonhosted.org/packages/e0/4a/a548bdfae6369c0d078bab5769f7b66f17f1bfaa6fa28f81d6be6959066b/httptools-0.7.1-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:df091cf961a3be783d6aebae963cc9b71e00d57fa6f149025075217bc6a55a7b", size = 470831, upload-time = "2025-10-10T03:54:57.219Z" }, - { url = "https://files.pythonhosted.org/packages/4d/31/14df99e1c43bd132eec921c2e7e11cda7852f65619bc0fc5bdc2d0cb126c/httptools-0.7.1-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:f084813239e1eb403ddacd06a30de3d3e09a9b76e7894dcda2b22f8a726e9c60", size = 452631, upload-time = "2025-10-10T03:54:58.219Z" }, - { url = "https://files.pythonhosted.org/packages/22/d2/b7e131f7be8d854d48cb6d048113c30f9a46dca0c9a8b08fcb3fcd588cdc/httptools-0.7.1-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:7347714368fb2b335e9063bc2b96f2f87a9ceffcd9758ac295f8bbcd3ffbc0ca", size = 452910, upload-time = "2025-10-10T03:54:59.366Z" }, - { url = "https://files.pythonhosted.org/packages/53/cf/878f3b91e4e6e011eff6d1fa9ca39f7eb17d19c9d7971b04873734112f30/httptools-0.7.1-cp314-cp314-win_amd64.whl", hash = "sha256:cfabda2a5bb85aa2a904ce06d974a3f30fb36cc63d7feaddec05d2050acede96", size = 88205, upload-time = "2025-10-10T03:55:00.389Z" }, -] - [[package]] name = "httpx" version = "0.28.1" @@ -1608,11 +1221,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/2a/39/e50c7c3a983047577ee07d2a9e53faf5a69493943ec3f6a384bdc792deb2/httpx-0.28.1-py3-none-any.whl", hash = "sha256:d909fcccc110f8c7faf814ca82a9a4d816bc5a6dbfea25d6591d6985b8ba59ad", size = 73517, upload-time = "2024-12-06T15:37:21.509Z" }, ] -[package.optional-dependencies] -http2 = [ - { name = "h2" }, -] - [[package]] name = "httpx-sse" version = "0.4.3" @@ -1622,15 +1230,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/d2/fd/6668e5aec43ab844de6fc74927e155a3b37bf40d7c3790e49fc0406b6578/httpx_sse-0.4.3-py3-none-any.whl", hash = "sha256:0ac1c9fe3c0afad2e0ebb25a934a59f4c7823b60792691f779fad2c5568830fc", size = 8960, upload-time = "2025-10-10T21:48:21.158Z" }, ] -[[package]] -name = "hyperframe" -version = "6.1.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/02/e7/94f8232d4a74cc99514c13a9f995811485a6903d48e5d952771ef6322e30/hyperframe-6.1.0.tar.gz", hash = "sha256:f630908a00854a7adeabd6382b43923a4c4cd4b821fcb527e6ab9e15382a3b08", size = 26566, upload-time = "2025-01-22T21:41:49.302Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/48/30/47d0bf6072f7252e6521f3447ccfa40b421b6824517f82854703d0f5a98b/hyperframe-6.1.0-py3-none-any.whl", hash = "sha256:b03380493a519fce58ea5af42e4a42317bf9bd425596f7a0835ffce80f1a42e5", size = 13007, upload-time = "2025-01-22T21:41:47.295Z" }, -] - [[package]] name = "idna" version = "3.11" @@ -1794,18 +1393,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/1e/e8/685f47e0d754320684db4425a0967f7d3fa70126bffd76110b7009a0090f/joblib-1.5.2-py3-none-any.whl", hash = "sha256:4e1f0bdbb987e6d843c70cf43714cb276623def372df3c22fe5266b2670bc241", size = 308396, upload-time = "2025-08-27T12:15:45.188Z" }, ] -[[package]] -name = "jsonpath-ng" -version = "1.7.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "ply" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/6d/86/08646239a313f895186ff0a4573452038eed8c86f54380b3ebac34d32fb2/jsonpath-ng-1.7.0.tar.gz", hash = "sha256:f6f5f7fd4e5ff79c785f1573b394043b39849fb2bb47bcead935d12b00beab3c", size = 37838, upload-time = "2024-10-11T15:41:42.404Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/35/5a/73ecb3d82f8615f32ccdadeb9356726d6cae3a4bbc840b437ceb95708063/jsonpath_ng-1.7.0-py3-none-any.whl", hash = "sha256:f3d7f9e848cba1b6da28c55b1c26ff915dc9e0b1ba7e752a53d6da8d5cbd00b6", size = 30105, upload-time = "2024-11-20T17:58:30.418Z" }, -] - [[package]] name = "jsonschema" version = "4.25.1" @@ -1972,7 +1559,7 @@ wheels = [ [[package]] name = "mcp" -version = "1.13.1" +version = "1.26.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "anyio" }, @@ -1981,15 +1568,18 @@ dependencies = [ { name = "jsonschema" }, { name = "pydantic" }, { name = "pydantic-settings" }, + { name = "pyjwt", extra = ["crypto"] }, { name = "python-multipart" }, { name = "pywin32", marker = "sys_platform == 'win32'" }, { name = "sse-starlette" }, { name = "starlette" }, + { name = "typing-extensions" }, + { name = "typing-inspection" }, { name = "uvicorn", marker = "sys_platform != 'emscripten'" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/66/3c/82c400c2d50afdac4fbefb5b4031fd327e2ad1f23ccef8eee13c5909aa48/mcp-1.13.1.tar.gz", hash = "sha256:165306a8fd7991dc80334edd2de07798175a56461043b7ae907b279794a834c5", size = 438198, upload-time = "2025-08-22T09:22:16.061Z" } +sdist = { url = "https://files.pythonhosted.org/packages/fc/6d/62e76bbb8144d6ed86e202b5edd8a4cb631e7c8130f3f4893c3f90262b10/mcp-1.26.0.tar.gz", hash = "sha256:db6e2ef491eecc1a0d93711a76f28dec2e05999f93afd48795da1c1137142c66", size = 608005, upload-time = "2026-01-24T19:40:32.468Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/19/3f/d085c7f49ade6d273b185d61ec9405e672b6433f710ea64a90135a8dd445/mcp-1.13.1-py3-none-any.whl", hash = "sha256:c314e7c8bd477a23ba3ef472ee5a32880316c42d03e06dcfa31a1cc7a73b65df", size = 161494, upload-time = "2025-08-22T09:22:14.705Z" }, + { url = "https://files.pythonhosted.org/packages/fd/d9/eaa1f80170d2b7c5ba23f3b59f766f3a0bb41155fbc32a69adfa1adaaef9/mcp-1.26.0-py3-none-any.whl", hash = "sha256:904a21c33c25aa98ddbeb47273033c435e595bbacfdb177f4bd87f6dceebe1ca", size = 233615, upload-time = "2026-01-24T19:40:30.652Z" }, ] [package.optional-dependencies] @@ -1997,101 +1587,6 @@ ws = [ { name = "websockets" }, ] -[[package]] -name = "mem0ai" -version = "1.0.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "openai" }, - { name = "posthog" }, - { name = "protobuf" }, - { name = "pydantic" }, - { name = "pytz" }, - { name = "qdrant-client" }, - { name = "sqlalchemy" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/99/02/b6c3bba83b4bb6450e6c8a07e4419b24644007588f5ef427b680addbd30f/mem0ai-1.0.0.tar.gz", hash = "sha256:8a891502e6547436adb526a59acf091cacaa689e182e186f4dd8baf185d75224", size = 177780, upload-time = "2025-10-16T10:36:23.871Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/61/49/eed6e2a77bf90e37da25c9a336af6a6129b0baae76551409ee995f0a1f0c/mem0ai-1.0.0-py3-none-any.whl", hash = "sha256:107fd2990613eba34880ca6578e6cdd4a8158fd35f5b80be031b6e2b5a66a1f1", size = 268141, upload-time = "2025-10-16T10:36:21.63Z" }, -] - -[[package]] -name = "microsoft-agents-activity" -version = "0.5.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pydantic" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/7e/51/2698980f425cda122f5b755a957c3c2db604c0b9a787c6add5aa4649c237/microsoft_agents_activity-0.5.3.tar.gz", hash = "sha256:d80b055591df561df8cebda9e1712012352581a396b36459133a951982b3a760", size = 55892, upload-time = "2025-10-31T15:40:49.332Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/75/3d/9618243e7b6f1f6295642c4e2dfca65b3a37794efbe1bdec15f0a93827d9/microsoft_agents_activity-0.5.3-py3-none-any.whl", hash = "sha256:5ae2447ac47c32f03c614694f520817cd225c9c502ec08b90d448311fb5bf3b4", size = 127861, upload-time = "2025-10-31T15:40:57.628Z" }, -] - -[[package]] -name = "microsoft-agents-copilotstudio-client" -version = "0.5.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "microsoft-agents-hosting-core" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/7e/22/109164fb585c4baee40d2372c5d76254ec4a28219908f11cd27ac92aa6c1/microsoft_agents_copilotstudio_client-0.5.3.tar.gz", hash = "sha256:a57ea6b3cb47dbb5ad22e59c986208ace6479e35da3f644e6346f4dfd85db57c", size = 11161, upload-time = "2025-10-31T15:40:51.444Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c4/65/984e139c85657ff0c8df0ed98a167c8b9434f4fd4f32862b4a6490b8c714/microsoft_agents_copilotstudio_client-0.5.3-py3-none-any.whl", hash = "sha256:6a36fce5c8c1a2df6f5142e35b12c69be80959ecff6d60cc309661018c40f00a", size = 11091, upload-time = "2025-10-31T15:40:59.718Z" }, -] - -[[package]] -name = "microsoft-agents-hosting-core" -version = "0.5.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "azure-core" }, - { name = "isodate" }, - { name = "microsoft-agents-activity" }, - { name = "pyjwt" }, - { name = "python-dotenv" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/b1/98/7755c07b2ae5faf3e4dc14b17e44680a600c8b840b3003fb326d5720dea1/microsoft_agents_hosting_core-0.5.3.tar.gz", hash = "sha256:b113d4ea5c9e555bbf61037bb2a1a7a3ce7e5e4a7a0f681a3bd4719ba72ff821", size = 81672, upload-time = "2025-10-31T15:40:53.557Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/95/57/c9e98475971c9da9cc9ff88195bbfcfae90dba511ebe14610be79f23ab3f/microsoft_agents_hosting_core-0.5.3-py3-none-any.whl", hash = "sha256:8c228a8814dcf1a86dd60e4c7574a2e86078962695fabd693a118097e703e982", size = 120668, upload-time = "2025-10-31T15:41:01.691Z" }, -] - -[[package]] -name = "ml-dtypes" -version = "0.5.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "numpy" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/78/a7/aad060393123cfb383956dca68402aff3db1e1caffd5764887ed5153f41b/ml_dtypes-0.5.3.tar.gz", hash = "sha256:95ce33057ba4d05df50b1f3cfefab22e351868a843b3b15a46c65836283670c9", size = 692316, upload-time = "2025-07-29T18:39:19.454Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/af/f1/720cb1409b5d0c05cff9040c0e9fba73fa4c67897d33babf905d5d46a070/ml_dtypes-0.5.3-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4a177b882667c69422402df6ed5c3428ce07ac2c1f844d8a1314944651439458", size = 667412, upload-time = "2025-07-29T18:38:25.275Z" }, - { url = "https://files.pythonhosted.org/packages/6a/d5/05861ede5d299f6599f86e6bc1291714e2116d96df003cfe23cc54bcc568/ml_dtypes-0.5.3-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9849ce7267444c0a717c80c6900997de4f36e2815ce34ac560a3edb2d9a64cd2", size = 4964606, upload-time = "2025-07-29T18:38:27.045Z" }, - { url = "https://files.pythonhosted.org/packages/db/dc/72992b68de367741bfab8df3b3fe7c29f982b7279d341aa5bf3e7ef737ea/ml_dtypes-0.5.3-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c3f5ae0309d9f888fd825c2e9d0241102fadaca81d888f26f845bc8c13c1e4ee", size = 4938435, upload-time = "2025-07-29T18:38:29.193Z" }, - { url = "https://files.pythonhosted.org/packages/81/1c/d27a930bca31fb07d975a2d7eaf3404f9388114463b9f15032813c98f893/ml_dtypes-0.5.3-cp311-cp311-win_amd64.whl", hash = "sha256:58e39349d820b5702bb6f94ea0cb2dc8ec62ee81c0267d9622067d8333596a46", size = 206334, upload-time = "2025-07-29T18:38:30.687Z" }, - { url = "https://files.pythonhosted.org/packages/1a/d8/6922499effa616012cb8dc445280f66d100a7ff39b35c864cfca019b3f89/ml_dtypes-0.5.3-cp311-cp311-win_arm64.whl", hash = "sha256:66c2756ae6cfd7f5224e355c893cfd617fa2f747b8bbd8996152cbdebad9a184", size = 157584, upload-time = "2025-07-29T18:38:32.187Z" }, - { url = "https://files.pythonhosted.org/packages/0d/eb/bc07c88a6ab002b4635e44585d80fa0b350603f11a2097c9d1bfacc03357/ml_dtypes-0.5.3-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:156418abeeda48ea4797db6776db3c5bdab9ac7be197c1233771e0880c304057", size = 663864, upload-time = "2025-07-29T18:38:33.777Z" }, - { url = "https://files.pythonhosted.org/packages/cf/89/11af9b0f21b99e6386b6581ab40fb38d03225f9de5f55cf52097047e2826/ml_dtypes-0.5.3-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1db60c154989af253f6c4a34e8a540c2c9dce4d770784d426945e09908fbb177", size = 4951313, upload-time = "2025-07-29T18:38:36.45Z" }, - { url = "https://files.pythonhosted.org/packages/d8/a9/b98b86426c24900b0c754aad006dce2863df7ce0bb2bcc2c02f9cc7e8489/ml_dtypes-0.5.3-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1b255acada256d1fa8c35ed07b5f6d18bc21d1556f842fbc2d5718aea2cd9e55", size = 4928805, upload-time = "2025-07-29T18:38:38.29Z" }, - { url = "https://files.pythonhosted.org/packages/50/c1/85e6be4fc09c6175f36fb05a45917837f30af9a5146a5151cb3a3f0f9e09/ml_dtypes-0.5.3-cp312-cp312-win_amd64.whl", hash = "sha256:da65e5fd3eea434ccb8984c3624bc234ddcc0d9f4c81864af611aaebcc08a50e", size = 208182, upload-time = "2025-07-29T18:38:39.72Z" }, - { url = "https://files.pythonhosted.org/packages/9e/17/cf5326d6867be057f232d0610de1458f70a8ce7b6290e4b4a277ea62b4cd/ml_dtypes-0.5.3-cp312-cp312-win_arm64.whl", hash = "sha256:8bb9cd1ce63096567f5f42851f5843b5a0ea11511e50039a7649619abfb4ba6d", size = 161560, upload-time = "2025-07-29T18:38:41.072Z" }, - { url = "https://files.pythonhosted.org/packages/2d/87/1bcc98a66de7b2455dfb292f271452cac9edc4e870796e0d87033524d790/ml_dtypes-0.5.3-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:5103856a225465371fe119f2fef737402b705b810bd95ad5f348e6e1a6ae21af", size = 663781, upload-time = "2025-07-29T18:38:42.984Z" }, - { url = "https://files.pythonhosted.org/packages/fd/2c/bd2a79ba7c759ee192b5601b675b180a3fd6ccf48ffa27fe1782d280f1a7/ml_dtypes-0.5.3-cp313-cp313-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4cae435a68861660af81fa3c5af16b70ca11a17275c5b662d9c6f58294e0f113", size = 4956217, upload-time = "2025-07-29T18:38:44.65Z" }, - { url = "https://files.pythonhosted.org/packages/14/f3/091ba84e5395d7fe5b30c081a44dec881cd84b408db1763ee50768b2ab63/ml_dtypes-0.5.3-cp313-cp313-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6936283b56d74fbec431ca57ce58a90a908fdbd14d4e2d22eea6d72bb208a7b7", size = 4933109, upload-time = "2025-07-29T18:38:46.405Z" }, - { url = "https://files.pythonhosted.org/packages/bc/24/054036dbe32c43295382c90a1363241684c4d6aaa1ecc3df26bd0c8d5053/ml_dtypes-0.5.3-cp313-cp313-win_amd64.whl", hash = "sha256:d0f730a17cf4f343b2c7ad50cee3bd19e969e793d2be6ed911f43086460096e4", size = 208187, upload-time = "2025-07-29T18:38:48.24Z" }, - { url = "https://files.pythonhosted.org/packages/a6/3d/7dc3ec6794a4a9004c765e0c341e32355840b698f73fd2daff46f128afc1/ml_dtypes-0.5.3-cp313-cp313-win_arm64.whl", hash = "sha256:2db74788fc01914a3c7f7da0763427280adfc9cd377e9604b6b64eb8097284bd", size = 161559, upload-time = "2025-07-29T18:38:50.493Z" }, - { url = "https://files.pythonhosted.org/packages/12/91/e6c7a0d67a152b9330445f9f0cf8ae6eee9b83f990b8c57fe74631e42a90/ml_dtypes-0.5.3-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:93c36a08a6d158db44f2eb9ce3258e53f24a9a4a695325a689494f0fdbc71770", size = 689321, upload-time = "2025-07-29T18:38:52.03Z" }, - { url = "https://files.pythonhosted.org/packages/9e/6c/b7b94b84a104a5be1883305b87d4c6bd6ae781504474b4cca067cb2340ec/ml_dtypes-0.5.3-cp313-cp313t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0e44a3761f64bc009d71ddb6d6c71008ba21b53ab6ee588dadab65e2fa79eafc", size = 5274495, upload-time = "2025-07-29T18:38:53.797Z" }, - { url = "https://files.pythonhosted.org/packages/5b/38/6266604dffb43378055394ea110570cf261a49876fc48f548dfe876f34cc/ml_dtypes-0.5.3-cp313-cp313t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bdf40d2aaabd3913dec11840f0d0ebb1b93134f99af6a0a4fd88ffe924928ab4", size = 5285422, upload-time = "2025-07-29T18:38:56.603Z" }, - { url = "https://files.pythonhosted.org/packages/7c/88/8612ff177d043a474b9408f0382605d881eeb4125ba89d4d4b3286573a83/ml_dtypes-0.5.3-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:aec640bd94c4c85c0d11e2733bd13cbb10438fb004852996ec0efbc6cacdaf70", size = 661182, upload-time = "2025-07-29T18:38:58.414Z" }, - { url = "https://files.pythonhosted.org/packages/6f/2b/0569a5e88b29240d373e835107c94ae9256fb2191d3156b43b2601859eff/ml_dtypes-0.5.3-cp314-cp314-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:bda32ce212baa724e03c68771e5c69f39e584ea426bfe1a701cb01508ffc7035", size = 4956187, upload-time = "2025-07-29T18:39:00.611Z" }, - { url = "https://files.pythonhosted.org/packages/51/66/273c2a06ae44562b104b61e6b14444da00061fd87652506579d7eb2c40b1/ml_dtypes-0.5.3-cp314-cp314-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c205cac07d24a29840c163d6469f61069ce4b065518519216297fc2f261f8db9", size = 4930911, upload-time = "2025-07-29T18:39:02.405Z" }, - { url = "https://files.pythonhosted.org/packages/93/ab/606be3e87dc0821bd360c8c1ee46108025c31a4f96942b63907bb441b87d/ml_dtypes-0.5.3-cp314-cp314-win_amd64.whl", hash = "sha256:cd7c0bb22d4ff86d65ad61b5dd246812e8993fbc95b558553624c33e8b6903ea", size = 216664, upload-time = "2025-07-29T18:39:03.927Z" }, - { url = "https://files.pythonhosted.org/packages/30/a2/e900690ca47d01dffffd66375c5de8c4f8ced0f1ef809ccd3b25b3e6b8fa/ml_dtypes-0.5.3-cp314-cp314-win_arm64.whl", hash = "sha256:9d55ea7f7baf2aed61bf1872116cefc9d0c3693b45cae3916897ee27ef4b835e", size = 160203, upload-time = "2025-07-29T18:39:05.671Z" }, - { url = "https://files.pythonhosted.org/packages/53/21/783dfb51f40d2660afeb9bccf3612b99f6a803d980d2a09132b0f9d216ab/ml_dtypes-0.5.3-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:e12e29764a0e66a7a31e9b8bf1de5cc0423ea72979f45909acd4292de834ccd3", size = 689324, upload-time = "2025-07-29T18:39:07.567Z" }, - { url = "https://files.pythonhosted.org/packages/09/f7/a82d249c711abf411ac027b7163f285487f5e615c3e0716c61033ce996ab/ml_dtypes-0.5.3-cp314-cp314t-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:19f6c3a4f635c2fc9e2aa7d91416bd7a3d649b48350c51f7f715a09370a90d93", size = 5275917, upload-time = "2025-07-29T18:39:09.339Z" }, - { url = "https://files.pythonhosted.org/packages/7f/3c/541c4b30815ab90ebfbb51df15d0b4254f2f9f1e2b4907ab229300d5e6f2/ml_dtypes-0.5.3-cp314-cp314t-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:5ab039ffb40f3dc0aeeeba84fd6c3452781b5e15bef72e2d10bcb33e4bbffc39", size = 5285284, upload-time = "2025-07-29T18:39:11.532Z" }, -] - [[package]] name = "more-itertools" version = "10.8.0" @@ -2376,7 +1871,7 @@ wheels = [ [[package]] name = "openai" -version = "1.105.0" +version = "2.16.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "anyio" }, @@ -2388,9 +1883,9 @@ dependencies = [ { name = "tqdm" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/6f/a9/c8c2dea8066a8f3079f69c242f7d0d75aaad4c4c3431da5b0df22a24e75d/openai-1.105.0.tar.gz", hash = "sha256:a68a47adce0506d34def22dd78a42cbb6cfecae1cf6a5fe37f38776d32bbb514", size = 557265, upload-time = "2025-09-03T14:14:08.586Z" } +sdist = { url = "https://files.pythonhosted.org/packages/b1/6c/e4c964fcf1d527fdf4739e7cc940c60075a4114d50d03871d5d5b1e13a88/openai-2.16.0.tar.gz", hash = "sha256:42eaa22ca0d8ded4367a77374104d7a2feafee5bd60a107c3c11b5243a11cd12", size = 629649, upload-time = "2026-01-27T23:28:02.579Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/51/01/186845829d3a3609bb5b474067959076244dd62540d3e336797319b13924/openai-1.105.0-py3-none-any.whl", hash = "sha256:3ad7635132b0705769ccae31ca7319f59ec0c7d09e94e5e713ce2d130e5b021f", size = 928203, upload-time = "2025-09-03T14:14:06.842Z" }, + { url = "https://files.pythonhosted.org/packages/16/83/0315bf2cfd75a2ce8a7e54188e9456c60cec6c0cf66728ed07bd9859ff26/openai-2.16.0-py3-none-any.whl", hash = "sha256:5f46643a8f42899a84e80c38838135d7038e7718333ce61396994f887b09a59b", size = 1068612, upload-time = "2026-01-27T23:28:00.356Z" }, ] [[package]] @@ -2444,32 +1939,32 @@ wheels = [ [[package]] name = "opentelemetry-api" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "importlib-metadata" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/27/d2/c782c88b8afbf961d6972428821c302bd1e9e7bc361352172f0ca31296e2/opentelemetry_api-1.36.0.tar.gz", hash = "sha256:9a72572b9c416d004d492cbc6e61962c0501eaf945ece9b5a0f56597d8348aa0", size = 64780, upload-time = "2025-07-29T15:12:06.02Z" } +sdist = { url = "https://files.pythonhosted.org/packages/c0/0b/e5428c009d4d9af0515b0a8371a8aaae695371af291f45e702f7969dce6b/opentelemetry_api-1.39.0.tar.gz", hash = "sha256:6130644268c5ac6bdffaf660ce878f10906b3e789f7e2daa5e169b047a2933b9", size = 65763, upload-time = "2025-12-03T13:19:56.378Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/bb/ee/6b08dde0a022c463b88f55ae81149584b125a42183407dc1045c486cc870/opentelemetry_api-1.36.0-py3-none-any.whl", hash = "sha256:02f20bcacf666e1333b6b1f04e647dc1d5111f86b8e510238fcc56d7762cda8c", size = 65564, upload-time = "2025-07-29T15:11:47.998Z" }, + { url = "https://files.pythonhosted.org/packages/05/85/d831a9bc0a9e0e1a304ff3d12c1489a5fbc9bf6690a15dcbdae372bbca45/opentelemetry_api-1.39.0-py3-none-any.whl", hash = "sha256:3c3b3ca5c5687b1b5b37e5c5027ff68eacea8675241b29f13110a8ffbb8f0459", size = 66357, upload-time = "2025-12-03T13:19:33.043Z" }, ] [[package]] name = "opentelemetry-exporter-otlp-proto-common" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-proto" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/34/da/7747e57eb341c59886052d733072bc878424bf20f1d8cf203d508bbece5b/opentelemetry_exporter_otlp_proto_common-1.36.0.tar.gz", hash = "sha256:6c496ccbcbe26b04653cecadd92f73659b814c6e3579af157d8716e5f9f25cbf", size = 20302, upload-time = "2025-07-29T15:12:07.71Z" } +sdist = { url = "https://files.pythonhosted.org/packages/11/cb/3a29ce606b10c76d413d6edd42d25a654af03e73e50696611e757d2602f3/opentelemetry_exporter_otlp_proto_common-1.39.0.tar.gz", hash = "sha256:a135fceed1a6d767f75be65bd2845da344dd8b9258eeed6bc48509d02b184409", size = 20407, upload-time = "2025-12-03T13:19:59.003Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d0/ed/22290dca7db78eb32e0101738366b5bbda00d0407f00feffb9bf8c3fdf87/opentelemetry_exporter_otlp_proto_common-1.36.0-py3-none-any.whl", hash = "sha256:0fc002a6ed63eac235ada9aa7056e5492e9a71728214a61745f6ad04b923f840", size = 18349, upload-time = "2025-07-29T15:11:51.327Z" }, + { url = "https://files.pythonhosted.org/packages/ef/c6/215edba62d13a3948c718b289539f70e40965bc37fc82ecd55bb0b749c1a/opentelemetry_exporter_otlp_proto_common-1.39.0-py3-none-any.whl", hash = "sha256:3d77be7c4bdf90f1a76666c934368b8abed730b5c6f0547a2ec57feb115849ac", size = 18367, upload-time = "2025-12-03T13:19:36.906Z" }, ] [[package]] name = "opentelemetry-exporter-otlp-proto-grpc" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "googleapis-common-protos" }, @@ -2480,14 +1975,14 @@ dependencies = [ { name = "opentelemetry-sdk" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/72/6f/6c1b0bdd0446e5532294d1d41bf11fbaea39c8a2423a4cdfe4fe6b708127/opentelemetry_exporter_otlp_proto_grpc-1.36.0.tar.gz", hash = "sha256:b281afbf7036b325b3588b5b6c8bb175069e3978d1bd24071f4a59d04c1e5bbf", size = 23822, upload-time = "2025-07-29T15:12:08.292Z" } +sdist = { url = "https://files.pythonhosted.org/packages/7e/62/4db083ee9620da3065eeb559e9fc128f41a1d15e7c48d7c83aafbccd354c/opentelemetry_exporter_otlp_proto_grpc-1.39.0.tar.gz", hash = "sha256:7e7bb3f436006836c0e0a42ac619097746ad5553ad7128a5bd4d3e727f37fc06", size = 24650, upload-time = "2025-12-03T13:20:00.06Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/0c/67/5f6bd188d66d0fd8e81e681bbf5822e53eb150034e2611dd2b935d3ab61a/opentelemetry_exporter_otlp_proto_grpc-1.36.0-py3-none-any.whl", hash = "sha256:734e841fc6a5d6f30e7be4d8053adb703c70ca80c562ae24e8083a28fadef211", size = 18828, upload-time = "2025-07-29T15:11:52.235Z" }, + { url = "https://files.pythonhosted.org/packages/56/e8/d420b94ffddfd8cff85bb4aa5d98da26ce7935dc3cf3eca6b83cd39ab436/opentelemetry_exporter_otlp_proto_grpc-1.39.0-py3-none-any.whl", hash = "sha256:758641278050de9bb895738f35ff8840e4a47685b7e6ef4a201fe83196ba7a05", size = 19765, upload-time = "2025-12-03T13:19:38.143Z" }, ] [[package]] name = "opentelemetry-exporter-otlp-proto-http" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "googleapis-common-protos" }, @@ -2498,14 +1993,14 @@ dependencies = [ { name = "requests" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/25/85/6632e7e5700ba1ce5b8a065315f92c1e6d787ccc4fb2bdab15139eaefc82/opentelemetry_exporter_otlp_proto_http-1.36.0.tar.gz", hash = "sha256:dd3637f72f774b9fc9608ab1ac479f8b44d09b6fb5b2f3df68a24ad1da7d356e", size = 16213, upload-time = "2025-07-29T15:12:08.932Z" } +sdist = { url = "https://files.pythonhosted.org/packages/81/dc/1e9bf3f6a28e29eba516bc0266e052996d02bc7e92675f3cd38169607609/opentelemetry_exporter_otlp_proto_http-1.39.0.tar.gz", hash = "sha256:28d78fc0eb82d5a71ae552263d5012fa3ebad18dfd189bf8d8095ba0e65ee1ed", size = 17287, upload-time = "2025-12-03T13:20:01.134Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/7f/41/a680d38b34f8f5ddbd78ed9f0042e1cc712d58ec7531924d71cb1e6c629d/opentelemetry_exporter_otlp_proto_http-1.36.0-py3-none-any.whl", hash = "sha256:3d769f68e2267e7abe4527f70deb6f598f40be3ea34c6adc35789bea94a32902", size = 18752, upload-time = "2025-07-29T15:11:53.164Z" }, + { url = "https://files.pythonhosted.org/packages/bc/46/e4a102e17205bb05a50dbf24ef0e92b66b648cd67db9a68865af06a242fd/opentelemetry_exporter_otlp_proto_http-1.39.0-py3-none-any.whl", hash = "sha256:5789cb1375a8b82653328c0ce13a054d285f774099faf9d068032a49de4c7862", size = 19639, upload-time = "2025-12-03T13:19:39.536Z" }, ] [[package]] name = "opentelemetry-instrumentation" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2513,14 +2008,14 @@ dependencies = [ { name = "packaging" }, { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/12/37/cf17cf28f945a3aca5a038cfbb45ee01317d4f7f3a0e5209920883fe9b08/opentelemetry_instrumentation-0.57b0.tar.gz", hash = "sha256:f2a30135ba77cdea2b0e1df272f4163c154e978f57214795d72f40befd4fcf05", size = 30807, upload-time = "2025-07-29T15:42:44.746Z" } +sdist = { url = "https://files.pythonhosted.org/packages/55/3c/bd53dbb42eff93d18e3047c7be11224aa9966ce98ac4cc5bfb860a32c95a/opentelemetry_instrumentation-0.60b0.tar.gz", hash = "sha256:4e9fec930f283a2677a2217754b40aaf9ef76edae40499c165bc7f1d15366a74", size = 31707, upload-time = "2025-12-03T13:22:00.352Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/d0/6f/f20cd1542959f43fb26a5bf9bb18cd81a1ea0700e8870c8f369bd07f5c65/opentelemetry_instrumentation-0.57b0-py3-none-any.whl", hash = "sha256:9109280f44882e07cec2850db28210b90600ae9110b42824d196de357cbddf7e", size = 32460, upload-time = "2025-07-29T15:41:40.883Z" }, + { url = "https://files.pythonhosted.org/packages/5c/7b/5b5b9f8cfe727a28553acf9cd287b1d7f706f5c0a00d6e482df55b169483/opentelemetry_instrumentation-0.60b0-py3-none-any.whl", hash = "sha256:aaafa1483543a402819f1bdfb06af721c87d60dd109501f9997332862a35c76a", size = 33096, upload-time = "2025-12-03T13:20:51.785Z" }, ] [[package]] name = "opentelemetry-instrumentation-asgi" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "asgiref" }, @@ -2529,14 +2024,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/97/10/7ba59b586eb099fa0155521b387d857de476687c670096597f618d889323/opentelemetry_instrumentation_asgi-0.57b0.tar.gz", hash = "sha256:a6f880b5d1838f65688fc992c65fbb1d3571f319d370990c32e759d3160e510b", size = 24654, upload-time = "2025-07-29T15:42:48.199Z" } +sdist = { url = "https://files.pythonhosted.org/packages/b0/0a/715ea7044708d3c215385fb2a1c6ffe429aacb3cd23a348060aaeda52834/opentelemetry_instrumentation_asgi-0.60b0.tar.gz", hash = "sha256:928731218050089dca69f0fe980b8bfe109f384be8b89802d7337372ddb67b91", size = 26083, upload-time = "2025-12-03T13:22:05.672Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/9e/07/ab97dd7e8bc680b479203f7d3b2771b7a097468135a669a38da3208f96cb/opentelemetry_instrumentation_asgi-0.57b0-py3-none-any.whl", hash = "sha256:47debbde6af066a7e8e911f7193730d5e40d62effc1ac2e1119908347790a3ea", size = 16599, upload-time = "2025-07-29T15:41:48.332Z" }, + { url = "https://files.pythonhosted.org/packages/9b/8c/c6c59127fd996107243ca45669355665a7daff578ddafb86d6d2d3b01428/opentelemetry_instrumentation_asgi-0.60b0-py3-none-any.whl", hash = "sha256:9d76a541269452c718a0384478f3291feb650c5a3f29e578fdc6613ea3729cf3", size = 16907, upload-time = "2025-12-03T13:20:58.962Z" }, ] [[package]] name = "opentelemetry-instrumentation-dbapi" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2544,14 +2039,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/15/dc/5a17b2fb593901ba5257278073b28d0ed31497e56985990c26046e4da2d9/opentelemetry_instrumentation_dbapi-0.57b0.tar.gz", hash = "sha256:7ad9e39c91f6212f118435fd6fab842a1f78b2cbad1167f228c025bba2a8fc2d", size = 14176, upload-time = "2025-07-29T15:42:56.249Z" } +sdist = { url = "https://files.pythonhosted.org/packages/12/7f/b4c1fbce01b29daad5ef1396427c9cd3c7a55ee68e75f8c11089c7e2533d/opentelemetry_instrumentation_dbapi-0.60b0.tar.gz", hash = "sha256:2b7eb38e46890cebe5bc1a1c03d2ab07fc159b0b7b91342941ee33dd73876d84", size = 16311, upload-time = "2025-12-03T13:22:15.369Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/2c/71/21a7e862dead70267b7c7bd5aa4e0b61fbc9fa9b4be57f4e183766abbad9/opentelemetry_instrumentation_dbapi-0.57b0-py3-none-any.whl", hash = "sha256:c1b110a5e86ec9b52b970460917523f47afa0c73f131e7f03c6a7c1921822dc4", size = 12466, upload-time = "2025-07-29T15:41:59.775Z" }, + { url = "https://files.pythonhosted.org/packages/23/0a/65e100c6d803de59a9113a993dcd371a4027453ba15ce4dabdb0343ca154/opentelemetry_instrumentation_dbapi-0.60b0-py3-none-any.whl", hash = "sha256:429d8ca34a44a4296b9b09a1bd373fff350998d200525c6e79883c3328559b03", size = 13966, upload-time = "2025-12-03T13:21:12.435Z" }, ] [[package]] name = "opentelemetry-instrumentation-django" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2560,14 +2055,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/5a/88/d88268c37aabbd2bcc54f4f868394316fa6fdfd3b91e011d229617d862d3/opentelemetry_instrumentation_django-0.57b0.tar.gz", hash = "sha256:df4116d2ea2c6bbbbf8853b843deb74d66bd0d573ddd372ec84fd60adaf977c6", size = 25005, upload-time = "2025-07-29T15:42:56.88Z" } +sdist = { url = "https://files.pythonhosted.org/packages/7c/d2/8ddd9a5c61cd5048d422be8d22fac40f603aa82f0babf9f7c40db871080c/opentelemetry_instrumentation_django-0.60b0.tar.gz", hash = "sha256:461e6fca27936ba97eec26da38bb5f19310783370478c7ca3a3e40faaceac9cc", size = 26596, upload-time = "2025-12-03T13:22:16.069Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/97/f0/1d5022f2fe16d50b79d9f1f5b70bd08d0e59819e0f6b237cff82c3dbda0f/opentelemetry_instrumentation_django-0.57b0-py3-none-any.whl", hash = "sha256:3d702d79a9ec0c836ccf733becf34630c6afb3c86c25c330c5b7601debe1e7c5", size = 19597, upload-time = "2025-07-29T15:42:00.657Z" }, + { url = "https://files.pythonhosted.org/packages/18/d6/28684547bf6c699582e998a172ba8bb08405cf6706729b0d6a16042e998f/opentelemetry_instrumentation_django-0.60b0-py3-none-any.whl", hash = "sha256:95495649c8c34ce9217c6873cdd10fc4fcaa67c25f8329adc54f5b286999e40b", size = 21169, upload-time = "2025-12-03T13:21:13.475Z" }, ] [[package]] name = "opentelemetry-instrumentation-fastapi" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2576,14 +2071,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/47/a8/7c22a33ff5986523a7f9afcb5f4d749533842c3cc77ef55b46727580edd0/opentelemetry_instrumentation_fastapi-0.57b0.tar.gz", hash = "sha256:73ac22f3c472a8f9cb21d1fbe5a4bf2797690c295fff4a1c040e9b1b1688a105", size = 20277, upload-time = "2025-07-29T15:42:58.68Z" } +sdist = { url = "https://files.pythonhosted.org/packages/fe/51/a021a7c929b5103fcb6bfdfa5a99abcaeb3b505faf9e3ee3ec14612c1ef9/opentelemetry_instrumentation_fastapi-0.60b0.tar.gz", hash = "sha256:5d34d67eb634a08bfe9e530680d6177521cd9da79285144e6d5a8f42683ed1b3", size = 24960, upload-time = "2025-12-03T13:22:18.468Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/3b/df/f20fc21c88c7af5311bfefc15fc4e606bab5edb7c193aa8c73c354904c35/opentelemetry_instrumentation_fastapi-0.57b0-py3-none-any.whl", hash = "sha256:61e6402749ffe0bfec582e58155e0d81dd38723cd9bc4562bca1acca80334006", size = 12712, upload-time = "2025-07-29T15:42:03.332Z" }, + { url = "https://files.pythonhosted.org/packages/b1/5a/e238c108eb65a726d75184439377a87d532050036b54e718e4c789b26d1a/opentelemetry_instrumentation_fastapi-0.60b0-py3-none-any.whl", hash = "sha256:415c6602db01ee339276ea4cabe3e80177c9e955631c087f2ef60a75e31bfaee", size = 13478, upload-time = "2025-12-03T13:21:16.804Z" }, ] [[package]] name = "opentelemetry-instrumentation-flask" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2593,9 +2088,9 @@ dependencies = [ { name = "opentelemetry-util-http" }, { name = "packaging" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/5f/98/8a8fa41f624069ac2912141b65bd528fd345d65e14a359c4d896fc3dc291/opentelemetry_instrumentation_flask-0.57b0.tar.gz", hash = "sha256:c5244a40b03664db966d844a32f43c900181431b77929be62a68d4907e86ed25", size = 19381, upload-time = "2025-07-29T15:42:59.38Z" } +sdist = { url = "https://files.pythonhosted.org/packages/30/cc/e0758c23d66fd49956169cb24b5b06130373da2ce8d49945abce82003518/opentelemetry_instrumentation_flask-0.60b0.tar.gz", hash = "sha256:560f08598ef40cdcf7ca05bfb2e3ea74fab076e676f4c18bb36bb379bf5c4a1b", size = 20336, upload-time = "2025-12-03T13:22:19.162Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/bb/3f/79b6c9a240221f5614a143eab6a0ecacdcb23b93cc35ff2b78234f68804f/opentelemetry_instrumentation_flask-0.57b0-py3-none-any.whl", hash = "sha256:5ecd614f194825725b61ee9ba8e37dcd4d3f9b5d40fef759df8650d6a91b1cb9", size = 14688, upload-time = "2025-07-29T15:42:04.162Z" }, + { url = "https://files.pythonhosted.org/packages/9b/b5/387ce11f59e5ce65b890adc3f9c457877143b8a6d107a3a0b305397933a1/opentelemetry_instrumentation_flask-0.60b0-py3-none-any.whl", hash = "sha256:106e5774f79ac9b86dd0d949c1b8f46c807a8af16184301e10d24fc94e680d04", size = 15189, upload-time = "2025-12-03T13:21:18.672Z" }, ] [[package]] @@ -2615,21 +2110,21 @@ wheels = [ [[package]] name = "opentelemetry-instrumentation-psycopg2" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, { name = "opentelemetry-instrumentation" }, { name = "opentelemetry-instrumentation-dbapi" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/a8/66/f2004cde131663810e62b47bb48b684660632876f120c6b1d400a04ccb06/opentelemetry_instrumentation_psycopg2-0.57b0.tar.gz", hash = "sha256:4e9d05d661c50985f0a5d7f090a7f399d453b467c9912c7611fcef693d15b038", size = 10722, upload-time = "2025-07-29T15:43:05.644Z" } +sdist = { url = "https://files.pythonhosted.org/packages/f8/68/5ae8a3b9a28c2fdf8d3d050e451ddb2612ca963679b08a2959f01f6dda4b/opentelemetry_instrumentation_psycopg2-0.60b0.tar.gz", hash = "sha256:59e527fd97739440380634ffcf9431aa7f2965d939d8d5829790886e2b54ede9", size = 11266, upload-time = "2025-12-03T13:22:26.025Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/02/40/00f9c1334fb0c9d74c99d37c4a730cbe6dc941eea5fae6f9bc36e5a53d19/opentelemetry_instrumentation_psycopg2-0.57b0-py3-none-any.whl", hash = "sha256:94fdde02b7451c8e85d43b4b9dd13a34fee96ffd43324d1b3567f47d2903b99f", size = 10721, upload-time = "2025-07-29T15:42:15.698Z" }, + { url = "https://files.pythonhosted.org/packages/d4/24/66b5a41a2b0d1d07cc9b0fbd80f8b5c66b46a4d4731743505891da8b3cbe/opentelemetry_instrumentation_psycopg2-0.60b0-py3-none-any.whl", hash = "sha256:ea136a32babd559aa717c04dddf6aa78aa94b816fb4e10dfe06751727ef306d4", size = 11284, upload-time = "2025-12-03T13:21:31.23Z" }, ] [[package]] name = "opentelemetry-instrumentation-requests" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2637,14 +2132,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/0d/e1/01f5c28a60ffbc4c04946ad35bc8bf16382d333e41afaa042b31c35364b9/opentelemetry_instrumentation_requests-0.57b0.tar.gz", hash = "sha256:193bd3fd1f14737721876fb1952dffc7d43795586118df633a91ecd9057446ff", size = 15182, upload-time = "2025-07-29T15:43:11.812Z" } +sdist = { url = "https://files.pythonhosted.org/packages/26/0f/94c6181e95c867f559715887c418170a9eadd92ea6090122d464e375ff56/opentelemetry_instrumentation_requests-0.60b0.tar.gz", hash = "sha256:5079ed8df96d01dab915a0766cd28a49be7c33439ce43d6d39843ed6dee3204f", size = 16173, upload-time = "2025-12-03T13:22:31.458Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b5/7d/40144701fa22521e3b3fce23e2f0a5684a9385c90b119b70e7598b3cb607/opentelemetry_instrumentation_requests-0.57b0-py3-none-any.whl", hash = "sha256:66a576ac8080724ddc8a14c39d16bb5f430991bd504fdbea844c7a063f555971", size = 12966, upload-time = "2025-07-29T15:42:24.608Z" }, + { url = "https://files.pythonhosted.org/packages/f1/e1/2f13b41c5679243ba8eae651170c4ce2f532349877819566ae4a89a2b47f/opentelemetry_instrumentation_requests-0.60b0-py3-none-any.whl", hash = "sha256:e9957f3a650ae55502fa227b29ff985b37d63e41c85e6e1555d48039f092ea83", size = 13122, upload-time = "2025-12-03T13:21:38.983Z" }, ] [[package]] name = "opentelemetry-instrumentation-urllib" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2652,14 +2147,14 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/86/a5/9d400dd978ac5e81356fe8435ca264e140a7d4cf77a88db43791d62311d5/opentelemetry_instrumentation_urllib-0.57b0.tar.gz", hash = "sha256:657225ceae8bb52b67bd5c26dcb8a33f0efb041f1baea4c59dbd1adbc63a4162", size = 13929, upload-time = "2025-07-29T15:43:16.498Z" } +sdist = { url = "https://files.pythonhosted.org/packages/95/db/be895de04bd56d7a2b2ef6d267a4c52f6cd325b6647d1c15ae888b1b0f6a/opentelemetry_instrumentation_urllib-0.60b0.tar.gz", hash = "sha256:89b8796f9ab64d0ea0833cfea98745963baa0d7e4a775b3d2a77791aa97cf3f9", size = 13931, upload-time = "2025-12-03T13:22:37.44Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/79/47/3c9535a68b9dd125eb6a25c086984e5cee7285e4f36bfa37eeb40e95d2b5/opentelemetry_instrumentation_urllib-0.57b0-py3-none-any.whl", hash = "sha256:bb3a01172109a6f56bfcc38ea83b9d4a61c4c2cac6b9a190e757063daadf545c", size = 12671, upload-time = "2025-07-29T15:42:34.561Z" }, + { url = "https://files.pythonhosted.org/packages/2b/e0/178914d5cec77baef797c6d47412da478ff871b05eb8732d64037b87c868/opentelemetry_instrumentation_urllib-0.60b0-py3-none-any.whl", hash = "sha256:80e3545d02505dc0ea61b3a0a141ec2828e11bee6b7dedfd3ee7ed9a7adbf862", size = 12673, upload-time = "2025-12-03T13:21:48.139Z" }, ] [[package]] name = "opentelemetry-instrumentation-urllib3" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2668,14 +2163,14 @@ dependencies = [ { name = "opentelemetry-util-http" }, { name = "wrapt" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/9a/2d/c241e9716c94704dbddf64e2c7367b57642425455befdbc622936bec78e9/opentelemetry_instrumentation_urllib3-0.57b0.tar.gz", hash = "sha256:f49d8c3d1d81ae56304a08b14a7f564d250733ed75cd2210ccef815b5af2eea1", size = 15790, upload-time = "2025-07-29T15:43:17.05Z" } +sdist = { url = "https://files.pythonhosted.org/packages/25/a8/16a32239e84741fae1a2932badeade5e72b73bfc331b53f7049a648ca00b/opentelemetry_instrumentation_urllib3-0.60b0.tar.gz", hash = "sha256:6ae1640a993901bae8eda5496d8b1440fb326a29e4ba1db342738b8868174aad", size = 15789, upload-time = "2025-12-03T13:22:38.073Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/06/0e/a5467ab57d815caa58cbabb3a7f3906c3718c599221ac770482d13187306/opentelemetry_instrumentation_urllib3-0.57b0-py3-none-any.whl", hash = "sha256:337ecac6df3ff92026b51c64df7dd4a3fff52f2dc96036ea9371670243bf83c6", size = 13186, upload-time = "2025-07-29T15:42:35.775Z" }, + { url = "https://files.pythonhosted.org/packages/16/b2/ca27479eaf1f3f4825481769eb0cb200cad839040b8d5f42662d0398a256/opentelemetry_instrumentation_urllib3-0.60b0-py3-none-any.whl", hash = "sha256:9a07504560feae650a9205b3e2a579a835819bb1d55498d26a5db477fe04bba0", size = 13187, upload-time = "2025-12-03T13:21:49.482Z" }, ] [[package]] name = "opentelemetry-instrumentation-wsgi" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, @@ -2683,21 +2178,21 @@ dependencies = [ { name = "opentelemetry-semantic-conventions" }, { name = "opentelemetry-util-http" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/f8/3f/d1ab49d68f2f6ebbe3c2fa5ff609ee5603a9cc68915203c454afb3a38d5b/opentelemetry_instrumentation_wsgi-0.57b0.tar.gz", hash = "sha256:d7e16b3b87930c30fc4c1bbc8b58c5dd6eefade493a3a5e7343bc24d572bc5b7", size = 18376, upload-time = "2025-07-29T15:43:17.683Z" } +sdist = { url = "https://files.pythonhosted.org/packages/10/ad/ae04e35f3b96d9c20d5d3df94a4c296eabf7a54d35d6c831179471128270/opentelemetry_instrumentation_wsgi-0.60b0.tar.gz", hash = "sha256:5815195b1b9890f55c4baafec94ff98591579a7d9b16256064adea8ee5784651", size = 19104, upload-time = "2025-12-03T13:22:38.733Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/1f/0c/7760f9e14f4f8128e4880b4fd5f232ef4eb00cb29ee560c972dbf7801369/opentelemetry_instrumentation_wsgi-0.57b0-py3-none-any.whl", hash = "sha256:b9cf0c6e61489f7503fc17ef04d169bd214e7a825650ee492f5d2b4d73b17b54", size = 14450, upload-time = "2025-07-29T15:42:37.351Z" }, + { url = "https://files.pythonhosted.org/packages/73/0e/1ed4d3cdce7b2e00a24f79933b3472e642d4db98aaccc09769be5cbe5296/opentelemetry_instrumentation_wsgi-0.60b0-py3-none-any.whl", hash = "sha256:0ff80614c1e73f7e94a5860c7e6222a51195eebab3dc5f50d89013db3d5d2f13", size = 14553, upload-time = "2025-12-03T13:21:50.491Z" }, ] [[package]] name = "opentelemetry-proto" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "protobuf" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/fd/02/f6556142301d136e3b7e95ab8ea6a5d9dc28d879a99f3dd673b5f97dca06/opentelemetry_proto-1.36.0.tar.gz", hash = "sha256:0f10b3c72f74c91e0764a5ec88fd8f1c368ea5d9c64639fb455e2854ef87dd2f", size = 46152, upload-time = "2025-07-29T15:12:15.717Z" } +sdist = { url = "https://files.pythonhosted.org/packages/48/b5/64d2f8c3393cd13ea2092106118f7b98461ba09333d40179a31444c6f176/opentelemetry_proto-1.39.0.tar.gz", hash = "sha256:c1fa48678ad1a1624258698e59be73f990b7fc1f39e73e16a9d08eef65dd838c", size = 46153, upload-time = "2025-12-03T13:20:08.729Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/b3/57/3361e06136225be8180e879199caea520f38026f8071366241ac458beb8d/opentelemetry_proto-1.36.0-py3-none-any.whl", hash = "sha256:151b3bf73a09f94afc658497cf77d45a565606f62ce0c17acb08cd9937ca206e", size = 72537, upload-time = "2025-07-29T15:12:02.243Z" }, + { url = "https://files.pythonhosted.org/packages/e3/4d/d500e1862beed68318705732d1976c390f4a72ca8009c4983ff627acff20/opentelemetry_proto-1.39.0-py3-none-any.whl", hash = "sha256:1e086552ac79acb501485ff0ce75533f70f3382d43d0a30728eeee594f7bf818", size = 72534, upload-time = "2025-12-03T13:19:50.251Z" }, ] [[package]] @@ -2714,29 +2209,29 @@ wheels = [ [[package]] name = "opentelemetry-sdk" -version = "1.36.0" +version = "1.39.0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, { name = "opentelemetry-semantic-conventions" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/4c/85/8567a966b85a2d3f971c4d42f781c305b2b91c043724fa08fd37d158e9dc/opentelemetry_sdk-1.36.0.tar.gz", hash = "sha256:19c8c81599f51b71670661ff7495c905d8fdf6976e41622d5245b791b06fa581", size = 162557, upload-time = "2025-07-29T15:12:16.76Z" } +sdist = { url = "https://files.pythonhosted.org/packages/51/e3/7cd989003e7cde72e0becfe830abff0df55c69d237ee7961a541e0167833/opentelemetry_sdk-1.39.0.tar.gz", hash = "sha256:c22204f12a0529e07aa4d985f1bca9d6b0e7b29fe7f03e923548ae52e0e15dde", size = 171322, upload-time = "2025-12-03T13:20:09.651Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/0b/59/7bed362ad1137ba5886dac8439e84cd2df6d087be7c09574ece47ae9b22c/opentelemetry_sdk-1.36.0-py3-none-any.whl", hash = "sha256:19fe048b42e98c5c1ffe85b569b7073576ad4ce0bcb6e9b4c6a39e890a6c45fb", size = 119995, upload-time = "2025-07-29T15:12:03.181Z" }, + { url = "https://files.pythonhosted.org/packages/a4/b4/2adc8bc83eb1055ecb592708efb6f0c520cc2eb68970b02b0f6ecda149cf/opentelemetry_sdk-1.39.0-py3-none-any.whl", hash = "sha256:90cfb07600dfc0d2de26120cebc0c8f27e69bf77cd80ef96645232372709a514", size = 132413, upload-time = "2025-12-03T13:19:51.364Z" }, ] [[package]] name = "opentelemetry-semantic-conventions" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } dependencies = [ { name = "opentelemetry-api" }, { name = "typing-extensions" }, ] -sdist = { url = "https://files.pythonhosted.org/packages/7e/31/67dfa252ee88476a29200b0255bda8dfc2cf07b56ad66dc9a6221f7dc787/opentelemetry_semantic_conventions-0.57b0.tar.gz", hash = "sha256:609a4a79c7891b4620d64c7aac6898f872d790d75f22019913a660756f27ff32", size = 124225, upload-time = "2025-07-29T15:12:17.873Z" } +sdist = { url = "https://files.pythonhosted.org/packages/71/0e/176a7844fe4e3cb5de604212094dffaed4e18b32f1c56b5258bcbcba85c2/opentelemetry_semantic_conventions-0.60b0.tar.gz", hash = "sha256:227d7aa73cbb8a2e418029d6b6465553aa01cf7e78ec9d0bc3255c7b3ac5bf8f", size = 137935, upload-time = "2025-12-03T13:20:12.395Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/05/75/7d591371c6c39c73de5ce5da5a2cc7b72d1d1cd3f8f4638f553c01c37b11/opentelemetry_semantic_conventions-0.57b0-py3-none-any.whl", hash = "sha256:757f7e76293294f124c827e514c2a3144f191ef175b069ce8d1211e1e38e9e78", size = 201627, upload-time = "2025-07-29T15:12:04.174Z" }, + { url = "https://files.pythonhosted.org/packages/d0/56/af0306666f91bae47db14d620775604688361f0f76a872e0005277311131/opentelemetry_semantic_conventions-0.60b0-py3-none-any.whl", hash = "sha256:069530852691136018087b52688857d97bba61cd641d0f8628d2d92788c4f78a", size = 219981, upload-time = "2025-12-03T13:19:53.585Z" }, ] [[package]] @@ -2750,11 +2245,11 @@ wheels = [ [[package]] name = "opentelemetry-util-http" -version = "0.57b0" +version = "0.60b0" source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/9b/1b/6229c45445e08e798fa825f5376f6d6a4211d29052a4088eed6d577fa653/opentelemetry_util_http-0.57b0.tar.gz", hash = "sha256:f7417595ead0eb42ed1863ec9b2f839fc740368cd7bbbfc1d0a47bc1ab0aba11", size = 9405, upload-time = "2025-07-29T15:43:19.916Z" } +sdist = { url = "https://files.pythonhosted.org/packages/38/0d/786a713445cf338131fef3a84fab1378e4b2ef3c3ea348eeb0c915eb804a/opentelemetry_util_http-0.60b0.tar.gz", hash = "sha256:e42b7bb49bba43b6f34390327d97e5016eb1c47949ceaf37c4795472a4e3a82d", size = 10576, upload-time = "2025-12-03T13:22:41.224Z" } wheels = [ - { url = "https://files.pythonhosted.org/packages/0b/a6/b98d508d189b9c208f5978d0906141747d7e6df7c7cafec03657ed1ed559/opentelemetry_util_http-0.57b0-py3-none-any.whl", hash = "sha256:e54c0df5543951e471c3d694f85474977cd5765a3b7654398c83bab3d2ffb8e9", size = 7643, upload-time = "2025-07-29T15:42:41.744Z" }, + { url = "https://files.pythonhosted.org/packages/53/5d/a448862f6d10c95685ed0e703596b6bd1784074e7ad90bffdc550abb7b68/opentelemetry_util_http-0.60b0-py3-none-any.whl", hash = "sha256:4f366f1a48adb74ffa6f80aee26f96882e767e01b03cd1cfb948b6e1020341fe", size = 8742, upload-time = "2025-12-03T13:21:54.553Z" }, ] [[package]] @@ -2868,44 +2363,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/54/20/4d324d65cc6d9205fabedc306948156824eb9f0ee1633355a8f7ec5c66bf/pluggy-1.6.0-py3-none-any.whl", hash = "sha256:e920276dd6813095e9377c0bc5566d94c932c33b27a3e3945d8389c374dd4746", size = 20538, upload-time = "2025-05-15T12:30:06.134Z" }, ] -[[package]] -name = "ply" -version = "3.11" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/e5/69/882ee5c9d017149285cab114ebeab373308ef0f874fcdac9beb90e0ac4da/ply-3.11.tar.gz", hash = "sha256:00c7c1aaa88358b9c765b6d3000c6eec0ba42abca5351b095321aef446081da3", size = 159130, upload-time = "2018-02-15T19:01:31.097Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/a3/58/35da89ee790598a0700ea49b2a66594140f44dec458c07e8e3d4979137fc/ply-3.11-py2.py3-none-any.whl", hash = "sha256:096f9b8350b65ebd2fd1346b12452efe5b9607f7482813ffca50c22722a807ce", size = 49567, upload-time = "2018-02-15T19:01:27.172Z" }, -] - -[[package]] -name = "portalocker" -version = "3.2.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pywin32", marker = "sys_platform == 'win32'" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/5e/77/65b857a69ed876e1951e88aaba60f5ce6120c33703f7cb61a3c894b8c1b6/portalocker-3.2.0.tar.gz", hash = "sha256:1f3002956a54a8c3730586c5c77bf18fae4149e07eaf1c29fc3faf4d5a3f89ac", size = 95644, upload-time = "2025-06-14T13:20:40.03Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/4b/a6/38c8e2f318bf67d338f4d629e93b0b4b9af331f455f0390ea8ce4a099b26/portalocker-3.2.0-py3-none-any.whl", hash = "sha256:3cdc5f565312224bc570c49337bd21428bba0ef363bbcf58b9ef4a9f11779968", size = 22424, upload-time = "2025-06-14T13:20:38.083Z" }, -] - -[[package]] -name = "posthog" -version = "6.9.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "backoff" }, - { name = "distro" }, - { name = "python-dateutil" }, - { name = "requests" }, - { name = "six" }, - { name = "typing-extensions" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/36/5e/137aaf1d45cc6fafa5573d24dfae795ceae75fdf3232d298828f2e54d688/posthog-6.9.1.tar.gz", hash = "sha256:0bf1115261369b76e2f643d04805cec434236f23fb69972ed5d1bd49b5a9a6fe", size = 126229, upload-time = "2025-11-07T15:57:26.347Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/91/72/ad1961cc3423f679bceb6c098ec67c5db7ab55dbafc71c5a4faf4ec99d68/posthog-6.9.1-py3-none-any.whl", hash = "sha256:a8e33fef54275c32077afea4b2a0e2ca554b226b63d6fcd319447c81154faa1f", size = 144481, upload-time = "2025-11-07T15:57:25.183Z" }, -] - [[package]] name = "prance" version = "25.4.8.0" @@ -3020,18 +2477,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/5b/5a/bc7b4a4ef808fa59a816c17b20c4bef6884daebbdf627ff2a161da67da19/propcache-0.4.1-py3-none-any.whl", hash = "sha256:af2a6052aeb6cf17d3e46ee169099044fd8224cbaf75c76a2ef596e8163e2237", size = 13305, upload-time = "2025-10-08T19:49:00.792Z" }, ] -[[package]] -name = "proto-plus" -version = "1.26.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "protobuf" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/f4/ac/87285f15f7cce6d4a008f33f1757fb5a13611ea8914eb58c3d0d26243468/proto_plus-1.26.1.tar.gz", hash = "sha256:21a515a4c4c0088a773899e23c7bbade3d18f9c66c73edd4c7ee3816bc96a012", size = 56142, upload-time = "2025-03-10T15:54:38.843Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/4e/6d/280c4c2ce28b1593a19ad5239c8b826871fc6ec275c21afc8e1820108039/proto_plus-1.26.1-py3-none-any.whl", hash = "sha256:13285478c2dcf2abb829db158e1047e2f1e8d63a077d94263c2b88b043c75a66", size = 50163, upload-time = "2025-03-10T15:54:37.335Z" }, -] - [[package]] name = "protobuf" version = "5.29.5" @@ -3081,27 +2526,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/22/a6/858897256d0deac81a172289110f31629fc4cee19b6f01283303e18c8db3/ptyprocess-0.7.0-py2.py3-none-any.whl", hash = "sha256:4b41f3967fce3af57cc7e94b888626c18bf37a083e3651ca8feeb66d492fef35", size = 13993, upload-time = "2020-12-28T15:15:28.35Z" }, ] -[[package]] -name = "pyasn1" -version = "0.6.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/ba/e9/01f1a64245b89f039897cb0130016d79f77d52669aae6ee7b159a6c4c018/pyasn1-0.6.1.tar.gz", hash = "sha256:6f580d2bdd84365380830acf45550f2511469f673cb4a5ae3857a3170128b034", size = 145322, upload-time = "2024-09-10T22:41:42.55Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c8/f1/d6a797abb14f6283c0ddff96bbdd46937f64122b8c925cab503dd37f8214/pyasn1-0.6.1-py3-none-any.whl", hash = "sha256:0d632f46f2ba09143da3a8afe9e33fb6f92fa2320ab7e886e2d0f7672af84629", size = 83135, upload-time = "2024-09-11T16:00:36.122Z" }, -] - -[[package]] -name = "pyasn1-modules" -version = "0.4.2" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pyasn1" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/e9/e6/78ebbb10a8c8e4b61a59249394a4a594c1a7af95593dc933a349c8d00964/pyasn1_modules-0.4.2.tar.gz", hash = "sha256:677091de870a80aae844b1ca6134f54652fa2c8c5a52aa396440ac3106e941e6", size = 307892, upload-time = "2025-03-28T02:41:22.17Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/47/8d/d529b5d697919ba8c11ad626e835d4039be708a35b0d22de83a269a6682c/pyasn1_modules-0.4.2-py3-none-any.whl", hash = "sha256:29253a9207ce32b64c3ac6600edc75368f98473906e8fd1043bd6b5b1de2c14a", size = 181259, upload-time = "2025-03-28T02:41:19.028Z" }, -] - [[package]] name = "pybars4" version = "0.9.13" @@ -3404,15 +2828,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/45/58/38b5afbc1a800eeea951b9285d3912613f2603bdf897a4ab0f4bd7f405fc/python_multipart-0.0.20-py3-none-any.whl", hash = "sha256:8a62d3a8335e06589fe01f2a3e178cdcc632f3fbe0d492ad9ee0ec35aab1f104", size = 24546, upload-time = "2024-12-16T19:45:44.423Z" }, ] -[[package]] -name = "python-ulid" -version = "3.1.0" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/40/7e/0d6c82b5ccc71e7c833aed43d9e8468e1f2ff0be1b3f657a6fcafbb8433d/python_ulid-3.1.0.tar.gz", hash = "sha256:ff0410a598bc5f6b01b602851a3296ede6f91389f913a5d5f8c496003836f636", size = 93175, upload-time = "2025-08-18T16:09:26.305Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/6c/a0/4ed6632b70a52de845df056654162acdebaf97c20e3212c559ac43e7216e/python_ulid-3.1.0-py3-none-any.whl", hash = "sha256:e2cdc979c8c877029b4b7a38a6fba3bc4578e4f109a308419ff4d3ccf0a46619", size = 11577, upload-time = "2025-08-18T16:09:25.047Z" }, -] - [[package]] name = "pytz" version = "2025.2" @@ -3496,55 +2911,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/f1/12/de94a39c2ef588c7e6455cfbe7343d3b2dc9d6b6b2f40c4c6565744c873d/pyyaml-6.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:ebc55a14a21cb14062aa4162f906cd962b28e2e9ea38f9b4391244cd8de4ae0b", size = 149341, upload-time = "2025-09-25T21:32:56.828Z" }, ] -[[package]] -name = "qdrant-client" -version = "1.15.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "grpcio" }, - { name = "httpx", extra = ["http2"] }, - { name = "numpy" }, - { name = "portalocker" }, - { name = "protobuf" }, - { name = "pydantic" }, - { name = "urllib3" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/79/8b/76c7d325e11d97cb8eb5e261c3759e9ed6664735afbf32fdded5b580690c/qdrant_client-1.15.1.tar.gz", hash = "sha256:631f1f3caebfad0fd0c1fba98f41be81d9962b7bf3ca653bed3b727c0e0cbe0e", size = 295297, upload-time = "2025-07-31T19:35:19.627Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/ef/33/d8df6a2b214ffbe4138db9a1efe3248f67dc3c671f82308bea1582ecbbb7/qdrant_client-1.15.1-py3-none-any.whl", hash = "sha256:2b975099b378382f6ca1cfb43f0d59e541be6e16a5892f282a4b8de7eff5cb63", size = 337331, upload-time = "2025-07-31T19:35:17.539Z" }, -] - -[[package]] -name = "redis" -version = "6.4.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "async-timeout", marker = "python_full_version < '3.11.3'" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/0d/d6/e8b92798a5bd67d659d51a18170e91c16ac3b59738d91894651ee255ed49/redis-6.4.0.tar.gz", hash = "sha256:b01bc7282b8444e28ec36b261df5375183bb47a07eb9c603f284e89cbc5ef010", size = 4647399, upload-time = "2025-08-07T08:10:11.441Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/e8/02/89e2ed7e85db6c93dfa9e8f691c5087df4e3551ab39081a4d7c6d1f90e05/redis-6.4.0-py3-none-any.whl", hash = "sha256:f0544fa9604264e9464cdf4814e7d4830f74b165d52f2a330a760a88dd248b7f", size = 279847, upload-time = "2025-08-07T08:10:09.84Z" }, -] - -[[package]] -name = "redisvl" -version = "0.11.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "jsonpath-ng" }, - { name = "ml-dtypes" }, - { name = "numpy" }, - { name = "pydantic" }, - { name = "python-ulid" }, - { name = "pyyaml" }, - { name = "redis" }, - { name = "tenacity" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/f5/dc/72f69eca73c31d6df705ba8a2c25a541248f34d1bd03dd9baef6d9e14fce/redisvl-0.11.0.tar.gz", hash = "sha256:8bd52e059a805756160320f547b04372fe00517596364431f813107d96c6cbf8", size = 670173, upload-time = "2025-11-07T23:55:47.566Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/36/cc/db92f58766f1dfc0472961044d94c755430afa2312967ab8eb411660414c/redisvl-0.11.0-py3-none-any.whl", hash = "sha256:7e2029fd5fc73baf5f024415002d91cdce88168e51113afc1dbc4fcd0f8a210a", size = 172269, upload-time = "2025-11-07T23:55:45.831Z" }, -] - [[package]] name = "referencing" version = "0.36.2" @@ -3799,18 +3165,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/ed/d2/4a73b18821fd4669762c855fd1f4e80ceb66fb72d71162d14da58444a763/rpds_py-0.28.0-pp311-pypy311_pp73-musllinux_1_2_x86_64.whl", hash = "sha256:5d0145edba8abd3db0ab22b5300c99dc152f5c9021fab861be0f0544dc3cbc5f", size = 552199, upload-time = "2025-10-22T22:24:26.54Z" }, ] -[[package]] -name = "rsa" -version = "4.9.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "pyasn1" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/da/8a/22b7beea3ee0d44b1916c0c1cb0ee3af23b700b6da9f04991899d0c555d4/rsa-4.9.1.tar.gz", hash = "sha256:e7bdbfdb5497da4c07dfd35530e1a902659db6ff241e39d9953cad06ebd0ae75", size = 29034, upload-time = "2025-04-16T09:51:18.218Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/64/8d/0133e4eb4beed9e425d9a98ed6e081a55d195481b7632472be1af08d2f6b/rsa-4.9.1-py3-none-any.whl", hash = "sha256:68635866661c6836b8d39430f97a996acbd61bfa49406748ea243539fe239762", size = 34696, upload-time = "2025-04-16T09:51:17.142Z" }, -] - [[package]] name = "ruamel-yaml" version = "0.18.16" @@ -3863,6 +3217,8 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/6b/fa/3234f913fe9a6525a7b97c6dad1f51e72b917e6872e051a5e2ffd8b16fbb/ruamel.yaml.clib-0.2.14-cp314-cp314-macosx_15_0_arm64.whl", hash = "sha256:70eda7703b8126f5e52fcf276e6c0f40b0d314674f896fc58c47b0aef2b9ae83", size = 137970, upload-time = "2025-09-22T19:51:09.472Z" }, { url = "https://files.pythonhosted.org/packages/ef/ec/4edbf17ac2c87fa0845dd366ef8d5852b96eb58fcd65fc1ecf5fe27b4641/ruamel.yaml.clib-0.2.14-cp314-cp314-musllinux_1_2_i686.whl", hash = "sha256:a0cb71ccc6ef9ce36eecb6272c81afdc2f565950cdcec33ae8e6cd8f7fc86f27", size = 739639, upload-time = "2025-09-22T19:51:10.566Z" }, { url = "https://files.pythonhosted.org/packages/15/18/b0e1fafe59051de9e79cdd431863b03593ecfa8341c110affad7c8121efc/ruamel.yaml.clib-0.2.14-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:e7cb9ad1d525d40f7d87b6df7c0ff916a66bc52cb61b66ac1b2a16d0c1b07640", size = 764456, upload-time = "2025-09-22T19:51:11.736Z" }, + { url = "https://files.pythonhosted.org/packages/e7/cd/150fdb96b8fab27fe08d8a59fe67554568727981806e6bc2677a16081ec7/ruamel_yaml_clib-0.2.14-cp314-cp314-win32.whl", hash = "sha256:9b4104bf43ca0cd4e6f738cb86326a3b2f6eef00f417bd1e7efb7bdffe74c539", size = 102394, upload-time = "2025-11-14T21:57:36.703Z" }, + { url = "https://files.pythonhosted.org/packages/bd/e6/a3fa40084558c7e1dc9546385f22a93949c890a8b2e445b2ba43935f51da/ruamel_yaml_clib-0.2.14-cp314-cp314-win_amd64.whl", hash = "sha256:13997d7d354a9890ea1ec5937a219817464e5cc344805b37671562a401ca3008", size = 122673, upload-time = "2025-11-14T21:57:38.177Z" }, ] [[package]] @@ -3987,43 +3343,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235, upload-time = "2024-02-25T23:20:01.196Z" }, ] -[[package]] -name = "sqlalchemy" -version = "2.0.44" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "greenlet", marker = "platform_machine == 'AMD64' or platform_machine == 'WIN32' or platform_machine == 'aarch64' or platform_machine == 'amd64' or platform_machine == 'ppc64le' or platform_machine == 'win32' or platform_machine == 'x86_64'" }, - { name = "typing-extensions" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/f0/f2/840d7b9496825333f532d2e3976b8eadbf52034178aac53630d09fe6e1ef/sqlalchemy-2.0.44.tar.gz", hash = "sha256:0ae7454e1ab1d780aee69fd2aae7d6b8670a581d8847f2d1e0f7ddfbf47e5a22", size = 9819830, upload-time = "2025-10-10T14:39:12.935Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/e3/81/15d7c161c9ddf0900b076b55345872ed04ff1ed6a0666e5e94ab44b0163c/sqlalchemy-2.0.44-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0fe3917059c7ab2ee3f35e77757062b1bea10a0b6ca633c58391e3f3c6c488dd", size = 2140517, upload-time = "2025-10-10T15:36:15.64Z" }, - { url = "https://files.pythonhosted.org/packages/d4/d5/4abd13b245c7d91bdf131d4916fd9e96a584dac74215f8b5bc945206a974/sqlalchemy-2.0.44-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:de4387a354ff230bc979b46b2207af841dc8bf29847b6c7dbe60af186d97aefa", size = 2130738, upload-time = "2025-10-10T15:36:16.91Z" }, - { url = "https://files.pythonhosted.org/packages/cb/3c/8418969879c26522019c1025171cefbb2a8586b6789ea13254ac602986c0/sqlalchemy-2.0.44-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c3678a0fb72c8a6a29422b2732fe423db3ce119c34421b5f9955873eb9b62c1e", size = 3304145, upload-time = "2025-10-10T15:34:19.569Z" }, - { url = "https://files.pythonhosted.org/packages/94/2d/fdb9246d9d32518bda5d90f4b65030b9bf403a935cfe4c36a474846517cb/sqlalchemy-2.0.44-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3cf6872a23601672d61a68f390e44703442639a12ee9dd5a88bbce52a695e46e", size = 3304511, upload-time = "2025-10-10T15:47:05.088Z" }, - { url = "https://files.pythonhosted.org/packages/7d/fb/40f2ad1da97d5c83f6c1269664678293d3fe28e90ad17a1093b735420549/sqlalchemy-2.0.44-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:329aa42d1be9929603f406186630135be1e7a42569540577ba2c69952b7cf399", size = 3235161, upload-time = "2025-10-10T15:34:21.193Z" }, - { url = "https://files.pythonhosted.org/packages/95/cb/7cf4078b46752dca917d18cf31910d4eff6076e5b513c2d66100c4293d83/sqlalchemy-2.0.44-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:70e03833faca7166e6a9927fbee7c27e6ecde436774cd0b24bbcc96353bce06b", size = 3261426, upload-time = "2025-10-10T15:47:07.196Z" }, - { url = "https://files.pythonhosted.org/packages/f8/3b/55c09b285cb2d55bdfa711e778bdffdd0dc3ffa052b0af41f1c5d6e582fa/sqlalchemy-2.0.44-cp311-cp311-win32.whl", hash = "sha256:253e2f29843fb303eca6b2fc645aca91fa7aa0aa70b38b6950da92d44ff267f3", size = 2105392, upload-time = "2025-10-10T15:38:20.051Z" }, - { url = "https://files.pythonhosted.org/packages/c7/23/907193c2f4d680aedbfbdf7bf24c13925e3c7c292e813326c1b84a0b878e/sqlalchemy-2.0.44-cp311-cp311-win_amd64.whl", hash = "sha256:7a8694107eb4308a13b425ca8c0e67112f8134c846b6e1f722698708741215d5", size = 2130293, upload-time = "2025-10-10T15:38:21.601Z" }, - { url = "https://files.pythonhosted.org/packages/62/c4/59c7c9b068e6813c898b771204aad36683c96318ed12d4233e1b18762164/sqlalchemy-2.0.44-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:72fea91746b5890f9e5e0997f16cbf3d53550580d76355ba2d998311b17b2250", size = 2139675, upload-time = "2025-10-10T16:03:31.064Z" }, - { url = "https://files.pythonhosted.org/packages/d6/ae/eeb0920537a6f9c5a3708e4a5fc55af25900216bdb4847ec29cfddf3bf3a/sqlalchemy-2.0.44-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:585c0c852a891450edbb1eaca8648408a3cc125f18cf433941fa6babcc359e29", size = 2127726, upload-time = "2025-10-10T16:03:35.934Z" }, - { url = "https://files.pythonhosted.org/packages/d8/d5/2ebbabe0379418eda8041c06b0b551f213576bfe4c2f09d77c06c07c8cc5/sqlalchemy-2.0.44-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9b94843a102efa9ac68a7a30cd46df3ff1ed9c658100d30a725d10d9c60a2f44", size = 3327603, upload-time = "2025-10-10T15:35:28.322Z" }, - { url = "https://files.pythonhosted.org/packages/45/e5/5aa65852dadc24b7d8ae75b7efb8d19303ed6ac93482e60c44a585930ea5/sqlalchemy-2.0.44-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:119dc41e7a7defcefc57189cfa0e61b1bf9c228211aba432b53fb71ef367fda1", size = 3337842, upload-time = "2025-10-10T15:43:45.431Z" }, - { url = "https://files.pythonhosted.org/packages/41/92/648f1afd3f20b71e880ca797a960f638d39d243e233a7082c93093c22378/sqlalchemy-2.0.44-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:0765e318ee9179b3718c4fd7ba35c434f4dd20332fbc6857a5e8df17719c24d7", size = 3264558, upload-time = "2025-10-10T15:35:29.93Z" }, - { url = "https://files.pythonhosted.org/packages/40/cf/e27d7ee61a10f74b17740918e23cbc5bc62011b48282170dc4c66da8ec0f/sqlalchemy-2.0.44-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:2e7b5b079055e02d06a4308d0481658e4f06bc7ef211567edc8f7d5dce52018d", size = 3301570, upload-time = "2025-10-10T15:43:48.407Z" }, - { url = "https://files.pythonhosted.org/packages/3b/3d/3116a9a7b63e780fb402799b6da227435be878b6846b192f076d2f838654/sqlalchemy-2.0.44-cp312-cp312-win32.whl", hash = "sha256:846541e58b9a81cce7dee8329f352c318de25aa2f2bbe1e31587eb1f057448b4", size = 2103447, upload-time = "2025-10-10T15:03:21.678Z" }, - { url = "https://files.pythonhosted.org/packages/25/83/24690e9dfc241e6ab062df82cc0df7f4231c79ba98b273fa496fb3dd78ed/sqlalchemy-2.0.44-cp312-cp312-win_amd64.whl", hash = "sha256:7cbcb47fd66ab294703e1644f78971f6f2f1126424d2b300678f419aa73c7b6e", size = 2130912, upload-time = "2025-10-10T15:03:24.656Z" }, - { url = "https://files.pythonhosted.org/packages/45/d3/c67077a2249fdb455246e6853166360054c331db4613cda3e31ab1cadbef/sqlalchemy-2.0.44-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:ff486e183d151e51b1d694c7aa1695747599bb00b9f5f604092b54b74c64a8e1", size = 2135479, upload-time = "2025-10-10T16:03:37.671Z" }, - { url = "https://files.pythonhosted.org/packages/2b/91/eabd0688330d6fd114f5f12c4f89b0d02929f525e6bf7ff80aa17ca802af/sqlalchemy-2.0.44-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:0b1af8392eb27b372ddb783b317dea0f650241cea5bd29199b22235299ca2e45", size = 2123212, upload-time = "2025-10-10T16:03:41.755Z" }, - { url = "https://files.pythonhosted.org/packages/b0/bb/43e246cfe0e81c018076a16036d9b548c4cc649de241fa27d8d9ca6f85ab/sqlalchemy-2.0.44-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2b61188657e3a2b9ac4e8f04d6cf8e51046e28175f79464c67f2fd35bceb0976", size = 3255353, upload-time = "2025-10-10T15:35:31.221Z" }, - { url = "https://files.pythonhosted.org/packages/b9/96/c6105ed9a880abe346b64d3b6ddef269ddfcab04f7f3d90a0bf3c5a88e82/sqlalchemy-2.0.44-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b87e7b91a5d5973dda5f00cd61ef72ad75a1db73a386b62877d4875a8840959c", size = 3260222, upload-time = "2025-10-10T15:43:50.124Z" }, - { url = "https://files.pythonhosted.org/packages/44/16/1857e35a47155b5ad927272fee81ae49d398959cb749edca6eaa399b582f/sqlalchemy-2.0.44-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:15f3326f7f0b2bfe406ee562e17f43f36e16167af99c4c0df61db668de20002d", size = 3189614, upload-time = "2025-10-10T15:35:32.578Z" }, - { url = "https://files.pythonhosted.org/packages/88/ee/4afb39a8ee4fc786e2d716c20ab87b5b1fb33d4ac4129a1aaa574ae8a585/sqlalchemy-2.0.44-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:1e77faf6ff919aa8cd63f1c4e561cac1d9a454a191bb864d5dd5e545935e5a40", size = 3226248, upload-time = "2025-10-10T15:43:51.862Z" }, - { url = "https://files.pythonhosted.org/packages/32/d5/0e66097fc64fa266f29a7963296b40a80d6a997b7ac13806183700676f86/sqlalchemy-2.0.44-cp313-cp313-win32.whl", hash = "sha256:ee51625c2d51f8baadf2829fae817ad0b66b140573939dd69284d2ba3553ae73", size = 2101275, upload-time = "2025-10-10T15:03:26.096Z" }, - { url = "https://files.pythonhosted.org/packages/03/51/665617fe4f8c6450f42a6d8d69243f9420f5677395572c2fe9d21b493b7b/sqlalchemy-2.0.44-cp313-cp313-win_amd64.whl", hash = "sha256:c1c80faaee1a6c3428cecf40d16a2365bcf56c424c92c2b6f0f9ad204b899e9e", size = 2127901, upload-time = "2025-10-10T15:03:27.548Z" }, - { url = "https://files.pythonhosted.org/packages/9c/5e/6a29fa884d9fb7ddadf6b69490a9d45fded3b38541713010dad16b77d015/sqlalchemy-2.0.44-py3-none-any.whl", hash = "sha256:19de7ca1246fbef9f9d1bff8f1ab25641569df226364a0e40457dc5457c54b05", size = 1928718, upload-time = "2025-10-10T15:29:45.32Z" }, -] - [[package]] name = "sse-starlette" version = "3.0.3" @@ -4049,15 +3368,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/ce/fd/901cfa59aaa5b30a99e16876f11abe38b59a1a2c51ffb3d7142bb6089069/starlette-0.47.3-py3-none-any.whl", hash = "sha256:89c0778ca62a76b826101e7c709e70680a1699ca7da6b44d38eb0a7e61fe4b51", size = 72991, upload-time = "2025-08-24T13:36:40.887Z" }, ] -[[package]] -name = "tenacity" -version = "9.1.2" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/0a/d4/2b0cd0fe285e14b36db076e78c93766ff1d529d70408bd1d2a5a84f1d929/tenacity-9.1.2.tar.gz", hash = "sha256:1169d376c297e7de388d18b4481760d478b0e99a777cad3a9c86e556f4b697cb", size = 48036, upload-time = "2025-04-02T08:25:09.966Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/e5/30/643397144bfbfec6f6ef821f36f33e57d35946c44a2352d3c9f0ae847619/tenacity-9.1.2-py3-none-any.whl", hash = "sha256:f77bf36710d8b73a50b2dd155c97b870017ad21afe6ab300326b0371b3b05138", size = 28248, upload-time = "2025-04-02T08:25:07.678Z" }, -] - [[package]] name = "tomli" version = "2.3.0" @@ -4180,142 +3490,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/d2/e2/dc81b1bd1dcfe91735810265e9d26bc8ec5da45b4c0f6237e286819194c3/uvicorn-0.35.0-py3-none-any.whl", hash = "sha256:197535216b25ff9b785e29a0b79199f55222193d47f820816e7da751e9bc8d4a", size = 66406, upload-time = "2025-06-28T16:15:44.816Z" }, ] -[package.optional-dependencies] -standard = [ - { name = "colorama", marker = "sys_platform == 'win32'" }, - { name = "httptools" }, - { name = "python-dotenv" }, - { name = "pyyaml" }, - { name = "uvloop", marker = "platform_python_implementation != 'PyPy' and sys_platform != 'cygwin' and sys_platform != 'win32'" }, - { name = "watchfiles" }, - { name = "websockets" }, -] - -[[package]] -name = "uvloop" -version = "0.22.1" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/06/f0/18d39dbd1971d6d62c4629cc7fa67f74821b0dc1f5a77af43719de7936a7/uvloop-0.22.1.tar.gz", hash = "sha256:6c84bae345b9147082b17371e3dd5d42775bddce91f885499017f4607fdaf39f", size = 2443250, upload-time = "2025-10-16T22:17:19.342Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/c7/d5/69900f7883235562f1f50d8184bb7dd84a2fb61e9ec63f3782546fdbd057/uvloop-0.22.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:c60ebcd36f7b240b30788554b6f0782454826a0ed765d8430652621b5de674b9", size = 1352420, upload-time = "2025-10-16T22:16:21.187Z" }, - { url = "https://files.pythonhosted.org/packages/a8/73/c4e271b3bce59724e291465cc936c37758886a4868787da0278b3b56b905/uvloop-0.22.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3b7f102bf3cb1995cfeaee9321105e8f5da76fdb104cdad8986f85461a1b7b77", size = 748677, upload-time = "2025-10-16T22:16:22.558Z" }, - { url = "https://files.pythonhosted.org/packages/86/94/9fb7fad2f824d25f8ecac0d70b94d0d48107ad5ece03769a9c543444f78a/uvloop-0.22.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:53c85520781d84a4b8b230e24a5af5b0778efdb39142b424990ff1ef7c48ba21", size = 3753819, upload-time = "2025-10-16T22:16:23.903Z" }, - { url = "https://files.pythonhosted.org/packages/74/4f/256aca690709e9b008b7108bc85fba619a2bc37c6d80743d18abad16ee09/uvloop-0.22.1-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:56a2d1fae65fd82197cb8c53c367310b3eabe1bbb9fb5a04d28e3e3520e4f702", size = 3804529, upload-time = "2025-10-16T22:16:25.246Z" }, - { url = "https://files.pythonhosted.org/packages/7f/74/03c05ae4737e871923d21a76fe28b6aad57f5c03b6e6bfcfa5ad616013e4/uvloop-0.22.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:40631b049d5972c6755b06d0bfe8233b1bd9a8a6392d9d1c45c10b6f9e9b2733", size = 3621267, upload-time = "2025-10-16T22:16:26.819Z" }, - { url = "https://files.pythonhosted.org/packages/75/be/f8e590fe61d18b4a92070905497aec4c0e64ae1761498cad09023f3f4b3e/uvloop-0.22.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:535cc37b3a04f6cd2c1ef65fa1d370c9a35b6695df735fcff5427323f2cd5473", size = 3723105, upload-time = "2025-10-16T22:16:28.252Z" }, - { url = "https://files.pythonhosted.org/packages/3d/ff/7f72e8170be527b4977b033239a83a68d5c881cc4775fca255c677f7ac5d/uvloop-0.22.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:fe94b4564e865d968414598eea1a6de60adba0c040ba4ed05ac1300de402cd42", size = 1359936, upload-time = "2025-10-16T22:16:29.436Z" }, - { url = "https://files.pythonhosted.org/packages/c3/c6/e5d433f88fd54d81ef4be58b2b7b0cea13c442454a1db703a1eea0db1a59/uvloop-0.22.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:51eb9bd88391483410daad430813d982010f9c9c89512321f5b60e2cddbdddd6", size = 752769, upload-time = "2025-10-16T22:16:30.493Z" }, - { url = "https://files.pythonhosted.org/packages/24/68/a6ac446820273e71aa762fa21cdcc09861edd3536ff47c5cd3b7afb10eeb/uvloop-0.22.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:700e674a166ca5778255e0e1dc4e9d79ab2acc57b9171b79e65feba7184b3370", size = 4317413, upload-time = "2025-10-16T22:16:31.644Z" }, - { url = "https://files.pythonhosted.org/packages/5f/6f/e62b4dfc7ad6518e7eff2516f680d02a0f6eb62c0c212e152ca708a0085e/uvloop-0.22.1-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:7b5b1ac819a3f946d3b2ee07f09149578ae76066d70b44df3fa990add49a82e4", size = 4426307, upload-time = "2025-10-16T22:16:32.917Z" }, - { url = "https://files.pythonhosted.org/packages/90/60/97362554ac21e20e81bcef1150cb2a7e4ffdaf8ea1e5b2e8bf7a053caa18/uvloop-0.22.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:e047cc068570bac9866237739607d1313b9253c3051ad84738cbb095be0537b2", size = 4131970, upload-time = "2025-10-16T22:16:34.015Z" }, - { url = "https://files.pythonhosted.org/packages/99/39/6b3f7d234ba3964c428a6e40006340f53ba37993f46ed6e111c6e9141d18/uvloop-0.22.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:512fec6815e2dd45161054592441ef76c830eddaad55c8aa30952e6fe1ed07c0", size = 4296343, upload-time = "2025-10-16T22:16:35.149Z" }, - { url = "https://files.pythonhosted.org/packages/89/8c/182a2a593195bfd39842ea68ebc084e20c850806117213f5a299dfc513d9/uvloop-0.22.1-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:561577354eb94200d75aca23fbde86ee11be36b00e52a4eaf8f50fb0c86b7705", size = 1358611, upload-time = "2025-10-16T22:16:36.833Z" }, - { url = "https://files.pythonhosted.org/packages/d2/14/e301ee96a6dc95224b6f1162cd3312f6d1217be3907b79173b06785f2fe7/uvloop-0.22.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:1cdf5192ab3e674ca26da2eada35b288d2fa49fdd0f357a19f0e7c4e7d5077c8", size = 751811, upload-time = "2025-10-16T22:16:38.275Z" }, - { url = "https://files.pythonhosted.org/packages/b7/02/654426ce265ac19e2980bfd9ea6590ca96a56f10c76e63801a2df01c0486/uvloop-0.22.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6e2ea3d6190a2968f4a14a23019d3b16870dd2190cd69c8180f7c632d21de68d", size = 4288562, upload-time = "2025-10-16T22:16:39.375Z" }, - { url = "https://files.pythonhosted.org/packages/15/c0/0be24758891ef825f2065cd5db8741aaddabe3e248ee6acc5e8a80f04005/uvloop-0.22.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0530a5fbad9c9e4ee3f2b33b148c6a64d47bbad8000ea63704fa8260f4cf728e", size = 4366890, upload-time = "2025-10-16T22:16:40.547Z" }, - { url = "https://files.pythonhosted.org/packages/d2/53/8369e5219a5855869bcee5f4d317f6da0e2c669aecf0ef7d371e3d084449/uvloop-0.22.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:bc5ef13bbc10b5335792360623cc378d52d7e62c2de64660616478c32cd0598e", size = 4119472, upload-time = "2025-10-16T22:16:41.694Z" }, - { url = "https://files.pythonhosted.org/packages/f8/ba/d69adbe699b768f6b29a5eec7b47dd610bd17a69de51b251126a801369ea/uvloop-0.22.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:1f38ec5e3f18c8a10ded09742f7fb8de0108796eb673f30ce7762ce1b8550cad", size = 4239051, upload-time = "2025-10-16T22:16:43.224Z" }, - { url = "https://files.pythonhosted.org/packages/90/cd/b62bdeaa429758aee8de8b00ac0dd26593a9de93d302bff3d21439e9791d/uvloop-0.22.1-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:3879b88423ec7e97cd4eba2a443aa26ed4e59b45e6b76aabf13fe2f27023a142", size = 1362067, upload-time = "2025-10-16T22:16:44.503Z" }, - { url = "https://files.pythonhosted.org/packages/0d/f8/a132124dfda0777e489ca86732e85e69afcd1ff7686647000050ba670689/uvloop-0.22.1-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:4baa86acedf1d62115c1dc6ad1e17134476688f08c6efd8a2ab076e815665c74", size = 752423, upload-time = "2025-10-16T22:16:45.968Z" }, - { url = "https://files.pythonhosted.org/packages/a3/94/94af78c156f88da4b3a733773ad5ba0b164393e357cc4bd0ab2e2677a7d6/uvloop-0.22.1-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:297c27d8003520596236bdb2335e6b3f649480bd09e00d1e3a99144b691d2a35", size = 4272437, upload-time = "2025-10-16T22:16:47.451Z" }, - { url = "https://files.pythonhosted.org/packages/b5/35/60249e9fd07b32c665192cec7af29e06c7cd96fa1d08b84f012a56a0b38e/uvloop-0.22.1-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c1955d5a1dd43198244d47664a5858082a3239766a839b2102a269aaff7a4e25", size = 4292101, upload-time = "2025-10-16T22:16:49.318Z" }, - { url = "https://files.pythonhosted.org/packages/02/62/67d382dfcb25d0a98ce73c11ed1a6fba5037a1a1d533dcbb7cab033a2636/uvloop-0.22.1-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:b31dc2fccbd42adc73bc4e7cdbae4fc5086cf378979e53ca5d0301838c5682c6", size = 4114158, upload-time = "2025-10-16T22:16:50.517Z" }, - { url = "https://files.pythonhosted.org/packages/f0/7a/f1171b4a882a5d13c8b7576f348acfe6074d72eaf52cccef752f748d4a9f/uvloop-0.22.1-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:93f617675b2d03af4e72a5333ef89450dfaa5321303ede6e67ba9c9d26878079", size = 4177360, upload-time = "2025-10-16T22:16:52.646Z" }, - { url = "https://files.pythonhosted.org/packages/79/7b/b01414f31546caf0919da80ad57cbfe24c56b151d12af68cee1b04922ca8/uvloop-0.22.1-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:37554f70528f60cad66945b885eb01f1bb514f132d92b6eeed1c90fd54ed6289", size = 1454790, upload-time = "2025-10-16T22:16:54.355Z" }, - { url = "https://files.pythonhosted.org/packages/d4/31/0bb232318dd838cad3fa8fb0c68c8b40e1145b32025581975e18b11fab40/uvloop-0.22.1-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:b76324e2dc033a0b2f435f33eb88ff9913c156ef78e153fb210e03c13da746b3", size = 796783, upload-time = "2025-10-16T22:16:55.906Z" }, - { url = "https://files.pythonhosted.org/packages/42/38/c9b09f3271a7a723a5de69f8e237ab8e7803183131bc57c890db0b6bb872/uvloop-0.22.1-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:badb4d8e58ee08dad957002027830d5c3b06aea446a6a3744483c2b3b745345c", size = 4647548, upload-time = "2025-10-16T22:16:57.008Z" }, - { url = "https://files.pythonhosted.org/packages/c1/37/945b4ca0ac27e3dc4952642d4c900edd030b3da6c9634875af6e13ae80e5/uvloop-0.22.1-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b91328c72635f6f9e0282e4a57da7470c7350ab1c9f48546c0f2866205349d21", size = 4467065, upload-time = "2025-10-16T22:16:58.206Z" }, - { url = "https://files.pythonhosted.org/packages/97/cc/48d232f33d60e2e2e0b42f4e73455b146b76ebe216487e862700457fbf3c/uvloop-0.22.1-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:daf620c2995d193449393d6c62131b3fbd40a63bf7b307a1527856ace637fe88", size = 4328384, upload-time = "2025-10-16T22:16:59.36Z" }, - { url = "https://files.pythonhosted.org/packages/e4/16/c1fd27e9549f3c4baf1dc9c20c456cd2f822dbf8de9f463824b0c0357e06/uvloop-0.22.1-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:6cde23eeda1a25c75b2e07d39970f3374105d5eafbaab2a4482be82f272d5a5e", size = 4296730, upload-time = "2025-10-16T22:17:00.744Z" }, -] - -[[package]] -name = "watchfiles" -version = "1.1.1" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "anyio" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/c2/c9/8869df9b2a2d6c59d79220a4db37679e74f807c559ffe5265e08b227a210/watchfiles-1.1.1.tar.gz", hash = "sha256:a173cb5c16c4f40ab19cecf48a534c409f7ea983ab8fed0741304a1c0a31b3f2", size = 94440, upload-time = "2025-10-14T15:06:21.08Z" } -wheels = [ - { url = "https://files.pythonhosted.org/packages/1f/f8/2c5f479fb531ce2f0564eda479faecf253d886b1ab3630a39b7bf7362d46/watchfiles-1.1.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:f57b396167a2565a4e8b5e56a5a1c537571733992b226f4f1197d79e94cf0ae5", size = 406529, upload-time = "2025-10-14T15:04:32.899Z" }, - { url = "https://files.pythonhosted.org/packages/fe/cd/f515660b1f32f65df671ddf6f85bfaca621aee177712874dc30a97397977/watchfiles-1.1.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:421e29339983e1bebc281fab40d812742268ad057db4aee8c4d2bce0af43b741", size = 394384, upload-time = "2025-10-14T15:04:33.761Z" }, - { url = "https://files.pythonhosted.org/packages/7b/c3/28b7dc99733eab43fca2d10f55c86e03bd6ab11ca31b802abac26b23d161/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6e43d39a741e972bab5d8100b5cdacf69db64e34eb19b6e9af162bccf63c5cc6", size = 448789, upload-time = "2025-10-14T15:04:34.679Z" }, - { url = "https://files.pythonhosted.org/packages/4a/24/33e71113b320030011c8e4316ccca04194bf0cbbaeee207f00cbc7d6b9f5/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f537afb3276d12814082a2e9b242bdcf416c2e8fd9f799a737990a1dbe906e5b", size = 460521, upload-time = "2025-10-14T15:04:35.963Z" }, - { url = "https://files.pythonhosted.org/packages/f4/c3/3c9a55f255aa57b91579ae9e98c88704955fa9dac3e5614fb378291155df/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b2cd9e04277e756a2e2d2543d65d1e2166d6fd4c9b183f8808634fda23f17b14", size = 488722, upload-time = "2025-10-14T15:04:37.091Z" }, - { url = "https://files.pythonhosted.org/packages/49/36/506447b73eb46c120169dc1717fe2eff07c234bb3232a7200b5f5bd816e9/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5f3f58818dc0b07f7d9aa7fe9eb1037aecb9700e63e1f6acfed13e9fef648f5d", size = 596088, upload-time = "2025-10-14T15:04:38.39Z" }, - { url = "https://files.pythonhosted.org/packages/82/ab/5f39e752a9838ec4d52e9b87c1e80f1ee3ccdbe92e183c15b6577ab9de16/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9bb9f66367023ae783551042d31b1d7fd422e8289eedd91f26754a66f44d5cff", size = 472923, upload-time = "2025-10-14T15:04:39.666Z" }, - { url = "https://files.pythonhosted.org/packages/af/b9/a419292f05e302dea372fa7e6fda5178a92998411f8581b9830d28fb9edb/watchfiles-1.1.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aebfd0861a83e6c3d1110b78ad54704486555246e542be3e2bb94195eabb2606", size = 456080, upload-time = "2025-10-14T15:04:40.643Z" }, - { url = "https://files.pythonhosted.org/packages/b0/c3/d5932fd62bde1a30c36e10c409dc5d54506726f08cb3e1d8d0ba5e2bc8db/watchfiles-1.1.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:5fac835b4ab3c6487b5dbad78c4b3724e26bcc468e886f8ba8cc4306f68f6701", size = 629432, upload-time = "2025-10-14T15:04:41.789Z" }, - { url = "https://files.pythonhosted.org/packages/f7/77/16bddd9779fafb795f1a94319dc965209c5641db5bf1edbbccace6d1b3c0/watchfiles-1.1.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:399600947b170270e80134ac854e21b3ccdefa11a9529a3decc1327088180f10", size = 623046, upload-time = "2025-10-14T15:04:42.718Z" }, - { url = "https://files.pythonhosted.org/packages/46/ef/f2ecb9a0f342b4bfad13a2787155c6ee7ce792140eac63a34676a2feeef2/watchfiles-1.1.1-cp311-cp311-win32.whl", hash = "sha256:de6da501c883f58ad50db3a32ad397b09ad29865b5f26f64c24d3e3281685849", size = 271473, upload-time = "2025-10-14T15:04:43.624Z" }, - { url = "https://files.pythonhosted.org/packages/94/bc/f42d71125f19731ea435c3948cad148d31a64fccde3867e5ba4edee901f9/watchfiles-1.1.1-cp311-cp311-win_amd64.whl", hash = "sha256:35c53bd62a0b885bf653ebf6b700d1bf05debb78ad9292cf2a942b23513dc4c4", size = 287598, upload-time = "2025-10-14T15:04:44.516Z" }, - { url = "https://files.pythonhosted.org/packages/57/c9/a30f897351f95bbbfb6abcadafbaca711ce1162f4db95fc908c98a9165f3/watchfiles-1.1.1-cp311-cp311-win_arm64.whl", hash = "sha256:57ca5281a8b5e27593cb7d82c2ac927ad88a96ed406aa446f6344e4328208e9e", size = 277210, upload-time = "2025-10-14T15:04:45.883Z" }, - { url = "https://files.pythonhosted.org/packages/74/d5/f039e7e3c639d9b1d09b07ea412a6806d38123f0508e5f9b48a87b0a76cc/watchfiles-1.1.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:8c89f9f2f740a6b7dcc753140dd5e1ab9215966f7a3530d0c0705c83b401bd7d", size = 404745, upload-time = "2025-10-14T15:04:46.731Z" }, - { url = "https://files.pythonhosted.org/packages/a5/96/a881a13aa1349827490dab2d363c8039527060cfcc2c92cc6d13d1b1049e/watchfiles-1.1.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:bd404be08018c37350f0d6e34676bd1e2889990117a2b90070b3007f172d0610", size = 391769, upload-time = "2025-10-14T15:04:48.003Z" }, - { url = "https://files.pythonhosted.org/packages/4b/5b/d3b460364aeb8da471c1989238ea0e56bec24b6042a68046adf3d9ddb01c/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8526e8f916bb5b9a0a777c8317c23ce65de259422bba5b31325a6fa6029d33af", size = 449374, upload-time = "2025-10-14T15:04:49.179Z" }, - { url = "https://files.pythonhosted.org/packages/b9/44/5769cb62d4ed055cb17417c0a109a92f007114a4e07f30812a73a4efdb11/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2edc3553362b1c38d9f06242416a5d8e9fe235c204a4072e988ce2e5bb1f69f6", size = 459485, upload-time = "2025-10-14T15:04:50.155Z" }, - { url = "https://files.pythonhosted.org/packages/19/0c/286b6301ded2eccd4ffd0041a1b726afda999926cf720aab63adb68a1e36/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:30f7da3fb3f2844259cba4720c3fc7138eb0f7b659c38f3bfa65084c7fc7abce", size = 488813, upload-time = "2025-10-14T15:04:51.059Z" }, - { url = "https://files.pythonhosted.org/packages/c7/2b/8530ed41112dd4a22f4dcfdb5ccf6a1baad1ff6eed8dc5a5f09e7e8c41c7/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f8979280bdafff686ba5e4d8f97840f929a87ed9cdf133cbbd42f7766774d2aa", size = 594816, upload-time = "2025-10-14T15:04:52.031Z" }, - { url = "https://files.pythonhosted.org/packages/ce/d2/f5f9fb49489f184f18470d4f99f4e862a4b3e9ac2865688eb2099e3d837a/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dcc5c24523771db3a294c77d94771abcfcb82a0e0ee8efd910c37c59ec1b31bb", size = 475186, upload-time = "2025-10-14T15:04:53.064Z" }, - { url = "https://files.pythonhosted.org/packages/cf/68/5707da262a119fb06fbe214d82dd1fe4a6f4af32d2d14de368d0349eb52a/watchfiles-1.1.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1db5d7ae38ff20153d542460752ff397fcf5c96090c1230803713cf3147a6803", size = 456812, upload-time = "2025-10-14T15:04:55.174Z" }, - { url = "https://files.pythonhosted.org/packages/66/ab/3cbb8756323e8f9b6f9acb9ef4ec26d42b2109bce830cc1f3468df20511d/watchfiles-1.1.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:28475ddbde92df1874b6c5c8aaeb24ad5be47a11f87cde5a28ef3835932e3e94", size = 630196, upload-time = "2025-10-14T15:04:56.22Z" }, - { url = "https://files.pythonhosted.org/packages/78/46/7152ec29b8335f80167928944a94955015a345440f524d2dfe63fc2f437b/watchfiles-1.1.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:36193ed342f5b9842edd3532729a2ad55c4160ffcfa3700e0d54be496b70dd43", size = 622657, upload-time = "2025-10-14T15:04:57.521Z" }, - { url = "https://files.pythonhosted.org/packages/0a/bf/95895e78dd75efe9a7f31733607f384b42eb5feb54bd2eb6ed57cc2e94f4/watchfiles-1.1.1-cp312-cp312-win32.whl", hash = "sha256:859e43a1951717cc8de7f4c77674a6d389b106361585951d9e69572823f311d9", size = 272042, upload-time = "2025-10-14T15:04:59.046Z" }, - { url = "https://files.pythonhosted.org/packages/87/0a/90eb755f568de2688cb220171c4191df932232c20946966c27a59c400850/watchfiles-1.1.1-cp312-cp312-win_amd64.whl", hash = "sha256:91d4c9a823a8c987cce8fa2690923b069966dabb196dd8d137ea2cede885fde9", size = 288410, upload-time = "2025-10-14T15:05:00.081Z" }, - { url = "https://files.pythonhosted.org/packages/36/76/f322701530586922fbd6723c4f91ace21364924822a8772c549483abed13/watchfiles-1.1.1-cp312-cp312-win_arm64.whl", hash = "sha256:a625815d4a2bdca61953dbba5a39d60164451ef34c88d751f6c368c3ea73d404", size = 278209, upload-time = "2025-10-14T15:05:01.168Z" }, - { url = "https://files.pythonhosted.org/packages/bb/f4/f750b29225fe77139f7ae5de89d4949f5a99f934c65a1f1c0b248f26f747/watchfiles-1.1.1-cp313-cp313-macosx_10_12_x86_64.whl", hash = "sha256:130e4876309e8686a5e37dba7d5e9bc77e6ed908266996ca26572437a5271e18", size = 404321, upload-time = "2025-10-14T15:05:02.063Z" }, - { url = "https://files.pythonhosted.org/packages/2b/f9/f07a295cde762644aa4c4bb0f88921d2d141af45e735b965fb2e87858328/watchfiles-1.1.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:5f3bde70f157f84ece3765b42b4a52c6ac1a50334903c6eaf765362f6ccca88a", size = 391783, upload-time = "2025-10-14T15:05:03.052Z" }, - { url = "https://files.pythonhosted.org/packages/bc/11/fc2502457e0bea39a5c958d86d2cb69e407a4d00b85735ca724bfa6e0d1a/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:14e0b1fe858430fc0251737ef3824c54027bedb8c37c38114488b8e131cf8219", size = 449279, upload-time = "2025-10-14T15:05:04.004Z" }, - { url = "https://files.pythonhosted.org/packages/e3/1f/d66bc15ea0b728df3ed96a539c777acfcad0eb78555ad9efcaa1274688f0/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f27db948078f3823a6bb3b465180db8ebecf26dd5dae6f6180bd87383b6b4428", size = 459405, upload-time = "2025-10-14T15:05:04.942Z" }, - { url = "https://files.pythonhosted.org/packages/be/90/9f4a65c0aec3ccf032703e6db02d89a157462fbb2cf20dd415128251cac0/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:059098c3a429f62fc98e8ec62b982230ef2c8df68c79e826e37b895bc359a9c0", size = 488976, upload-time = "2025-10-14T15:05:05.905Z" }, - { url = "https://files.pythonhosted.org/packages/37/57/ee347af605d867f712be7029bb94c8c071732a4b44792e3176fa3c612d39/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bfb5862016acc9b869bb57284e6cb35fdf8e22fe59f7548858e2f971d045f150", size = 595506, upload-time = "2025-10-14T15:05:06.906Z" }, - { url = "https://files.pythonhosted.org/packages/a8/78/cc5ab0b86c122047f75e8fc471c67a04dee395daf847d3e59381996c8707/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:319b27255aacd9923b8a276bb14d21a5f7ff82564c744235fc5eae58d95422ae", size = 474936, upload-time = "2025-10-14T15:05:07.906Z" }, - { url = "https://files.pythonhosted.org/packages/62/da/def65b170a3815af7bd40a3e7010bf6ab53089ef1b75d05dd5385b87cf08/watchfiles-1.1.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c755367e51db90e75b19454b680903631d41f9e3607fbd941d296a020c2d752d", size = 456147, upload-time = "2025-10-14T15:05:09.138Z" }, - { url = "https://files.pythonhosted.org/packages/57/99/da6573ba71166e82d288d4df0839128004c67d2778d3b566c138695f5c0b/watchfiles-1.1.1-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:c22c776292a23bfc7237a98f791b9ad3144b02116ff10d820829ce62dff46d0b", size = 630007, upload-time = "2025-10-14T15:05:10.117Z" }, - { url = "https://files.pythonhosted.org/packages/a8/51/7439c4dd39511368849eb1e53279cd3454b4a4dbace80bab88feeb83c6b5/watchfiles-1.1.1-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:3a476189be23c3686bc2f4321dd501cb329c0a0469e77b7b534ee10129ae6374", size = 622280, upload-time = "2025-10-14T15:05:11.146Z" }, - { url = "https://files.pythonhosted.org/packages/95/9c/8ed97d4bba5db6fdcdb2b298d3898f2dd5c20f6b73aee04eabe56c59677e/watchfiles-1.1.1-cp313-cp313-win32.whl", hash = "sha256:bf0a91bfb5574a2f7fc223cf95eeea79abfefa404bf1ea5e339c0c1560ae99a0", size = 272056, upload-time = "2025-10-14T15:05:12.156Z" }, - { url = "https://files.pythonhosted.org/packages/1f/f3/c14e28429f744a260d8ceae18bf58c1d5fa56b50d006a7a9f80e1882cb0d/watchfiles-1.1.1-cp313-cp313-win_amd64.whl", hash = "sha256:52e06553899e11e8074503c8e716d574adeeb7e68913115c4b3653c53f9bae42", size = 288162, upload-time = "2025-10-14T15:05:13.208Z" }, - { url = "https://files.pythonhosted.org/packages/dc/61/fe0e56c40d5cd29523e398d31153218718c5786b5e636d9ae8ae79453d27/watchfiles-1.1.1-cp313-cp313-win_arm64.whl", hash = "sha256:ac3cc5759570cd02662b15fbcd9d917f7ecd47efe0d6b40474eafd246f91ea18", size = 277909, upload-time = "2025-10-14T15:05:14.49Z" }, - { url = "https://files.pythonhosted.org/packages/79/42/e0a7d749626f1e28c7108a99fb9bf524b501bbbeb9b261ceecde644d5a07/watchfiles-1.1.1-cp313-cp313t-macosx_10_12_x86_64.whl", hash = "sha256:563b116874a9a7ce6f96f87cd0b94f7faf92d08d0021e837796f0a14318ef8da", size = 403389, upload-time = "2025-10-14T15:05:15.777Z" }, - { url = "https://files.pythonhosted.org/packages/15/49/08732f90ce0fbbc13913f9f215c689cfc9ced345fb1bcd8829a50007cc8d/watchfiles-1.1.1-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3ad9fe1dae4ab4212d8c91e80b832425e24f421703b5a42ef2e4a1e215aff051", size = 389964, upload-time = "2025-10-14T15:05:16.85Z" }, - { url = "https://files.pythonhosted.org/packages/27/0d/7c315d4bd5f2538910491a0393c56bf70d333d51bc5b34bee8e68e8cea19/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce70f96a46b894b36eba678f153f052967a0d06d5b5a19b336ab0dbbd029f73e", size = 448114, upload-time = "2025-10-14T15:05:17.876Z" }, - { url = "https://files.pythonhosted.org/packages/c3/24/9e096de47a4d11bc4df41e9d1e61776393eac4cb6eb11b3e23315b78b2cc/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:cb467c999c2eff23a6417e58d75e5828716f42ed8289fe6b77a7e5a91036ca70", size = 460264, upload-time = "2025-10-14T15:05:18.962Z" }, - { url = "https://files.pythonhosted.org/packages/cc/0f/e8dea6375f1d3ba5fcb0b3583e2b493e77379834c74fd5a22d66d85d6540/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:836398932192dae4146c8f6f737d74baeac8b70ce14831a239bdb1ca882fc261", size = 487877, upload-time = "2025-10-14T15:05:20.094Z" }, - { url = "https://files.pythonhosted.org/packages/ac/5b/df24cfc6424a12deb41503b64d42fbea6b8cb357ec62ca84a5a3476f654a/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:743185e7372b7bc7c389e1badcc606931a827112fbbd37f14c537320fca08620", size = 595176, upload-time = "2025-10-14T15:05:21.134Z" }, - { url = "https://files.pythonhosted.org/packages/8f/b5/853b6757f7347de4e9b37e8cc3289283fb983cba1ab4d2d7144694871d9c/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:afaeff7696e0ad9f02cbb8f56365ff4686ab205fcf9c4c5b6fdfaaa16549dd04", size = 473577, upload-time = "2025-10-14T15:05:22.306Z" }, - { url = "https://files.pythonhosted.org/packages/e1/f7/0a4467be0a56e80447c8529c9fce5b38eab4f513cb3d9bf82e7392a5696b/watchfiles-1.1.1-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3f7eb7da0eb23aa2ba036d4f616d46906013a68caf61b7fdbe42fc8b25132e77", size = 455425, upload-time = "2025-10-14T15:05:23.348Z" }, - { url = "https://files.pythonhosted.org/packages/8e/e0/82583485ea00137ddf69bc84a2db88bd92ab4a6e3c405e5fb878ead8d0e7/watchfiles-1.1.1-cp313-cp313t-musllinux_1_1_aarch64.whl", hash = "sha256:831a62658609f0e5c64178211c942ace999517f5770fe9436be4c2faeba0c0ef", size = 628826, upload-time = "2025-10-14T15:05:24.398Z" }, - { url = "https://files.pythonhosted.org/packages/28/9a/a785356fccf9fae84c0cc90570f11702ae9571036fb25932f1242c82191c/watchfiles-1.1.1-cp313-cp313t-musllinux_1_1_x86_64.whl", hash = "sha256:f9a2ae5c91cecc9edd47e041a930490c31c3afb1f5e6d71de3dc671bfaca02bf", size = 622208, upload-time = "2025-10-14T15:05:25.45Z" }, - { url = "https://files.pythonhosted.org/packages/c3/f4/0872229324ef69b2c3edec35e84bd57a1289e7d3fe74588048ed8947a323/watchfiles-1.1.1-cp314-cp314-macosx_10_12_x86_64.whl", hash = "sha256:d1715143123baeeaeadec0528bb7441103979a1d5f6fd0e1f915383fea7ea6d5", size = 404315, upload-time = "2025-10-14T15:05:26.501Z" }, - { url = "https://files.pythonhosted.org/packages/7b/22/16d5331eaed1cb107b873f6ae1b69e9ced582fcf0c59a50cd84f403b1c32/watchfiles-1.1.1-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:39574d6370c4579d7f5d0ad940ce5b20db0e4117444e39b6d8f99db5676c52fd", size = 390869, upload-time = "2025-10-14T15:05:27.649Z" }, - { url = "https://files.pythonhosted.org/packages/b2/7e/5643bfff5acb6539b18483128fdc0ef2cccc94a5b8fbda130c823e8ed636/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7365b92c2e69ee952902e8f70f3ba6360d0d596d9299d55d7d386df84b6941fb", size = 449919, upload-time = "2025-10-14T15:05:28.701Z" }, - { url = "https://files.pythonhosted.org/packages/51/2e/c410993ba5025a9f9357c376f48976ef0e1b1aefb73b97a5ae01a5972755/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:bfff9740c69c0e4ed32416f013f3c45e2ae42ccedd1167ef2d805c000b6c71a5", size = 460845, upload-time = "2025-10-14T15:05:30.064Z" }, - { url = "https://files.pythonhosted.org/packages/8e/a4/2df3b404469122e8680f0fcd06079317e48db58a2da2950fb45020947734/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b27cf2eb1dda37b2089e3907d8ea92922b673c0c427886d4edc6b94d8dfe5db3", size = 489027, upload-time = "2025-10-14T15:05:31.064Z" }, - { url = "https://files.pythonhosted.org/packages/ea/84/4587ba5b1f267167ee715b7f66e6382cca6938e0a4b870adad93e44747e6/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:526e86aced14a65a5b0ec50827c745597c782ff46b571dbfe46192ab9e0b3c33", size = 595615, upload-time = "2025-10-14T15:05:32.074Z" }, - { url = "https://files.pythonhosted.org/packages/6a/0f/c6988c91d06e93cd0bb3d4a808bcf32375ca1904609835c3031799e3ecae/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:04e78dd0b6352db95507fd8cb46f39d185cf8c74e4cf1e4fbad1d3df96faf510", size = 474836, upload-time = "2025-10-14T15:05:33.209Z" }, - { url = "https://files.pythonhosted.org/packages/b4/36/ded8aebea91919485b7bbabbd14f5f359326cb5ec218cd67074d1e426d74/watchfiles-1.1.1-cp314-cp314-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5c85794a4cfa094714fb9c08d4a218375b2b95b8ed1666e8677c349906246c05", size = 455099, upload-time = "2025-10-14T15:05:34.189Z" }, - { url = "https://files.pythonhosted.org/packages/98/e0/8c9bdba88af756a2fce230dd365fab2baf927ba42cd47521ee7498fd5211/watchfiles-1.1.1-cp314-cp314-musllinux_1_1_aarch64.whl", hash = "sha256:74d5012b7630714b66be7b7b7a78855ef7ad58e8650c73afc4c076a1f480a8d6", size = 630626, upload-time = "2025-10-14T15:05:35.216Z" }, - { url = "https://files.pythonhosted.org/packages/2a/84/a95db05354bf2d19e438520d92a8ca475e578c647f78f53197f5a2f17aaf/watchfiles-1.1.1-cp314-cp314-musllinux_1_1_x86_64.whl", hash = "sha256:8fbe85cb3201c7d380d3d0b90e63d520f15d6afe217165d7f98c9c649654db81", size = 622519, upload-time = "2025-10-14T15:05:36.259Z" }, - { url = "https://files.pythonhosted.org/packages/1d/ce/d8acdc8de545de995c339be67711e474c77d643555a9bb74a9334252bd55/watchfiles-1.1.1-cp314-cp314-win32.whl", hash = "sha256:3fa0b59c92278b5a7800d3ee7733da9d096d4aabcfabb9a928918bd276ef9b9b", size = 272078, upload-time = "2025-10-14T15:05:37.63Z" }, - { url = "https://files.pythonhosted.org/packages/c4/c9/a74487f72d0451524be827e8edec251da0cc1fcf111646a511ae752e1a3d/watchfiles-1.1.1-cp314-cp314-win_amd64.whl", hash = "sha256:c2047d0b6cea13b3316bdbafbfa0c4228ae593d995030fda39089d36e64fc03a", size = 287664, upload-time = "2025-10-14T15:05:38.95Z" }, - { url = "https://files.pythonhosted.org/packages/df/b8/8ac000702cdd496cdce998c6f4ee0ca1f15977bba51bdf07d872ebdfc34c/watchfiles-1.1.1-cp314-cp314-win_arm64.whl", hash = "sha256:842178b126593addc05acf6fce960d28bc5fae7afbaa2c6c1b3a7b9460e5be02", size = 277154, upload-time = "2025-10-14T15:05:39.954Z" }, - { url = "https://files.pythonhosted.org/packages/47/a8/e3af2184707c29f0f14b1963c0aace6529f9d1b8582d5b99f31bbf42f59e/watchfiles-1.1.1-cp314-cp314t-macosx_10_12_x86_64.whl", hash = "sha256:88863fbbc1a7312972f1c511f202eb30866370ebb8493aef2812b9ff28156a21", size = 403820, upload-time = "2025-10-14T15:05:40.932Z" }, - { url = "https://files.pythonhosted.org/packages/c0/ec/e47e307c2f4bd75f9f9e8afbe3876679b18e1bcec449beca132a1c5ffb2d/watchfiles-1.1.1-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:55c7475190662e202c08c6c0f4d9e345a29367438cf8e8037f3155e10a88d5a5", size = 390510, upload-time = "2025-10-14T15:05:41.945Z" }, - { url = "https://files.pythonhosted.org/packages/d5/a0/ad235642118090f66e7b2f18fd5c42082418404a79205cdfca50b6309c13/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3f53fa183d53a1d7a8852277c92b967ae99c2d4dcee2bfacff8868e6e30b15f7", size = 448408, upload-time = "2025-10-14T15:05:43.385Z" }, - { url = "https://files.pythonhosted.org/packages/df/85/97fa10fd5ff3332ae17e7e40e20784e419e28521549780869f1413742e9d/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:6aae418a8b323732fa89721d86f39ec8f092fc2af67f4217a2b07fd3e93c6101", size = 458968, upload-time = "2025-10-14T15:05:44.404Z" }, - { url = "https://files.pythonhosted.org/packages/47/c2/9059c2e8966ea5ce678166617a7f75ecba6164375f3b288e50a40dc6d489/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f096076119da54a6080e8920cbdaac3dbee667eb91dcc5e5b78840b87415bd44", size = 488096, upload-time = "2025-10-14T15:05:45.398Z" }, - { url = "https://files.pythonhosted.org/packages/94/44/d90a9ec8ac309bc26db808a13e7bfc0e4e78b6fc051078a554e132e80160/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:00485f441d183717038ed2e887a7c868154f216877653121068107b227a2f64c", size = 596040, upload-time = "2025-10-14T15:05:46.502Z" }, - { url = "https://files.pythonhosted.org/packages/95/68/4e3479b20ca305cfc561db3ed207a8a1c745ee32bf24f2026a129d0ddb6e/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a55f3e9e493158d7bfdb60a1165035f1cf7d320914e7b7ea83fe22c6023b58fc", size = 473847, upload-time = "2025-10-14T15:05:47.484Z" }, - { url = "https://files.pythonhosted.org/packages/4f/55/2af26693fd15165c4ff7857e38330e1b61ab8c37d15dc79118cdba115b7a/watchfiles-1.1.1-cp314-cp314t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8c91ed27800188c2ae96d16e3149f199d62f86c7af5f5f4d2c61a3ed8cd3666c", size = 455072, upload-time = "2025-10-14T15:05:48.928Z" }, - { url = "https://files.pythonhosted.org/packages/66/1d/d0d200b10c9311ec25d2273f8aad8c3ef7cc7ea11808022501811208a750/watchfiles-1.1.1-cp314-cp314t-musllinux_1_1_aarch64.whl", hash = "sha256:311ff15a0bae3714ffb603e6ba6dbfba4065ab60865d15a6ec544133bdb21099", size = 629104, upload-time = "2025-10-14T15:05:49.908Z" }, - { url = "https://files.pythonhosted.org/packages/e3/bd/fa9bb053192491b3867ba07d2343d9f2252e00811567d30ae8d0f78136fe/watchfiles-1.1.1-cp314-cp314t-musllinux_1_1_x86_64.whl", hash = "sha256:a916a2932da8f8ab582f242c065f5c81bed3462849ca79ee357dd9551b0e9b01", size = 622112, upload-time = "2025-10-14T15:05:50.941Z" }, - { url = "https://files.pythonhosted.org/packages/d3/8e/e500f8b0b77be4ff753ac94dc06b33d8f0d839377fee1b78e8c8d8f031bf/watchfiles-1.1.1-pp311-pypy311_pp73-macosx_10_12_x86_64.whl", hash = "sha256:db476ab59b6765134de1d4fe96a1a9c96ddf091683599be0f26147ea1b2e4b88", size = 408250, upload-time = "2025-10-14T15:06:10.264Z" }, - { url = "https://files.pythonhosted.org/packages/bd/95/615e72cd27b85b61eec764a5ca51bd94d40b5adea5ff47567d9ebc4d275a/watchfiles-1.1.1-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:89eef07eee5e9d1fda06e38822ad167a044153457e6fd997f8a858ab7564a336", size = 396117, upload-time = "2025-10-14T15:06:11.28Z" }, - { url = "https://files.pythonhosted.org/packages/c9/81/e7fe958ce8a7fb5c73cc9fb07f5aeaf755e6aa72498c57d760af760c91f8/watchfiles-1.1.1-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce19e06cbda693e9e7686358af9cd6f5d61312ab8b00488bc36f5aabbaf77e24", size = 450493, upload-time = "2025-10-14T15:06:12.321Z" }, - { url = "https://files.pythonhosted.org/packages/6e/d4/ed38dd3b1767193de971e694aa544356e63353c33a85d948166b5ff58b9e/watchfiles-1.1.1-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3e6f39af2eab0118338902798b5aa6664f46ff66bc0280de76fca67a7f262a49", size = 457546, upload-time = "2025-10-14T15:06:13.372Z" }, -] - [[package]] name = "websockets" version = "15.0.1" diff --git a/src/backend/v4/callbacks/response_handlers.py b/src/backend/v4/callbacks/response_handlers.py index 297c8814b..88c6085f5 100644 --- a/src/backend/v4/callbacks/response_handlers.py +++ b/src/backend/v4/callbacks/response_handlers.py @@ -8,10 +8,7 @@ import re from typing import Any -from agent_framework import ChatMessage -# Removed: from agent_framework._content import FunctionCallContent (does not exist) - -from agent_framework._workflows._magentic import AgentRunResponseUpdate # Streaming update type from workflows +from agent_framework import ChatMessage, AgentRunUpdateEvent from v4.config.settings import connection_config from v4.models.messages import ( @@ -111,26 +108,31 @@ def agent_response_callback( async def streaming_agent_response_callback( agent_id: str, - update: AgentRunResponseUpdate, + update, # AgentRunUpdateEvent.data or similar streaming update object is_final: bool, user_id: str | None = None, ) -> None: """ - Streaming callback for incremental agent output (AgentRunResponseUpdate). + Streaming callback for incremental agent output. """ if not user_id: return try: + # Handle both AgentRunUpdateEvent.data and raw text updates chunk_text = getattr(update, "text", None) - if not chunk_text: - contents = getattr(update, "contents", []) or [] - collected = [] - for item in contents: - txt = getattr(item, "text", None) - if txt: - collected.append(str(txt)) - chunk_text = "".join(collected) if collected else "" + + # If text is None, don't fall back to str(update) as that would show object repr + # Just skip if there's no actual text content + if chunk_text is None: + # Check if update is a ChatMessage + if isinstance(update, ChatMessage): + chunk_text = update.text or "" + elif hasattr(update, "content"): + chunk_text = str(update.content) if update.content else "" + else: + # Skip if no text content available + return cleaned = clean_citations(chunk_text or "") diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index bd9382ff2..88fde549b 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -10,8 +10,8 @@ MCPStreamableHTTPTool, ) -# from agent_framework.azure import AzureAIAgentClient -from agent_framework_azure_ai import AzureAIAgentClient +# from agent_framework.azure import AzureAIClient +from agent_framework_azure_ai import AzureAIClient from azure.ai.agents.aio import AgentsClient from azure.identity.aio import DefaultAzureCredential from common.database.database_base import DatabaseBase @@ -51,7 +51,7 @@ def __init__( self._agent: ChatAgent | None = None self.team_service: TeamService | None = team_service self.team_config: TeamConfiguration | None = team_config - self.client: Optional[AzureAIAgentClient] = None + self.client: Optional[AzureAIClient] = None self.project_endpoint = project_endpoint self.creds: Optional[DefaultAzureCredential] = None self.memory_store: Optional[DatabaseBase] = memory_store @@ -105,7 +105,7 @@ async def close(self) -> None: # Attempt to close the underlying agent/client if it exposes close() if self._agent and hasattr(self._agent, "close"): try: - await self._agent.close() # AzureAIAgentClient has async close + await self._agent.close() # AzureAIClient has async close except Exception as exc: # Best-effort close; log failure but continue teardown self.logger.warning( @@ -148,24 +148,22 @@ async def _after_open(self) -> None: """Subclasses must build self._agent here.""" raise NotImplementedError - def get_chat_client(self, chat_client) -> AzureAIAgentClient: - """Return the underlying ChatClientProtocol (AzureAIAgentClient).""" + def get_chat_client(self, chat_client) -> AzureAIClient: + """Return the underlying ChatClientProtocol (AzureAIClient).""" if chat_client: return chat_client if ( self._agent and self._agent.chat_client - and self._agent.chat_client.agent_id is not None ): return self._agent.chat_client # type: ignore - chat_client = AzureAIAgentClient( + chat_client = AzureAIClient( project_endpoint=self.project_endpoint, model_deployment_name=self.model_deployment_name, - async_credential=self.creds, + credential=self.creds, ) self.logger.info( - "Created new AzureAIAgentClient for get chat client", - extra={"agent_id": chat_client.agent_id}, + "Created new AzureAIClient for get chat client", ) return chat_client @@ -219,20 +217,17 @@ async def resolve_agent_id(self, agent_id: str) -> Optional[str]: return None def get_agent_id(self, chat_client) -> str: - """Return the underlying agent ID.""" - if chat_client and chat_client.agent_id is not None: - return chat_client.agent_id - if ( - self._agent - and self._agent.chat_client - and self._agent.chat_client.agent_id is not None - ): - return self._agent.chat_client.agent_id # type: ignore + """Return the underlying agent ID or generate a new one. + + Note: The new AzureAIClient doesn't expose agent_id directly. + We generate a new ID if not available. + """ + # Generate a new agent ID since AzureAIClient doesn't expose agent_id id = generate_assistant_id() self.logger.info("Generated new agent ID: %s", id) return id - async def get_database_team_agent(self) -> Optional[AzureAIAgentClient]: + async def get_database_team_agent(self) -> Optional[AzureAIClient]: """Retrieve existing team agent from database, if any.""" chat_client = None try: @@ -258,24 +253,24 @@ async def get_database_team_agent(self) -> Optional[AzureAIAgentClient]: # Create client with resolved ID, preferring project_client for RAI agents if self.agent_name == "RAIAgent" and self.project_client: - chat_client = AzureAIAgentClient( + chat_client = AzureAIClient( project_client=self.project_client, agent_id=resolved, - async_credential=self.creds, + credential=self.creds, ) self.logger.info( - "RAI.AgentReuseSuccess: Created AzureAIAgentClient via Projects SDK (id=%s)", + "RAI.AgentReuseSuccess: Created AzureAIClient via Projects SDK (id=%s)", resolved, ) else: - chat_client = AzureAIAgentClient( + chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_id=resolved, model_deployment_name=self.model_deployment_name, - async_credential=self.creds, + credential=self.creds, ) self.logger.info( - "Created AzureAIAgentClient via endpoint (id=%s)", resolved + "Created AzureAIClient via endpoint (id=%s)", resolved ) except Exception as ex: @@ -339,10 +334,10 @@ async def _prepare_mcp_tool(self) -> None: class AzureAgentBase(MCPEnabledBase): """ - Extends MCPEnabledBase with Azure credential + AzureAIAgentClient contexts. + Extends MCPEnabledBase with Azure credential + AzureAIClient contexts. Subclasses: - create or attach an Azure AI Agent definition - - instantiate an AzureAIAgentClient and assign to self._agent + - instantiate an AzureAIClient and assign to self._agent - optionally register themselves via agent_registry """ diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index df6221699..614904b6d 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -6,7 +6,7 @@ from agent_framework import (ChatAgent, ChatMessage, HostedCodeInterpreterTool, Role) from agent_framework_azure_ai import \ - AzureAIAgentClient # Provided by agent_framework + AzureAIClient # Provided by agent_framework from azure.ai.projects.models import ConnectionType from common.config.app_config import config from common.database.database_base import DatabaseBase @@ -111,7 +111,7 @@ async def _collect_tools(self) -> List: # ------------------------- # Azure Search helper # ------------------------- - async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional[AzureAIAgentClient]: + async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional[AzureAIClient]: """ Create a server-side Azure AI agent with Azure AI Search raw tool. @@ -123,7 +123,7 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional Returns: - AzureAIAgentClient | None + AzureAIClient | None """ if chatClient: return chatClient @@ -205,10 +205,10 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional query_type, ) - chat_client = AzureAIAgentClient( + chat_client = AzureAIClient( project_client=self.project_client, agent_id=azure_agent.id, - async_credential=self.creds, + credential=self.creds, ) return chat_client except Exception as ex: @@ -301,8 +301,8 @@ async def invoke(self, prompt: str): agent_saved = False async for update in self._agent.run_stream(messages): - # Save agent ID only once on first update (agent ID won't change during streaming) - if not agent_saved and self._agent.chat_client.agent_id: + # Save agent ID only once on first update + if not agent_saved: await self.save_database_team_agent() agent_saved = True yield update diff --git a/src/backend/v4/magentic_agents/proxy_agent.py b/src/backend/v4/magentic_agents/proxy_agent.py index a6ba9d3c4..79a84492b 100644 --- a/src/backend/v4/magentic_agents/proxy_agent.py +++ b/src/backend/v4/magentic_agents/proxy_agent.py @@ -16,13 +16,12 @@ from typing import Any, AsyncIterable from agent_framework import ( - AgentRunResponse, - AgentRunResponseUpdate, + AgentResponse, + AgentResponseUpdate, BaseAgent, ChatMessage, Role, - TextContent, - UsageContent, + Content, UsageDetails, AgentThread, ) @@ -88,7 +87,7 @@ async def run( *, thread: AgentThread | None = None, **kwargs: Any, - ) -> AgentRunResponse: + ) -> AgentResponse: """ Get complete clarification response (non-streaming). @@ -98,7 +97,7 @@ async def run( kwargs: Additional keyword arguments Returns: - AgentRunResponse with the clarification + AgentResponse with the clarification """ # Collect all streaming updates response_messages: list[ChatMessage] = [] @@ -113,7 +112,7 @@ async def run( ) ) - return AgentRunResponse( + return AgentResponse( messages=response_messages, response_id=response_id, ) @@ -124,7 +123,7 @@ def run_stream( *, thread: AgentThread | None = None, **kwargs: Any, - ) -> AsyncIterable[AgentRunResponseUpdate]: + ) -> AsyncIterable[AgentResponseUpdate]: """ Stream clarification process with human interaction. @@ -143,7 +142,7 @@ async def _invoke_stream_internal( messages: str | ChatMessage | list[str] | list[ChatMessage] | None, thread: AgentThread | None, **kwargs: Any, - ) -> AsyncIterable[AgentRunResponseUpdate]: + ) -> AsyncIterable[AgentResponseUpdate]: """ Internal streaming implementation. @@ -205,9 +204,10 @@ async def _invoke_stream_internal( message_id = str(uuid.uuid4()) # Yield final assistant text update with explicit text content - text_update = AgentRunResponseUpdate( + # New API: use Content.from_text() or pass text directly to AgentResponseUpdate + text_update = AgentResponseUpdate( role=Role.ASSISTANT, - contents=[TextContent(text=synthetic_reply)], + text=synthetic_reply, # New API accepts text directly author_name=self.name, response_id=response_id, message_id=message_id, @@ -218,10 +218,10 @@ async def _invoke_stream_internal( # Yield synthetic usage update for consistency # Use same message_id to indicate this is part of the same message - usage_update = AgentRunResponseUpdate( + usage_update = AgentResponseUpdate( role=Role.ASSISTANT, contents=[ - UsageContent( + Content.from_usage( UsageDetails( input_token_count=len(message_text.split()), output_token_count=len(synthetic_reply.split()), diff --git a/src/backend/v4/orchestration/human_approval_manager.py b/src/backend/v4/orchestration/human_approval_manager.py index 2a3ab5be7..654d72a23 100644 --- a/src/backend/v4/orchestration/human_approval_manager.py +++ b/src/backend/v4/orchestration/human_approval_manager.py @@ -34,11 +34,12 @@ class HumanApprovalMagenticManager(StandardMagenticManager): magentic_plan: Optional[MPlan] = None current_user_id: str # populated in __init__ - def __init__(self, user_id: str, *args, **kwargs): + def __init__(self, user_id: str, agent, *args, **kwargs): """ Initialize the HumanApprovalMagenticManager. Args: user_id: ID of the user to associate with this orchestration instance. + agent: The manager ChatAgent for orchestration (required by new API). *args: Additional positional arguments for the parent StandardMagenticManager. **kwargs: Additional keyword arguments for the parent StandardMagenticManager. """ @@ -76,7 +77,8 @@ def __init__(self, user_id: str, *args, **kwargs): kwargs["final_answer_prompt"] = ORCHESTRATOR_FINAL_ANSWER_PROMPT + final_append self.current_user_id = user_id - super().__init__(*args, **kwargs) + # New API: StandardMagenticManager takes agent as first positional argument + super().__init__(agent, *args, **kwargs) async def plan(self, magentic_context: MagenticContext) -> Any: """ diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index e105a34cb..ec4f7aea1 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -2,20 +2,22 @@ import asyncio import logging +import re import uuid from typing import List, Optional # agent_framework imports -from agent_framework_azure_ai import AzureAIAgentClient +from agent_framework_azure_ai import AzureAIClient from agent_framework import ( + ChatAgent, ChatMessage, WorkflowOutputEvent, MagenticBuilder, InMemoryCheckpointStorage, - MagenticOrchestratorMessageEvent, - MagenticAgentDeltaEvent, - MagenticAgentMessageEvent, - MagenticFinalResultEvent, + AgentRunUpdateEvent, + GroupChatRequestSentEvent, + MagenticOrchestratorEvent, + MagenticProgressLedger, ) from common.config.app_config import config @@ -58,7 +60,7 @@ async def init_orchestration( Initialize a Magentic workflow with: - Provided agents (participants) - HumanApprovalMagenticManager as orchestrator manager - - AzureAIAgentClient as the underlying chat client + - AzureAIClient as the underlying chat client - Event-based callbacks for streaming and final responses - Uses same deployment, endpoint, and credentials - Applies same execution settings (temperature, max_tokens) @@ -72,33 +74,46 @@ async def init_orchestration( # Create Azure AI Agent client for orchestration using config # This replaces AzureChatCompletion from SK - agent_name = team_config.name if team_config.name else "OrchestratorAgent" + # Sanitize agent name: must start/end with alphanumeric, only hyphens allowed, max 63 chars + raw_name = team_config.name if team_config.name else "OrchestratorAgent" + # Replace spaces and invalid chars with hyphens, strip leading/trailing hyphens + sanitized_name = re.sub(r'[^a-zA-Z0-9-]', '-', raw_name) + sanitized_name = re.sub(r'-+', '-', sanitized_name) # Collapse multiple hyphens + sanitized_name = sanitized_name.strip('-')[:63] # Trim and limit length + agent_name = sanitized_name if sanitized_name else "OrchestratorAgent" try: - chat_client = AzureAIAgentClient( + # Create the chat client (AzureAIClient) + chat_client = AzureAIClient( project_endpoint=config.AZURE_AI_PROJECT_ENDPOINT, model_deployment_name=team_config.deployment_name, agent_name=agent_name, - async_credential=credential, + credential=credential, + ) + + # New API: Create a ChatAgent to wrap the chat client for the manager + manager_agent = ChatAgent( + chat_client=chat_client, + name="MagenticManager", + description="Orchestrator that coordinates the team to complete complex tasks efficiently.", + instructions="You coordinate a team to complete complex tasks efficiently.", ) cls.logger.info( - "Created AzureAIAgentClient for orchestration with model '%s' at endpoint '%s'", + "Created AzureAIClient and manager ChatAgent for orchestration with model '%s' at endpoint '%s'", team_config.deployment_name, config.AZURE_AI_PROJECT_ENDPOINT, ) except Exception as e: - cls.logger.error("Failed to create AzureAIAgentClient: %s", e) + cls.logger.error("Failed to create AzureAIClient: %s", e) raise - # Create HumanApprovalMagenticManager with the chat client - # Execution settings (temperature=0.1, max_tokens=4000) are configured via - # orchestration_config.create_execution_settings() which matches old SK version + # Create HumanApprovalMagenticManager with the manager agent + # New API: StandardMagenticManager takes agent as first positional argument try: manager = HumanApprovalMagenticManager( user_id=user_id, - chat_client=chat_client, - instructions=None, # Orchestrator system instructions (optional) + agent=manager_agent, # New API: pass agent instead of chat_client max_round_count=orchestration_config.max_rounds, ) cls.logger.info( @@ -132,13 +147,18 @@ async def init_orchestration( # Assemble workflow with callback storage = InMemoryCheckpointStorage() + + # New API: .participants() accepts a list of agents + participant_list = list(participants.values()) + builder = ( MagenticBuilder() - .participants(**participants) - .with_standard_manager( - manager=manager, + .participants(participant_list) + .with_manager( + manager=manager, # Pass manager instance (extends StandardMagenticManager) max_round_count=orchestration_config.max_rounds, - max_stall_count=0, + max_stall_count=3, + max_reset_count=2, ) .with_checkpointing(storage) ) @@ -204,16 +224,20 @@ async def get_current_or_new_orchestration( raise try: cls.logger.info("Initializing new orchestration for user '%s'", user_id) - orchestration_config.orchestrations[user_id] = ( - await cls.init_orchestration( - agents, team_config, team_service.memory_context, user_id - ) + print(f"[DEBUG] Initializing new orchestration for user '{user_id}'") + workflow = await cls.init_orchestration( + agents, team_config, team_service.memory_context, user_id ) + orchestration_config.orchestrations[user_id] = workflow + print(f"[DEBUG] Stored workflow for user '{user_id}': {workflow is not None}") + print(f"[DEBUG] orchestrations keys: {list(orchestration_config.orchestrations.keys())}") except Exception as e: cls.logger.error( "Failed to initialize orchestration for user '%s': %s", user_id, e ) print(f"Failed to initialize orchestration for user '{user_id}': {e}") + import traceback + traceback.print_exc() raise return orchestration_config.get_current_orchestration(user_id) @@ -229,9 +253,13 @@ async def run_orchestration(self, user_id: str, input_task) -> None: self.logger.info( "Starting orchestration job '%s' for user '%s'", job_id, user_id ) + print(f"[DEBUG] run_orchestration called for user '{user_id}'") + print(f"[DEBUG] orchestrations keys before get: {list(orchestration_config.orchestrations.keys())}") workflow = orchestration_config.get_current_orchestration(user_id) + print(f"[DEBUG] workflow is None: {workflow is None}") if workflow is None: + print(f"[ERROR] Orchestration not initialized for user '{user_id}'") raise ValueError("Orchestration not initialized for user.") # Fresh thread per participant to avoid cross-run state bleed executors = getattr(workflow, "executors", {}) @@ -305,55 +333,71 @@ async def run_orchestration(self, user_id: str, input_task) -> None: final_output: str | None = None self.logger.info("Starting workflow execution...") + last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: - # Handle orchestrator messages (task assignments, coordination) - if isinstance(event, MagenticOrchestratorMessageEvent): - message_text = getattr(event.message, "text", "") - self.logger.info(f"[ORCHESTRATOR:{event.kind}] {message_text}") - - # Handle streaming updates from agents - elif isinstance(event, MagenticAgentDeltaEvent): + # Handle orchestrator events (plan, progress ledger) + if isinstance(event, MagenticOrchestratorEvent): + self.logger.info( + "[Magentic Orchestrator Event] Type: %s", + event.event_type.name + ) + if isinstance(event.data, ChatMessage): + self.logger.info("Plan message: %s", event.data.text[:200] if event.data.text else "") + elif isinstance(event.data, MagenticProgressLedger): + self.logger.info("Progress ledger received") + + # Handle agent streaming/updates (replaces MagenticAgentDeltaEvent and MagenticAgentMessageEvent) + elif isinstance(event, AgentRunUpdateEvent): + message_id = event.data.message_id if hasattr(event.data, 'message_id') else None + executor_id = event.executor_id + + # Stream the update try: await streaming_agent_response_callback( - event.agent_id, - event, # Pass the event itself as the update object - False, # Not final yet (streaming in progress) + executor_id, + event.data, # Pass the data object + False, # Not final yet user_id, ) except Exception as e: self.logger.error( - f"Error in streaming callback for agent {event.agent_id}: {e}" + "Error in streaming callback for agent %s: %s", + executor_id, e ) + + # Track message for formatting + if message_id != last_message_id: + last_message_id = message_id - # Handle final agent messages (complete response) - elif isinstance(event, MagenticAgentMessageEvent): - if event.message: - try: - agent_response_callback( - event.agent_id, event.message, user_id - ) - except Exception as e: - self.logger.error( - f"Error in agent callback for agent {event.agent_id}: {e}" - ) - - # Handle final result from the entire workflow - elif isinstance(event, MagenticFinalResultEvent): - final_text = getattr(event.message, "text", "") + # Handle group chat request sent + elif isinstance(event, GroupChatRequestSentEvent): self.logger.info( - f"[FINAL RESULT] Length: {len(final_text)} chars" + "[REQUEST SENT (round %d)] to agent: %s", + event.round_index, + event.participant_name ) # Handle workflow output event (captures final result) elif isinstance(event, WorkflowOutputEvent): output_data = event.data + # Handle different output formats if isinstance(output_data, ChatMessage): - final_output = getattr(output_data, "text", None) or str( - output_data - ) + final_output = output_data.text or "" + elif isinstance(output_data, list): + # Handle list of ChatMessage objects + texts = [] + for item in output_data: + if isinstance(item, ChatMessage): + if item.text: + texts.append(item.text) + else: + texts.append(str(item)) + final_output = "\n".join(texts) + elif hasattr(output_data, "text"): + final_output = output_data.text or "" else: - final_output = str(output_data) + final_output = str(output_data) if output_data else "" self.logger.debug("Received workflow output event") except Exception as e: From 95b0049c87df66c5927cddd4770ab8201a0acebf Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Thu, 5 Feb 2026 22:38:09 +0530 Subject: [PATCH 02/13] v2 changes --- .../v4/magentic_agents/foundry_agent.py | 8 +- .../v4/orchestration/orchestration_manager.py | 82 +++++++++++++++++++ 2 files changed, 89 insertions(+), 1 deletion(-) diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 614904b6d..9a7c8ebe8 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -125,7 +125,9 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional Returns: AzureAIClient | None """ + print(f"[DEBUG _create_azure_search_enabled_client] Agent={self.agent_name}, chatClient={chatClient}, search_config={self.search}") if chatClient: + self.logger.info("Reusing existing chatClient for agent '%s' (already has Azure Search configured)", self.agent_name) return chatClient if not self.search: @@ -234,12 +236,16 @@ async def _after_open(self) -> None: try: chatClient = await self.get_database_team_agent() + print(f"[DEBUG _after_open] Agent={self.agent_name}, _use_azure_search={self._use_azure_search}, search_config={self.search}, chatClient={chatClient}") if self._use_azure_search: # Azure Search mode (skip MCP + Code Interpreter due to incompatibility) self.logger.info( - "Initializing agent in Azure AI Search mode (exclusive)." + "Initializing agent '%s' in Azure AI Search mode (exclusive) with index=%s.", + self.agent_name, + getattr(self.search, "index_name", "N/A") if self.search else "N/A" ) + print(f"[DEBUG _after_open] Creating Azure Search client for {self.agent_name}") chat_client = await self._create_azure_search_enabled_client(chatClient) if not chat_client: raise RuntimeError( diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index ec4f7aea1..4fc0c209a 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -16,6 +16,8 @@ InMemoryCheckpointStorage, AgentRunUpdateEvent, GroupChatRequestSentEvent, + GroupChatResponseReceivedEvent, + ExecutorCompletedEvent, MagenticOrchestratorEvent, MagenticProgressLedger, ) @@ -45,6 +47,53 @@ def __init__(self): self.user_id: Optional[str] = None self.logger = self.__class__.logger + def _extract_response_text(self, data) -> str: + """ + Extract text content from various agent_framework response types. + + Handles: + - ChatMessage: Extract .text + - AgentResponse: Extract .text + - AgentExecutorResponse: Extract from agent_response.text or full_conversation[-1].text + - List of any of the above + """ + if data is None: + return "" + + # Direct ChatMessage + if isinstance(data, ChatMessage): + return data.text or "" + + # Has .text attribute directly (AgentResponse, etc.) + if hasattr(data, "text") and data.text: + return data.text + + # AgentExecutorResponse - has agent_response and full_conversation + if hasattr(data, "agent_response"): + # Try to get text from agent_response first + agent_resp = data.agent_response + if agent_resp and hasattr(agent_resp, "text") and agent_resp.text: + return agent_resp.text + # Fallback to last message in full_conversation + if hasattr(data, "full_conversation") and data.full_conversation: + last_msg = data.full_conversation[-1] + if isinstance(last_msg, ChatMessage) and last_msg.text: + return last_msg.text + + # List of items - could be AgentExecutorResponse, ChatMessage, etc. + if isinstance(data, list) and len(data) > 0: + texts = [] + for item in data: + # Recursively extract from each item + item_text = self._extract_response_text(item) + if item_text: + texts.append(item_text) + if texts: + # Return the last non-empty response (most recent) + return texts[-1] + + return "" + # --------------------------- # Orchestration construction # --------------------------- @@ -336,6 +385,11 @@ async def run_orchestration(self, user_id: str, input_task) -> None: last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: + # Only log non-streaming events (reduce noise) + event_type_name = type(event).__name__ + if event_type_name != "AgentRunUpdateEvent": + self.logger.info("[EVENT] %s", event_type_name) + # Handle orchestrator events (plan, progress ledger) if isinstance(event, MagenticOrchestratorEvent): self.logger.info( @@ -378,6 +432,34 @@ async def run_orchestration(self, user_id: str, input_task) -> None: event.participant_name ) + # Handle group chat response received - THIS IS WHERE AGENT RESPONSES COME + elif isinstance(event, GroupChatResponseReceivedEvent): + self.logger.info( + "[RESPONSE RECEIVED (round %d)] from agent: %s", + event.round_index, + event.participant_name + ) + # Send the agent response to the UI + if event.data: + response_text = self._extract_response_text(event.data) + + if response_text: + self.logger.info("Sending agent response to UI from %s", event.participant_name) + agent_response_callback( + event.participant_name, + ChatMessage(role="assistant", text=response_text), + user_id, + ) + + # Handle executor completed - just log, don't send to UI (GroupChatResponseReceivedEvent handles that) + elif isinstance(event, ExecutorCompletedEvent): + self.logger.debug( + "[EXECUTOR COMPLETED] agent: %s", + event.executor_id + ) + # Don't send to UI here - GroupChatResponseReceivedEvent already handles agent messages + # This avoids duplicate messages + # Handle workflow output event (captures final result) elif isinstance(event, WorkflowOutputEvent): output_data = event.data From 124fcf7f1cf773282996a3b4086574ad286b9781 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Tue, 10 Feb 2026 17:34:41 +0530 Subject: [PATCH 03/13] Enhance agent orchestration and configuration for Azure AI Search integration --- src/backend/v4/api/router.py | 11 ++ .../v4/magentic_agents/common/lifecycle.py | 43 +++-- .../v4/magentic_agents/foundry_agent.py | 150 ++++++++++-------- .../v4/magentic_agents/models/agent_models.py | 6 +- .../v4/orchestration/orchestration_manager.py | 11 +- 5 files changed, 144 insertions(+), 77 deletions(-) diff --git a/src/backend/v4/api/router.py b/src/backend/v4/api/router.py index 43f3f9f2f..d9a8e7c10 100644 --- a/src/backend/v4/api/router.py +++ b/src/backend/v4/api/router.py @@ -303,6 +303,17 @@ async def process_request( ) await memory_store.add_plan(plan) + # Ensure orchestration is initialized before running + # Force rebuild for each new task since Magentic workflows cannot be reused after completion + team_service = TeamService(memory_store) + await OrchestrationManager.get_current_or_new_orchestration( + user_id=user_id, + team_config=team, + team_switched=False, + team_service=team_service, + force_rebuild=True, # Always rebuild workflow for new tasks + ) + track_event_if_configured( "PlanCreated", { diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index 88fde549b..e45d733dc 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -51,7 +51,7 @@ def __init__( self._agent: ChatAgent | None = None self.team_service: TeamService | None = team_service self.team_config: TeamConfiguration | None = team_config - self.client: Optional[AzureAIClient] = None + self.client: Optional[AgentsClient] = None self.project_endpoint = project_endpoint self.creds: Optional[DefaultAzureCredential] = None self.memory_store: Optional[DatabaseBase] = memory_store @@ -228,7 +228,25 @@ def get_agent_id(self, chat_client) -> str: return id async def get_database_team_agent(self) -> Optional[AzureAIClient]: - """Retrieve existing team agent from database, if any.""" + """Retrieve existing team agent from database, if any. + + NOTE: Agent reuse is currently DISABLED to ensure fresh agents are created + with the correct Azure AI Search configuration. + This prevents issues with stale agents that may not have the search tool configured. + + To re-enable agent reuse, set ENABLE_AGENT_REUSE=true in environment. + """ + import os + + # DISABLED: Always create fresh agents to ensure Azure AI Search tool is configured + enable_reuse = os.environ.get("ENABLE_AGENT_REUSE", "false").lower() == "true" + if not enable_reuse: + self.logger.info( + "Agent reuse DISABLED: Creating fresh agent with search tools (agent_name=%s)", + self.agent_name, + ) + return None + chat_client = None try: agent_id = await get_database_team_agent_id( @@ -251,15 +269,15 @@ async def get_database_team_agent(self) -> Optional[AzureAIClient]: ) return None - # Create client with resolved ID, preferring project_client for RAI agents + # Create client with resolved ID if self.agent_name == "RAIAgent" and self.project_client: chat_client = AzureAIClient( - project_client=self.project_client, + project_endpoint=self.project_endpoint, agent_id=resolved, credential=self.creds, ) self.logger.info( - "RAI.AgentReuseSuccess: Created AzureAIClient via Projects SDK (id=%s)", + "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s)", resolved, ) else: @@ -284,17 +302,20 @@ async def get_database_team_agent(self) -> Optional[AzureAIClient]: async def save_database_team_agent(self) -> None: """Save current team agent to database (only if truly new or changed).""" try: - if self._agent.id is None: - self.logger.error("Cannot save database team agent: agent_id is None") + if self._agent is None or self._agent.id is None: + self.logger.error("Cannot save database team agent: agent or agent_id is None") return + # Use the agent ID from ChatAgent (set during creation) + agent_id = self._agent.id + # Check if stored ID matches current ID stored_id = await get_database_team_agent_id( self.memory_store, self.team_config, self.agent_name ) - if stored_id == self._agent.chat_client.agent_id: + if stored_id == agent_id: self.logger.info( - "RAI reuse: id unchanged (id=%s); skip save.", self._agent.id + "RAI reuse: id unchanged (id=%s); skip save.", agent_id ) return @@ -302,7 +323,7 @@ async def save_database_team_agent(self) -> None: team_id=self.team_config.team_id, team_name=self.team_config.name, agent_name=self.agent_name, - agent_foundry_id=self._agent.chat_client.agent_id, + agent_foundry_id=agent_id, agent_description=self.agent_description, agent_instructions=self.agent_instructions, ) @@ -310,7 +331,7 @@ async def save_database_team_agent(self) -> None: self.logger.info( "Saved team agent to database (agent_name=%s, id=%s)", self.agent_name, - self._agent.id, + agent_id, ) except Exception as ex: diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 9a7c8ebe8..d706af865 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -7,7 +7,12 @@ Role) from agent_framework_azure_ai import \ AzureAIClient # Provided by agent_framework -from azure.ai.projects.models import ConnectionType +from azure.ai.projects.models import ( + PromptAgentDefinition, + AzureAISearchAgentTool, + AzureAISearchToolResource, + AISearchIndexResource, +) from common.config.app_config import config from common.database.database_base import DatabaseBase from common.models.messages_af import TeamConfiguration @@ -65,19 +70,23 @@ def __init__( self._use_azure_search = self._is_azure_search_requested() self.use_reasoning = use_reasoning - # Placeholder for server-created Azure AI agent id (if Azure Search path) + # Placeholder for server-created Azure AI agent id/version (if Azure Search path) self._azure_server_agent_id: Optional[str] = None + self._azure_server_agent_version: Optional[str] = None # ------------------------- # Mode detection # ------------------------- def _is_azure_search_requested(self) -> bool: """Determine if Azure AI Search raw tool path should be used.""" + print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}, search={self.search}") if not self.search: + print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}: No search config, returning False") return False # Minimal heuristic: presence of required attributes has_index = hasattr(self.search, "index_name") and bool(self.search.index_name) + print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}: has_index={has_index}, index_name={getattr(self.search, 'index_name', None)}") if has_index: self.logger.info( "Azure AI Search requested (connection_id=%s, index=%s).", @@ -113,14 +122,17 @@ async def _collect_tools(self) -> List: # ------------------------- async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional[AzureAIClient]: """ - Create a server-side Azure AI agent with Azure AI Search raw tool. + Create a server-side Azure AI agent with Azure AI Search tool using create_version. + + This uses the AIProjectClient.agents.create_version() approach with: + - PromptAgentDefinition for agent configuration + - AzureAISearchAgentTool with AzureAISearchToolResource for search capability + - AISearchIndexResource for index configuration with project_connection_id Requirements: - - An Azure AI Project Connection (type=AZURE_AI_SEARCH) that contains either: - a) API key + endpoint, OR - b) Managed Identity (RBAC enabled on the Search service with Search Service Contributor + Search Index Data Reader). + - An Azure AI Project Connection for Azure AI Search - search_config.index_name must exist in the Search service. - + - search_config.connection_name should match the AI Project connection name Returns: AzureAIClient | None @@ -134,9 +146,16 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional self.logger.error("Search configuration missing.") return None - desired_connection_name = getattr(self.search, "connection_name", None) + # Get connection name - this is used as project_connection_id in create_version + connection_name = getattr(self.search, "connection_name", None) + if not connection_name: + # Fallback to environment variable + connection_name = config.AZURE_AI_SEARCH_CONNECTION_NAME + self.logger.info("Using connection_name from environment: %s", connection_name) + index_name = getattr(self.search, "index_name", "") query_type = getattr(self.search, "search_query_type", "simple") + top_k = getattr(self.search, "top_k", 5) if not index_name: self.logger.error( @@ -144,82 +163,89 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional ) return None - resolved_connection_id = None - - try: - async for connection in self.project_client.connections.list(): - if connection.type == ConnectionType.AZURE_AI_SEARCH: - - if ( - desired_connection_name - and connection.name == desired_connection_name - ): - resolved_connection_id = connection.id - break - # Fallback: if no specific connection requested and none resolved yet, take the first - if not desired_connection_name and not resolved_connection_id: - resolved_connection_id = connection.id - # Do not break yet; we log but allow chance to find a name match later. If not, this stays. - - if not resolved_connection_id: - self.logger.error( - "No Azure AI Search connection resolved. " "connection_name=%s", - desired_connection_name, - ) - # return None - - self.logger.info( - "Using Azure AI Search connection (id=%s, requested_name=%s).", - resolved_connection_id, - desired_connection_name, + if not connection_name: + self.logger.error( + "connection_name not provided; aborting Azure Search path." ) - except Exception as ex: - self.logger.error("Failed to enumerate connections: %s", ex) return None - # Create agent with raw tool + self.logger.info( + "Creating Azure AI Search agent with create_version: connection_name=%s, index=%s, query_type=%s, top_k=%s", + connection_name, + index_name, + query_type, + top_k, + ) + + # Create agent using create_version with PromptAgentDefinition and AzureAISearchAgentTool + # This approach matches the Knowledge Mining Solution Accelerator pattern try: - azure_agent = await self.client.create_agent( - model=self.model_deployment_name, - name=self.agent_name, - instructions=( - f"{self.agent_instructions} " - "Always use the Azure AI Search tool and configured index for knowledge retrieval." + enhanced_instructions = ( + f"{self.agent_instructions} " + "Always use the Azure AI Search tool and configured index for knowledge retrieval." + ) + + print(f"[DEBUG] Creating agent '{self.agent_name}' with instructions (first 200 chars): {enhanced_instructions[:200]}...") + print(f"[DEBUG] Agent model: {self.model_deployment_name}") + print(f"[DEBUG] Search config: connection={connection_name}, index={index_name}, query_type={query_type}, top_k={top_k}") + + azure_agent = await self.project_client.agents.create_version( + agent_name=self.agent_name, + definition=PromptAgentDefinition( + model=self.model_deployment_name, + instructions=enhanced_instructions, + tools=[ + AzureAISearchAgentTool( + azure_ai_search=AzureAISearchToolResource( + indexes=[ + AISearchIndexResource( + project_connection_id=connection_name, + index_name=index_name, + query_type=query_type, + top_k=top_k, + ) + ] + ) + ) + ], ), - tools=[{"type": "azure_ai_search"}], - tool_resources={ - "azure_ai_search": { - "indexes": [ - { - "index_connection_id": resolved_connection_id, - "index_name": index_name, - "query_type": query_type, - } - ] - } - }, ) + self._azure_server_agent_id = azure_agent.id + self._azure_server_agent_version = azure_agent.version self.logger.info( - "Created Azure server agent with Azure AI Search tool (agent_id=%s, index=%s, query_type=%s).", + "Created Azure AI Search agent via create_version (name=%s, id=%s, version=%s, connection=%s, index=%s, query_type=%s, top_k=%s).", + azure_agent.name, azure_agent.id, + azure_agent.version, + connection_name, index_name, query_type, + top_k, ) + print(f"[DEBUG] Created agent via create_version: name={azure_agent.name}, id={azure_agent.id}, version={azure_agent.version}") + print(f"[DEBUG] Agent definition: {azure_agent.definition}") + print(f"[DEBUG] Agent instructions from definition: {getattr(azure_agent.definition, 'instructions', 'N/A')}") + # Wrap in AzureAIClient using agent_name and agent_version (NOT agent_id) + # AzureAIClient constructor: agent_name, agent_version, project_endpoint, credential chat_client = AzureAIClient( - project_client=self.project_client, - agent_id=azure_agent.id, + project_endpoint=self.project_endpoint, + agent_name=azure_agent.name, + agent_version=azure_agent.version, # Use the specific version we just created credential=self.creds, ) return chat_client + except Exception as ex: self.logger.error( - "Failed to create Azure Search enabled agent (connection_id=%s, index=%s): %s", - resolved_connection_id, + "Failed to create Azure Search enabled agent via create_version (connection=%s, index=%s): %s", + connection_name, index_name, ex, ) + import traceback + traceback.print_exc() return None # ------------------------- diff --git a/src/backend/v4/magentic_agents/models/agent_models.py b/src/backend/v4/magentic_agents/models/agent_models.py index 5c6a3f2f1..4e10270fa 100644 --- a/src/backend/v4/magentic_agents/models/agent_models.py +++ b/src/backend/v4/magentic_agents/models/agent_models.py @@ -43,6 +43,8 @@ class SearchConfig: connection_name: str | None = None endpoint: str | None = None index_name: str | None = None + search_query_type: str = "simple" # Options: "simple", "vector_simple", "vector", "semantic", "hybrid" + top_k: int = 5 # Number of results to return @classmethod def from_env(cls, index_name: str) -> "SearchConfig": @@ -58,5 +60,7 @@ def from_env(cls, index_name: str) -> "SearchConfig": return cls( connection_name=connection_name, endpoint=endpoint, - index_name=index_name + index_name=index_name, + search_query_type="simple", # Use simple query type (keyword search) + top_k=5 ) diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 4fc0c209a..5a27c59b0 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -231,17 +231,22 @@ async def get_current_or_new_orchestration( team_config: TeamConfiguration, team_switched: bool, team_service: TeamService = None, + force_rebuild: bool = False, ): """ Return an existing workflow for the user or create a new one if: - None exists - Team switched flag is True + - force_rebuild is True (for new tasks after workflow completion) """ current = orchestration_config.get_current_orchestration(user_id) - if current is None or team_switched: - if current is not None and team_switched: + needs_rebuild = current is None or team_switched or force_rebuild + + if needs_rebuild: + if current is not None and (team_switched or force_rebuild): + reason = "team switched" if team_switched else "force rebuild for new task" cls.logger.info( - "Team switched, closing previous agents for user '%s'", user_id + "Rebuilding orchestration for user '%s' (reason: %s)", user_id, reason ) # Close prior agents (same logic as old version) for agent in getattr(current, "_participants", {}).values(): From 1bc5725bd4e0015cc42ebcd13242495b84edef15 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Wed, 11 Feb 2026 17:45:02 +0530 Subject: [PATCH 04/13] fix for agent multiple times calling --- .../v4/magentic_agents/foundry_agent.py | 18 ++++------ .../magentic_agents/magentic_agent_factory.py | 1 + .../orchestration/human_approval_manager.py | 20 ++++++++++- .../v4/orchestration/orchestration_manager.py | 34 +++++++++++++++---- 4 files changed, 54 insertions(+), 19 deletions(-) diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index d706af865..84297ebef 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -185,12 +185,12 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional "Always use the Azure AI Search tool and configured index for knowledge retrieval." ) - print(f"[DEBUG] Creating agent '{self.agent_name}' with instructions (first 200 chars): {enhanced_instructions[:200]}...") - print(f"[DEBUG] Agent model: {self.model_deployment_name}") - print(f"[DEBUG] Search config: connection={connection_name}, index={index_name}, query_type={query_type}, top_k={top_k}") + print(f"[AGENT CREATE] 🆕 Creating agent in Foundry: '{self.agent_name}'", flush=True) + print(f"[AGENT CREATE] Model: {self.model_deployment_name}", flush=True) + print(f"[AGENT CREATE] Search: connection={connection_name}, index={index_name}", flush=True) azure_agent = await self.project_client.agents.create_version( - agent_name=self.agent_name, + agent_name=self.agent_name, # Use original name definition=PromptAgentDefinition( model=self.model_deployment_name, instructions=enhanced_instructions, @@ -213,19 +213,13 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional self._azure_server_agent_id = azure_agent.id self._azure_server_agent_version = azure_agent.version + print(f"[AGENT CREATE] ✅ Created agent: name={azure_agent.name}, id={azure_agent.id}, version={azure_agent.version}", flush=True) self.logger.info( - "Created Azure AI Search agent via create_version (name=%s, id=%s, version=%s, connection=%s, index=%s, query_type=%s, top_k=%s).", + "Created Azure AI Search agent via create_version (name=%s, id=%s, version=%s).", azure_agent.name, azure_agent.id, azure_agent.version, - connection_name, - index_name, - query_type, - top_k, ) - print(f"[DEBUG] Created agent via create_version: name={azure_agent.name}, id={azure_agent.id}, version={azure_agent.version}") - print(f"[DEBUG] Agent definition: {azure_agent.definition}") - print(f"[DEBUG] Agent instructions from definition: {getattr(azure_agent.definition, 'instructions', 'N/A')}") # Wrap in AzureAIClient using agent_name and agent_version (NOT agent_id) # AzureAIClient constructor: agent_name, agent_version, project_endpoint, credential diff --git a/src/backend/v4/magentic_agents/magentic_agent_factory.py b/src/backend/v4/magentic_agents/magentic_agent_factory.py index 36544166d..3eafb5831 100644 --- a/src/backend/v4/magentic_agents/magentic_agent_factory.py +++ b/src/backend/v4/magentic_agents/magentic_agent_factory.py @@ -115,6 +115,7 @@ async def create_agent_from_config( index_name, "Reasoning" if use_reasoning else "Foundry", ) + print(f"[FACTORY] 🆕 Creating NEW agent: {agent_obj.name} (id={id(agent_obj)})", flush=True) agent = FoundryAgentTemplate( agent_name=agent_obj.name, diff --git a/src/backend/v4/orchestration/human_approval_manager.py b/src/backend/v4/orchestration/human_approval_manager.py index 654d72a23..39089247a 100644 --- a/src/backend/v4/orchestration/human_approval_manager.py +++ b/src/backend/v4/orchestration/human_approval_manager.py @@ -33,6 +33,7 @@ class HumanApprovalMagenticManager(StandardMagenticManager): approval_enabled: bool = True magentic_plan: Optional[MPlan] = None current_user_id: str # populated in __init__ + _called_agents: set # Track which agents have been called def __init__(self, user_id: str, agent, *args, **kwargs): """ @@ -43,6 +44,9 @@ def __init__(self, user_id: str, agent, *args, **kwargs): *args: Additional positional arguments for the parent StandardMagenticManager. **kwargs: Additional keyword arguments for the parent StandardMagenticManager. """ + + # Initialize called agents tracker + self._called_agents = set() plan_append = """ @@ -55,6 +59,9 @@ def __init__(self, user_id: str, agent, *args, **kwargs): to be taken. If a step involves multiple actions, separate them into distinct steps with an agent included in each step. If the step is taken by an agent that is not part of the team, such as the MagenticManager, please always list the MagenticManager as the agent for that step. At any time, if more information is needed from the user, use the ProxyAgent to request this information. +CRITICAL: Each agent should only be called ONCE to perform their task. Do NOT call the same agent multiple times. +After an agent has provided their response, move on to the next agent in the plan. + Here is an example of a well-structured plan: - **EnhancedResearchAgent** to gather authoritative data on the latest industry trends and best practices in employee onboarding - **EnhancedResearchAgent** to gather authoritative data on Innovative onboarding techniques that enhance new hire engagement and retention. @@ -62,6 +69,13 @@ def __init__(self, user_id: str, agent, *args, **kwargs): - **DocumentCreationAgent** to draft a comprehensive onboarding plan that includes a checklist of resources and materials needed for effective onboarding. - **ProxyAgent** to review the drafted onboarding plan for clarity and completeness. - **MagenticManager** to finalize the onboarding plan and prepare it for presentation to stakeholders. +""" + + # Add progress ledger prompt to prevent re-calling agents + progress_append = """ +CRITICAL RULE: DO NOT call the same agent more than once unless absolutely necessary. +If an agent has already provided a response, consider their task COMPLETE and move to the next agent. +Only re-call an agent if their previous response was explicitly an error or failure. """ final_append = """ @@ -75,6 +89,10 @@ def __init__(self, user_id: str, agent, *args, **kwargs): ORCHESTRATOR_TASK_LEDGER_PLAN_UPDATE_PROMPT + plan_append ) kwargs["final_answer_prompt"] = ORCHESTRATOR_FINAL_ANSWER_PROMPT + final_append + + # Override progress ledger prompt to discourage re-calling agents + from agent_framework._workflows._magentic import ORCHESTRATOR_PROGRESS_LEDGER_PROMPT + kwargs["progress_ledger_prompt"] = ORCHESTRATOR_PROGRESS_LEDGER_PROMPT + progress_append self.current_user_id = user_id # New API: StandardMagenticManager takes agent as first positional argument @@ -305,4 +323,4 @@ def plan_to_obj(self, magentic_context: MagenticContext, ledger) -> MPlan: task=task_text, ) - return return_plan + return return_plan \ No newline at end of file diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 5a27c59b0..58fc507e1 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -197,17 +197,19 @@ async def init_orchestration( # Assemble workflow with callback storage = InMemoryCheckpointStorage() - # New API: .participants() accepts a list of agents + # New SDK: participants() accepts a Sequence (list) of agents + # The orchestrator uses agent.name to identify them participant_list = list(participants.values()) + cls.logger.info("Participants for workflow: %s", list(participants.keys())) + print(f"[DEBUG] Participants for workflow: {list(participants.keys())}", flush=True) builder = ( MagenticBuilder() - .participants(participant_list) + .participants(participant_list) # New SDK: pass as list .with_manager( manager=manager, # Pass manager instance (extends StandardMagenticManager) max_round_count=orchestration_config.max_rounds, - max_stall_count=3, - max_reset_count=2, + max_stall_count=0, # CRITICAL: Prevent re-calling agents when stalled (default is 3!) ) .with_checkpointing(storage) ) @@ -381,12 +383,16 @@ async def run_orchestration(self, user_id: str, input_task) -> None: task_text = getattr(input_task, "description", str(input_task)) self.logger.debug("Task: %s", task_text) + # Track how many times each agent is called (for debugging duplicate calls) + agent_call_counts: dict = {} + try: # Execute workflow using run_stream with task as positional parameter # The execution settings are configured in the manager/client final_output: str | None = None self.logger.info("Starting workflow execution...") + print(f"[ORCHESTRATOR] 🚀 Starting workflow with max_rounds={orchestration_config.max_rounds}", flush=True) last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: @@ -431,11 +437,20 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Handle group chat request sent elif isinstance(event, GroupChatRequestSentEvent): + agent_name = event.participant_name + agent_call_counts[agent_name] = agent_call_counts.get(agent_name, 0) + 1 + call_num = agent_call_counts[agent_name] + self.logger.info( - "[REQUEST SENT (round %d)] to agent: %s", + "[REQUEST SENT (round %d)] to agent: %s (call #%d)", event.round_index, - event.participant_name + agent_name, + call_num ) + print(f"[ORCHESTRATOR] 📤 REQUEST SENT round={event.round_index} to agent={agent_name} (call #{call_num})", flush=True) + + if call_num > 1: + print(f"[ORCHESTRATOR] ⚠️ WARNING: Agent '{agent_name}' called {call_num} times!", flush=True) # Handle group chat response received - THIS IS WHERE AGENT RESPONSES COME elif isinstance(event, GroupChatResponseReceivedEvent): @@ -496,6 +511,13 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Extract final result final_text = final_output if final_output else "" + # Log agent call summary + print(f"\n[ORCHESTRATOR] 📊 AGENT CALL SUMMARY:", flush=True) + for agent_name, count in agent_call_counts.items(): + status = "✅" if count == 1 else "⚠️ DUPLICATE" + print(f" {status} {agent_name}: called {count} time(s)", flush=True) + self.logger.info("Agent call counts: %s", agent_call_counts) + # Log results self.logger.info("\nAgent responses:") self.logger.info( From 45582e15fedbfb6e850a554329800b138ebe4272 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 13 Feb 2026 09:15:56 +0530 Subject: [PATCH 05/13] Enhance get_chat_client to use latest agent version and log agent name --- src/backend/v4/magentic_agents/common/lifecycle.py | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index e45d733dc..af9dcb846 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -149,7 +149,10 @@ async def _after_open(self) -> None: raise NotImplementedError def get_chat_client(self, chat_client) -> AzureAIClient: - """Return the underlying ChatClientProtocol (AzureAIClient).""" + """Return the underlying ChatClientProtocol (AzureAIClient). + + Uses agent_name with use_latest_version=True to get the latest agent version + """ if chat_client: return chat_client if ( @@ -159,11 +162,14 @@ def get_chat_client(self, chat_client) -> AzureAIClient: return self._agent.chat_client # type: ignore chat_client = AzureAIClient( project_endpoint=self.project_endpoint, + agent_name=self.agent_name, model_deployment_name=self.model_deployment_name, credential=self.creds, + use_latest_version=True, ) self.logger.info( - "Created new AzureAIClient for get chat client", + "Created new AzureAIClient (agent_name=%s, use_latest_version=True)", + self.agent_name, ) return chat_client From 3a4dfbb3bf0e2a0bab58968f0c0000fa46c13423 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 13 Feb 2026 11:36:18 +0530 Subject: [PATCH 06/13] Update dependency versions in pyproject.toml and uv.lock for consistency --- src/backend/pyproject.toml | 18 +++++++++--------- src/backend/uv.lock | 18 +++++++++--------- 2 files changed, 18 insertions(+), 18 deletions(-) diff --git a/src/backend/pyproject.toml b/src/backend/pyproject.toml index 7750638dd..f5bd8fc6f 100644 --- a/src/backend/pyproject.toml +++ b/src/backend/pyproject.toml @@ -11,16 +11,16 @@ dependencies = [ "azure-cosmos==4.9.0", "azure-identity==1.24.0", "azure-monitor-events-extension==0.1.0", - "azure-monitor-opentelemetry>=1.8.0", + "azure-monitor-opentelemetry==1.8.5", "azure-search-documents==11.5.3", "fastapi==0.116.1", - "openai>=2.8.0", - "opentelemetry-api>=1.39.0", - "opentelemetry-exporter-otlp-proto-grpc>=1.39.0", - "opentelemetry-exporter-otlp-proto-http>=1.39.0", - "opentelemetry-instrumentation-fastapi>=0.57b0", - "opentelemetry-instrumentation-openai>=0.46.2", - "opentelemetry-sdk>=1.39.0", + "openai==2.16.0", + "opentelemetry-api==1.39.0", + "opentelemetry-exporter-otlp-proto-grpc==1.39.0", + "opentelemetry-exporter-otlp-proto-http==1.39.0", + "opentelemetry-instrumentation-fastapi==0.60b0", + "opentelemetry-instrumentation-openai==0.46.2", + "opentelemetry-sdk==1.39.0", "pytest==8.4.1", "pytest-asyncio==0.24.0", "pytest-cov==5.0.0", @@ -30,7 +30,7 @@ dependencies = [ "uvicorn==0.35.0", "pylint-pydantic==0.3.5", "pexpect==4.9.0", - "mcp>=1.24.0,<2", + "mcp==1.26.0", "agent-framework-azure-ai==1.0.0b260130", "agent-framework-core==1.0.0b260130" ] \ No newline at end of file diff --git a/src/backend/uv.lock b/src/backend/uv.lock index 526fe789a..3e7bbae7e 100644 --- a/src/backend/uv.lock +++ b/src/backend/uv.lock @@ -557,17 +557,17 @@ requires-dist = [ { name = "azure-cosmos", specifier = "==4.9.0" }, { name = "azure-identity", specifier = "==1.24.0" }, { name = "azure-monitor-events-extension", specifier = "==0.1.0" }, - { name = "azure-monitor-opentelemetry", specifier = ">=1.8.0" }, + { name = "azure-monitor-opentelemetry", specifier = "==1.8.5" }, { name = "azure-search-documents", specifier = "==11.5.3" }, { name = "fastapi", specifier = "==0.116.1" }, - { name = "mcp", specifier = ">=1.24.0,<2" }, - { name = "openai", specifier = ">=2.8.0" }, - { name = "opentelemetry-api", specifier = ">=1.39.0" }, - { name = "opentelemetry-exporter-otlp-proto-grpc", specifier = ">=1.39.0" }, - { name = "opentelemetry-exporter-otlp-proto-http", specifier = ">=1.39.0" }, - { name = "opentelemetry-instrumentation-fastapi", specifier = ">=0.57b0" }, - { name = "opentelemetry-instrumentation-openai", specifier = ">=0.46.2" }, - { name = "opentelemetry-sdk", specifier = ">=1.39.0" }, + { name = "mcp", specifier = "==1.26.0" }, + { name = "openai", specifier = "==2.16.0" }, + { name = "opentelemetry-api", specifier = "==1.39.0" }, + { name = "opentelemetry-exporter-otlp-proto-grpc", specifier = "==1.39.0" }, + { name = "opentelemetry-exporter-otlp-proto-http", specifier = "==1.39.0" }, + { name = "opentelemetry-instrumentation-fastapi", specifier = "==0.60b0" }, + { name = "opentelemetry-instrumentation-openai", specifier = "==0.46.2" }, + { name = "opentelemetry-sdk", specifier = "==1.39.0" }, { name = "pexpect", specifier = "==4.9.0" }, { name = "pylint-pydantic", specifier = "==0.3.5" }, { name = "pytest", specifier = "==8.4.1" }, From c513f4bde39bc162312a31fbb21b349df010d640 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 13 Feb 2026 17:14:32 +0530 Subject: [PATCH 07/13] Add user state cleanup and enhance AzureAIClient initialization with deployment name fallback --- src/backend/v4/config/settings.py | 20 ++++++ .../v4/magentic_agents/common/lifecycle.py | 15 ++++- .../v4/magentic_agents/foundry_agent.py | 4 +- .../v4/orchestration/orchestration_manager.py | 65 +++++++++---------- 4 files changed, 64 insertions(+), 40 deletions(-) diff --git a/src/backend/v4/config/settings.py b/src/backend/v4/config/settings.py index fa112fcd9..2a9687272 100644 --- a/src/backend/v4/config/settings.py +++ b/src/backend/v4/config/settings.py @@ -220,6 +220,26 @@ def cleanup_clarification(self, request_id: str) -> None: self.clarifications.pop(request_id, None) self._clarification_events.pop(request_id, None) + def cleanup_user_state(self, user_id: str) -> None: + """Clean up all state for a user to prevent cross-scenario leakage. + + This removes any pending approvals, clarifications, and plans + associated with the user to ensure fresh state for new runs. + """ + # Clean up any plans associated with this user + plans_to_remove = [ + plan_id for plan_id, plan in self.plans.items() + if getattr(plan, 'user_id', None) == user_id + ] + for plan_id in plans_to_remove: + self.plans.pop(plan_id, None) + self.cleanup_approval(plan_id) + + # Clean up any pending approvals/clarifications for this user + # Note: We can't easily map approvals to users without plan context, + # so this primarily clears the plans and their associated approvals + logger.debug("Cleaned up state for user %s (removed %d plans)", user_id, len(plans_to_remove)) + class ConnectionConfig: """Connection manager for WebSocket connections.""" diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index af9dcb846..c143d0c45 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -14,6 +14,7 @@ from agent_framework_azure_ai import AzureAIClient from azure.ai.agents.aio import AgentsClient from azure.identity.aio import DefaultAzureCredential +from common.config.app_config import config from common.database.database_base import DatabaseBase from common.models.messages_af import CurrentTeamAgent, TeamConfiguration from common.utils.utils_agents import ( @@ -160,10 +161,12 @@ def get_chat_client(self, chat_client) -> AzureAIClient: and self._agent.chat_client ): return self._agent.chat_client # type: ignore + # Use model_deployment_name with fallback to default model if empty + deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_name=self.agent_name, - model_deployment_name=self.model_deployment_name, + model_deployment_name=deployment_name, credential=self.creds, use_latest_version=True, ) @@ -277,20 +280,26 @@ async def get_database_team_agent(self) -> Optional[AzureAIClient]: # Create client with resolved ID if self.agent_name == "RAIAgent" and self.project_client: + # Use RAI deployment name for RAI agents + rai_deployment = config.AZURE_OPENAI_RAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_id=resolved, + model_deployment_name=rai_deployment, credential=self.creds, ) self.logger.info( - "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s)", + "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s, model=%s)", resolved, + rai_deployment, ) else: + # Use model_deployment_name with fallback to default model if empty + deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_id=resolved, - model_deployment_name=self.model_deployment_name, + model_deployment_name=deployment_name, credential=self.creds, ) self.logger.info( diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 84297ebef..3a95ea210 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -222,11 +222,13 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional ) # Wrap in AzureAIClient using agent_name and agent_version (NOT agent_id) - # AzureAIClient constructor: agent_name, agent_version, project_endpoint, credential + # Include model_deployment_name to ensure SDK has model info for streaming + deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_name=azure_agent.name, agent_version=azure_agent.version, # Use the specific version we just created + model_deployment_name=deployment_name, credential=self.creds, ) return chat_client diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 58fc507e1..12bd70ec8 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -133,9 +133,11 @@ async def init_orchestration( try: # Create the chat client (AzureAIClient) + # Use team deployment_name with fallback to default model if empty + deployment_name = team_config.deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=config.AZURE_AI_PROJECT_ENDPOINT, - model_deployment_name=team_config.deployment_name, + model_deployment_name=deployment_name, agent_name=agent_name, credential=credential, ) @@ -150,7 +152,7 @@ async def init_orchestration( cls.logger.info( "Created AzureAIClient and manager ChatAgent for orchestration with model '%s' at endpoint '%s'", - team_config.deployment_name, + deployment_name, config.AZURE_AI_PROJECT_ENDPOINT, ) except Exception as e: @@ -197,19 +199,17 @@ async def init_orchestration( # Assemble workflow with callback storage = InMemoryCheckpointStorage() - # New SDK: participants() accepts a Sequence (list) of agents - # The orchestrator uses agent.name to identify them + # New API: .participants() accepts a list of agents participant_list = list(participants.values()) - cls.logger.info("Participants for workflow: %s", list(participants.keys())) - print(f"[DEBUG] Participants for workflow: {list(participants.keys())}", flush=True) builder = ( MagenticBuilder() - .participants(participant_list) # New SDK: pass as list + .participants(participant_list) .with_manager( manager=manager, # Pass manager instance (extends StandardMagenticManager) max_round_count=orchestration_config.max_rounds, - max_stall_count=0, # CRITICAL: Prevent re-calling agents when stalled (default is 3!) + max_stall_count=3, + max_reset_count=2, ) .with_checkpointing(storage) ) @@ -239,16 +239,14 @@ async def get_current_or_new_orchestration( Return an existing workflow for the user or create a new one if: - None exists - Team switched flag is True - - force_rebuild is True (for new tasks after workflow completion) + - force_rebuild is True (for new tasks that need fresh workflow) """ current = orchestration_config.get_current_orchestration(user_id) - needs_rebuild = current is None or team_switched or force_rebuild - - if needs_rebuild: + if current is None or team_switched or force_rebuild: if current is not None and (team_switched or force_rebuild): - reason = "team switched" if team_switched else "force rebuild for new task" + reason = "team switched" if team_switched else "force rebuild" cls.logger.info( - "Rebuilding orchestration for user '%s' (reason: %s)", user_id, reason + "Closing previous agents for user '%s' (reason: %s)", user_id, reason ) # Close prior agents (same logic as old version) for agent in getattr(current, "_participants", {}).values(): @@ -305,6 +303,11 @@ async def run_orchestration(self, user_id: str, input_task) -> None: Execute the Magentic workflow for the provided user and task description. """ job_id = str(uuid.uuid4()) + + # Clean up any accumulated state from previous runs (cancelled plans, etc.) + # This prevents cross-scenario leakage + orchestration_config.cleanup_user_state(user_id) + orchestration_config.set_approval_pending(job_id) self.logger.info( "Starting orchestration job '%s' for user '%s'", job_id, user_id @@ -317,6 +320,16 @@ async def run_orchestration(self, user_id: str, input_task) -> None: if workflow is None: print(f"[ERROR] Orchestration not initialized for user '{user_id}'") raise ValueError("Orchestration not initialized for user.") + + # Reset manager's plan state to prevent leakage from cancelled plans + manager = getattr(workflow, "_manager", None) + if manager and hasattr(manager, "magentic_plan"): + manager.magentic_plan = None + self.logger.debug("Reset manager's magentic_plan for fresh run") + if manager and hasattr(manager, "task_ledger"): + manager.task_ledger = None + self.logger.debug("Reset manager's task_ledger for fresh run") + # Fresh thread per participant to avoid cross-run state bleed executors = getattr(workflow, "executors", {}) self.logger.debug("Executor keys at run start: %s", list(executors.keys())) @@ -383,16 +396,12 @@ async def run_orchestration(self, user_id: str, input_task) -> None: task_text = getattr(input_task, "description", str(input_task)) self.logger.debug("Task: %s", task_text) - # Track how many times each agent is called (for debugging duplicate calls) - agent_call_counts: dict = {} - try: # Execute workflow using run_stream with task as positional parameter # The execution settings are configured in the manager/client final_output: str | None = None self.logger.info("Starting workflow execution...") - print(f"[ORCHESTRATOR] 🚀 Starting workflow with max_rounds={orchestration_config.max_rounds}", flush=True) last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: @@ -437,20 +446,11 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Handle group chat request sent elif isinstance(event, GroupChatRequestSentEvent): - agent_name = event.participant_name - agent_call_counts[agent_name] = agent_call_counts.get(agent_name, 0) + 1 - call_num = agent_call_counts[agent_name] - self.logger.info( - "[REQUEST SENT (round %d)] to agent: %s (call #%d)", + "[REQUEST SENT (round %d)] to agent: %s", event.round_index, - agent_name, - call_num + event.participant_name ) - print(f"[ORCHESTRATOR] 📤 REQUEST SENT round={event.round_index} to agent={agent_name} (call #{call_num})", flush=True) - - if call_num > 1: - print(f"[ORCHESTRATOR] ⚠️ WARNING: Agent '{agent_name}' called {call_num} times!", flush=True) # Handle group chat response received - THIS IS WHERE AGENT RESPONSES COME elif isinstance(event, GroupChatResponseReceivedEvent): @@ -511,13 +511,6 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Extract final result final_text = final_output if final_output else "" - # Log agent call summary - print(f"\n[ORCHESTRATOR] 📊 AGENT CALL SUMMARY:", flush=True) - for agent_name, count in agent_call_counts.items(): - status = "✅" if count == 1 else "⚠️ DUPLICATE" - print(f" {status} {agent_name}: called {count} time(s)", flush=True) - self.logger.info("Agent call counts: %s", agent_call_counts) - # Log results self.logger.info("\nAgent responses:") self.logger.info( From f2eb8e38c892faf8966804d0629ea11f2b79f753 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 13 Feb 2026 17:31:46 +0530 Subject: [PATCH 08/13] Revert "Add user state cleanup and enhance AzureAIClient initialization with deployment name fallback" This reverts commit c513f4bde39bc162312a31fbb21b349df010d640. --- src/backend/v4/config/settings.py | 20 ------ .../v4/magentic_agents/common/lifecycle.py | 15 +---- .../v4/magentic_agents/foundry_agent.py | 4 +- .../v4/orchestration/orchestration_manager.py | 65 ++++++++++--------- 4 files changed, 40 insertions(+), 64 deletions(-) diff --git a/src/backend/v4/config/settings.py b/src/backend/v4/config/settings.py index 2a9687272..fa112fcd9 100644 --- a/src/backend/v4/config/settings.py +++ b/src/backend/v4/config/settings.py @@ -220,26 +220,6 @@ def cleanup_clarification(self, request_id: str) -> None: self.clarifications.pop(request_id, None) self._clarification_events.pop(request_id, None) - def cleanup_user_state(self, user_id: str) -> None: - """Clean up all state for a user to prevent cross-scenario leakage. - - This removes any pending approvals, clarifications, and plans - associated with the user to ensure fresh state for new runs. - """ - # Clean up any plans associated with this user - plans_to_remove = [ - plan_id for plan_id, plan in self.plans.items() - if getattr(plan, 'user_id', None) == user_id - ] - for plan_id in plans_to_remove: - self.plans.pop(plan_id, None) - self.cleanup_approval(plan_id) - - # Clean up any pending approvals/clarifications for this user - # Note: We can't easily map approvals to users without plan context, - # so this primarily clears the plans and their associated approvals - logger.debug("Cleaned up state for user %s (removed %d plans)", user_id, len(plans_to_remove)) - class ConnectionConfig: """Connection manager for WebSocket connections.""" diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index c143d0c45..af9dcb846 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -14,7 +14,6 @@ from agent_framework_azure_ai import AzureAIClient from azure.ai.agents.aio import AgentsClient from azure.identity.aio import DefaultAzureCredential -from common.config.app_config import config from common.database.database_base import DatabaseBase from common.models.messages_af import CurrentTeamAgent, TeamConfiguration from common.utils.utils_agents import ( @@ -161,12 +160,10 @@ def get_chat_client(self, chat_client) -> AzureAIClient: and self._agent.chat_client ): return self._agent.chat_client # type: ignore - # Use model_deployment_name with fallback to default model if empty - deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_name=self.agent_name, - model_deployment_name=deployment_name, + model_deployment_name=self.model_deployment_name, credential=self.creds, use_latest_version=True, ) @@ -280,26 +277,20 @@ async def get_database_team_agent(self) -> Optional[AzureAIClient]: # Create client with resolved ID if self.agent_name == "RAIAgent" and self.project_client: - # Use RAI deployment name for RAI agents - rai_deployment = config.AZURE_OPENAI_RAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_id=resolved, - model_deployment_name=rai_deployment, credential=self.creds, ) self.logger.info( - "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s, model=%s)", + "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s)", resolved, - rai_deployment, ) else: - # Use model_deployment_name with fallback to default model if empty - deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_id=resolved, - model_deployment_name=deployment_name, + model_deployment_name=self.model_deployment_name, credential=self.creds, ) self.logger.info( diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 3a95ea210..84297ebef 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -222,13 +222,11 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional ) # Wrap in AzureAIClient using agent_name and agent_version (NOT agent_id) - # Include model_deployment_name to ensure SDK has model info for streaming - deployment_name = self.model_deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME + # AzureAIClient constructor: agent_name, agent_version, project_endpoint, credential chat_client = AzureAIClient( project_endpoint=self.project_endpoint, agent_name=azure_agent.name, agent_version=azure_agent.version, # Use the specific version we just created - model_deployment_name=deployment_name, credential=self.creds, ) return chat_client diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 12bd70ec8..58fc507e1 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -133,11 +133,9 @@ async def init_orchestration( try: # Create the chat client (AzureAIClient) - # Use team deployment_name with fallback to default model if empty - deployment_name = team_config.deployment_name or config.AZURE_OPENAI_DEPLOYMENT_NAME chat_client = AzureAIClient( project_endpoint=config.AZURE_AI_PROJECT_ENDPOINT, - model_deployment_name=deployment_name, + model_deployment_name=team_config.deployment_name, agent_name=agent_name, credential=credential, ) @@ -152,7 +150,7 @@ async def init_orchestration( cls.logger.info( "Created AzureAIClient and manager ChatAgent for orchestration with model '%s' at endpoint '%s'", - deployment_name, + team_config.deployment_name, config.AZURE_AI_PROJECT_ENDPOINT, ) except Exception as e: @@ -199,17 +197,19 @@ async def init_orchestration( # Assemble workflow with callback storage = InMemoryCheckpointStorage() - # New API: .participants() accepts a list of agents + # New SDK: participants() accepts a Sequence (list) of agents + # The orchestrator uses agent.name to identify them participant_list = list(participants.values()) + cls.logger.info("Participants for workflow: %s", list(participants.keys())) + print(f"[DEBUG] Participants for workflow: {list(participants.keys())}", flush=True) builder = ( MagenticBuilder() - .participants(participant_list) + .participants(participant_list) # New SDK: pass as list .with_manager( manager=manager, # Pass manager instance (extends StandardMagenticManager) max_round_count=orchestration_config.max_rounds, - max_stall_count=3, - max_reset_count=2, + max_stall_count=0, # CRITICAL: Prevent re-calling agents when stalled (default is 3!) ) .with_checkpointing(storage) ) @@ -239,14 +239,16 @@ async def get_current_or_new_orchestration( Return an existing workflow for the user or create a new one if: - None exists - Team switched flag is True - - force_rebuild is True (for new tasks that need fresh workflow) + - force_rebuild is True (for new tasks after workflow completion) """ current = orchestration_config.get_current_orchestration(user_id) - if current is None or team_switched or force_rebuild: + needs_rebuild = current is None or team_switched or force_rebuild + + if needs_rebuild: if current is not None and (team_switched or force_rebuild): - reason = "team switched" if team_switched else "force rebuild" + reason = "team switched" if team_switched else "force rebuild for new task" cls.logger.info( - "Closing previous agents for user '%s' (reason: %s)", user_id, reason + "Rebuilding orchestration for user '%s' (reason: %s)", user_id, reason ) # Close prior agents (same logic as old version) for agent in getattr(current, "_participants", {}).values(): @@ -303,11 +305,6 @@ async def run_orchestration(self, user_id: str, input_task) -> None: Execute the Magentic workflow for the provided user and task description. """ job_id = str(uuid.uuid4()) - - # Clean up any accumulated state from previous runs (cancelled plans, etc.) - # This prevents cross-scenario leakage - orchestration_config.cleanup_user_state(user_id) - orchestration_config.set_approval_pending(job_id) self.logger.info( "Starting orchestration job '%s' for user '%s'", job_id, user_id @@ -320,16 +317,6 @@ async def run_orchestration(self, user_id: str, input_task) -> None: if workflow is None: print(f"[ERROR] Orchestration not initialized for user '{user_id}'") raise ValueError("Orchestration not initialized for user.") - - # Reset manager's plan state to prevent leakage from cancelled plans - manager = getattr(workflow, "_manager", None) - if manager and hasattr(manager, "magentic_plan"): - manager.magentic_plan = None - self.logger.debug("Reset manager's magentic_plan for fresh run") - if manager and hasattr(manager, "task_ledger"): - manager.task_ledger = None - self.logger.debug("Reset manager's task_ledger for fresh run") - # Fresh thread per participant to avoid cross-run state bleed executors = getattr(workflow, "executors", {}) self.logger.debug("Executor keys at run start: %s", list(executors.keys())) @@ -396,12 +383,16 @@ async def run_orchestration(self, user_id: str, input_task) -> None: task_text = getattr(input_task, "description", str(input_task)) self.logger.debug("Task: %s", task_text) + # Track how many times each agent is called (for debugging duplicate calls) + agent_call_counts: dict = {} + try: # Execute workflow using run_stream with task as positional parameter # The execution settings are configured in the manager/client final_output: str | None = None self.logger.info("Starting workflow execution...") + print(f"[ORCHESTRATOR] 🚀 Starting workflow with max_rounds={orchestration_config.max_rounds}", flush=True) last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: @@ -446,11 +437,20 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Handle group chat request sent elif isinstance(event, GroupChatRequestSentEvent): + agent_name = event.participant_name + agent_call_counts[agent_name] = agent_call_counts.get(agent_name, 0) + 1 + call_num = agent_call_counts[agent_name] + self.logger.info( - "[REQUEST SENT (round %d)] to agent: %s", + "[REQUEST SENT (round %d)] to agent: %s (call #%d)", event.round_index, - event.participant_name + agent_name, + call_num ) + print(f"[ORCHESTRATOR] 📤 REQUEST SENT round={event.round_index} to agent={agent_name} (call #{call_num})", flush=True) + + if call_num > 1: + print(f"[ORCHESTRATOR] ⚠️ WARNING: Agent '{agent_name}' called {call_num} times!", flush=True) # Handle group chat response received - THIS IS WHERE AGENT RESPONSES COME elif isinstance(event, GroupChatResponseReceivedEvent): @@ -511,6 +511,13 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Extract final result final_text = final_output if final_output else "" + # Log agent call summary + print(f"\n[ORCHESTRATOR] 📊 AGENT CALL SUMMARY:", flush=True) + for agent_name, count in agent_call_counts.items(): + status = "✅" if count == 1 else "⚠️ DUPLICATE" + print(f" {status} {agent_name}: called {count} time(s)", flush=True) + self.logger.info("Agent call counts: %s", agent_call_counts) + # Log results self.logger.info("\nAgent responses:") self.logger.info( From 643306f29f676432365bf8cb71218a0fdf037c3b Mon Sep 17 00:00:00 2001 From: Dhruvkumar-Microsoft Date: Fri, 13 Feb 2026 17:45:34 +0530 Subject: [PATCH 09/13] updated the logging and resolved the HR scenario issue --- src/backend/app.py | 11 +++++++++-- src/backend/v4/magentic_agents/foundry_agent.py | 3 +++ src/backend/v4/orchestration/orchestration_manager.py | 1 + 3 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src/backend/app.py b/src/backend/app.py index 35e4e47af..65381236a 100644 --- a/src/backend/app.py +++ b/src/backend/app.py @@ -3,9 +3,14 @@ from contextlib import asynccontextmanager +from common.config.app_config import config + +# Configure logging levels FIRST, before any logging calls +logging.basicConfig(level=getattr(logging, config.AZURE_BASIC_LOGGING_LEVEL.upper(), logging.INFO)) + from azure.monitor.opentelemetry import configure_azure_monitor -from common.config.app_config import config +#from common.config.app_config import config from common.models.messages_af import UserLanguage # FastAPI imports @@ -61,7 +66,7 @@ async def lifespan(app: FastAPI): ) # Configure logging levels from environment variables -logging.basicConfig(level=getattr(logging, config.AZURE_BASIC_LOGGING_LEVEL.upper(), logging.INFO)) +#logging.basicConfig(level=getattr(logging, config.AZURE_BASIC_LOGGING_LEVEL.upper(), logging.INFO)) # Configure Azure package logging levels azure_level = getattr(logging, config.AZURE_PACKAGE_LOGGING_LEVEL.upper(), logging.WARNING) @@ -73,6 +78,8 @@ async def lifespan(app: FastAPI): logging.getLogger("opentelemetry.sdk").setLevel(logging.ERROR) +logging.getLogger("azure.core.pipeline.policies.http_logging_policy").setLevel(logging.WARNING) + # Initialize the FastAPI app app = FastAPI(lifespan=lifespan) diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 84297ebef..b686a2aeb 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -227,6 +227,7 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional project_endpoint=self.project_endpoint, agent_name=azure_agent.name, agent_version=azure_agent.version, # Use the specific version we just created + model_deployment_name=self.model_deployment_name, credential=self.creds, ) return chat_client @@ -282,6 +283,7 @@ async def _after_open(self) -> None: tool_choice="required", # Force usage temperature=temp, model_id=self.model_deployment_name, + default_options={"store": False}, # Client-managed conversation to avoid stale tool call IDs across rounds ) else: # use MCP path @@ -297,6 +299,7 @@ async def _after_open(self) -> None: tool_choice="auto" if tools else "none", temperature=temp, model_id=self.model_deployment_name, + default_options={"store": False}, # Client-managed conversation to avoid stale tool call IDs across rounds ) self.logger.info("Initialized ChatAgent '%s'", self.agent_name) diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 58fc507e1..3e89c61ac 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -146,6 +146,7 @@ async def init_orchestration( name="MagenticManager", description="Orchestrator that coordinates the team to complete complex tasks efficiently.", instructions="You coordinate a team to complete complex tasks efficiently.", + default_options={"store": False}, # Client-managed conversation to avoid stale tool call IDs across rounds ) cls.logger.info( From 303c8667e21a3e8c37be0131e657c8b24293760c Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Mon, 16 Feb 2026 12:56:23 +0530 Subject: [PATCH 10/13] Refactor FoundryAgentTemplate and MagenticAgentFactory to remove debug print statements and streamline logging --- .../v4/magentic_agents/foundry_agent.py | 18 ++-------------- .../magentic_agents/magentic_agent_factory.py | 1 - .../orchestration/human_approval_manager.py | 4 ---- .../v4/orchestration/orchestration_manager.py | 21 ++----------------- 4 files changed, 4 insertions(+), 40 deletions(-) diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index b686a2aeb..f44523fa5 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -70,23 +70,18 @@ def __init__( self._use_azure_search = self._is_azure_search_requested() self.use_reasoning = use_reasoning - # Placeholder for server-created Azure AI agent id/version (if Azure Search path) + # Placeholder for server-created Azure AI agent id (if Azure Search path) self._azure_server_agent_id: Optional[str] = None - self._azure_server_agent_version: Optional[str] = None # ------------------------- # Mode detection # ------------------------- def _is_azure_search_requested(self) -> bool: """Determine if Azure AI Search raw tool path should be used.""" - print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}, search={self.search}") if not self.search: - print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}: No search config, returning False") return False # Minimal heuristic: presence of required attributes - has_index = hasattr(self.search, "index_name") and bool(self.search.index_name) - print(f"[DEBUG _is_azure_search_requested] Agent={self.agent_name}: has_index={has_index}, index_name={getattr(self.search, 'index_name', None)}") if has_index: self.logger.info( "Azure AI Search requested (connection_id=%s, index=%s).", @@ -137,7 +132,6 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional Returns: AzureAIClient | None """ - print(f"[DEBUG _create_azure_search_enabled_client] Agent={self.agent_name}, chatClient={chatClient}, search_config={self.search}") if chatClient: self.logger.info("Reusing existing chatClient for agent '%s' (already has Azure Search configured)", self.agent_name) return chatClient @@ -185,9 +179,6 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional "Always use the Azure AI Search tool and configured index for knowledge retrieval." ) - print(f"[AGENT CREATE] 🆕 Creating agent in Foundry: '{self.agent_name}'", flush=True) - print(f"[AGENT CREATE] Model: {self.model_deployment_name}", flush=True) - print(f"[AGENT CREATE] Search: connection={connection_name}, index={index_name}", flush=True) azure_agent = await self.project_client.agents.create_version( agent_name=self.agent_name, # Use original name @@ -212,8 +203,7 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional ) self._azure_server_agent_id = azure_agent.id - self._azure_server_agent_version = azure_agent.version - print(f"[AGENT CREATE] ✅ Created agent: name={azure_agent.name}, id={azure_agent.id}, version={azure_agent.version}", flush=True) + self.logger.info( "Created Azure AI Search agent via create_version (name=%s, id=%s, version=%s).", azure_agent.name, @@ -239,8 +229,6 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional index_name, ex, ) - import traceback - traceback.print_exc() return None # ------------------------- @@ -257,7 +245,6 @@ async def _after_open(self) -> None: try: chatClient = await self.get_database_team_agent() - print(f"[DEBUG _after_open] Agent={self.agent_name}, _use_azure_search={self._use_azure_search}, search_config={self.search}, chatClient={chatClient}") if self._use_azure_search: # Azure Search mode (skip MCP + Code Interpreter due to incompatibility) @@ -266,7 +253,6 @@ async def _after_open(self) -> None: self.agent_name, getattr(self.search, "index_name", "N/A") if self.search else "N/A" ) - print(f"[DEBUG _after_open] Creating Azure Search client for {self.agent_name}") chat_client = await self._create_azure_search_enabled_client(chatClient) if not chat_client: raise RuntimeError( diff --git a/src/backend/v4/magentic_agents/magentic_agent_factory.py b/src/backend/v4/magentic_agents/magentic_agent_factory.py index 3eafb5831..36544166d 100644 --- a/src/backend/v4/magentic_agents/magentic_agent_factory.py +++ b/src/backend/v4/magentic_agents/magentic_agent_factory.py @@ -115,7 +115,6 @@ async def create_agent_from_config( index_name, "Reasoning" if use_reasoning else "Foundry", ) - print(f"[FACTORY] 🆕 Creating NEW agent: {agent_obj.name} (id={id(agent_obj)})", flush=True) agent = FoundryAgentTemplate( agent_name=agent_obj.name, diff --git a/src/backend/v4/orchestration/human_approval_manager.py b/src/backend/v4/orchestration/human_approval_manager.py index 39089247a..00850ec26 100644 --- a/src/backend/v4/orchestration/human_approval_manager.py +++ b/src/backend/v4/orchestration/human_approval_manager.py @@ -33,7 +33,6 @@ class HumanApprovalMagenticManager(StandardMagenticManager): approval_enabled: bool = True magentic_plan: Optional[MPlan] = None current_user_id: str # populated in __init__ - _called_agents: set # Track which agents have been called def __init__(self, user_id: str, agent, *args, **kwargs): """ @@ -44,9 +43,6 @@ def __init__(self, user_id: str, agent, *args, **kwargs): *args: Additional positional arguments for the parent StandardMagenticManager. **kwargs: Additional keyword arguments for the parent StandardMagenticManager. """ - - # Initialize called agents tracker - self._called_agents = set() plan_append = """ diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 3e89c61ac..fa5801ae5 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -202,7 +202,6 @@ async def init_orchestration( # The orchestrator uses agent.name to identify them participant_list = list(participants.values()) cls.logger.info("Participants for workflow: %s", list(participants.keys())) - print(f"[DEBUG] Participants for workflow: {list(participants.keys())}", flush=True) builder = ( MagenticBuilder() @@ -277,24 +276,17 @@ async def get_current_or_new_orchestration( cls.logger.error( "Failed to create agents for user '%s': %s", user_id, e ) - print(f"Failed to create agents for user '{user_id}': {e}") raise try: cls.logger.info("Initializing new orchestration for user '%s'", user_id) - print(f"[DEBUG] Initializing new orchestration for user '{user_id}'") workflow = await cls.init_orchestration( agents, team_config, team_service.memory_context, user_id ) orchestration_config.orchestrations[user_id] = workflow - print(f"[DEBUG] Stored workflow for user '{user_id}': {workflow is not None}") - print(f"[DEBUG] orchestrations keys: {list(orchestration_config.orchestrations.keys())}") except Exception as e: cls.logger.error( "Failed to initialize orchestration for user '%s': %s", user_id, e ) - print(f"Failed to initialize orchestration for user '{user_id}': {e}") - import traceback - traceback.print_exc() raise return orchestration_config.get_current_orchestration(user_id) @@ -310,13 +302,9 @@ async def run_orchestration(self, user_id: str, input_task) -> None: self.logger.info( "Starting orchestration job '%s' for user '%s'", job_id, user_id ) - print(f"[DEBUG] run_orchestration called for user '{user_id}'") - print(f"[DEBUG] orchestrations keys before get: {list(orchestration_config.orchestrations.keys())}") workflow = orchestration_config.get_current_orchestration(user_id) - print(f"[DEBUG] workflow is None: {workflow is None}") if workflow is None: - print(f"[ERROR] Orchestration not initialized for user '{user_id}'") raise ValueError("Orchestration not initialized for user.") # Fresh thread per participant to avoid cross-run state bleed executors = getattr(workflow, "executors", {}) @@ -393,7 +381,7 @@ async def run_orchestration(self, user_id: str, input_task) -> None: final_output: str | None = None self.logger.info("Starting workflow execution...") - print(f"[ORCHESTRATOR] 🚀 Starting workflow with max_rounds={orchestration_config.max_rounds}", flush=True) + last_message_id: str | None = None async for event in workflow.run_stream(task_text): try: @@ -448,10 +436,9 @@ async def run_orchestration(self, user_id: str, input_task) -> None: agent_name, call_num ) - print(f"[ORCHESTRATOR] 📤 REQUEST SENT round={event.round_index} to agent={agent_name} (call #{call_num})", flush=True) if call_num > 1: - print(f"[ORCHESTRATOR] ⚠️ WARNING: Agent '{agent_name}' called {call_num} times!", flush=True) + self.logger.warning("Agent '%s' called %d times", agent_name, call_num) # Handle group chat response received - THIS IS WHERE AGENT RESPONSES COME elif isinstance(event, GroupChatResponseReceivedEvent): @@ -513,10 +500,6 @@ async def run_orchestration(self, user_id: str, input_task) -> None: final_text = final_output if final_output else "" # Log agent call summary - print(f"\n[ORCHESTRATOR] 📊 AGENT CALL SUMMARY:", flush=True) - for agent_name, count in agent_call_counts.items(): - status = "✅" if count == 1 else "⚠️ DUPLICATE" - print(f" {status} {agent_name}: called {count} time(s)", flush=True) self.logger.info("Agent call counts: %s", agent_call_counts) # Log results From bb5331e84b64e32e1f7590fa7bce57822cdb0a42 Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 20 Feb 2026 09:45:45 +0530 Subject: [PATCH 11/13] Refactor agent creation and configuration to avoid mutating original team objects and streamline AzureAIClient initialization --- src/backend/common/utils/utils_af.py | 14 +- src/backend/common/utils/utils_agents.py | 24 --- src/backend/v4/callbacks/response_handlers.py | 6 +- .../v4/magentic_agents/common/lifecycle.py | 178 +----------------- .../v4/magentic_agents/foundry_agent.py | 25 +-- 5 files changed, 28 insertions(+), 219 deletions(-) diff --git a/src/backend/common/utils/utils_af.py b/src/backend/common/utils/utils_af.py index 2d1dd794e..a22212144 100644 --- a/src/backend/common/utils/utils_af.py +++ b/src/backend/common/utils/utils_af.py @@ -88,9 +88,15 @@ async def create_RAI_agent( ) model_deployment_name = config.AZURE_OPENAI_RAI_DEPLOYMENT_NAME - team.team_id = "rai_team" # Use a fixed team ID for RAI agent - team.name = "RAI Team" - team.description = "Team responsible for Responsible AI checks" + + # Create a copy to avoid mutating the caller's team config. + # The original team object is reused later (e.g., for orchestration init), + # so mutating it would corrupt the real team name/id. + rai_team = team.model_copy() + rai_team.team_id = "rai_team" + rai_team.name = "RAI Team" + rai_team.description = "Team responsible for Responsible AI checks" + agent = FoundryAgentTemplate( agent_name=agent_name, agent_description=agent_description, @@ -101,7 +107,7 @@ async def create_RAI_agent( project_endpoint=config.AZURE_AI_PROJECT_ENDPOINT, mcp_config=None, search_config=None, - team_config=team, + team_config=rai_team, memory_store=memory_store, ) diff --git a/src/backend/common/utils/utils_agents.py b/src/backend/common/utils/utils_agents.py index 1e164f89c..c679f9f62 100644 --- a/src/backend/common/utils/utils_agents.py +++ b/src/backend/common/utils/utils_agents.py @@ -1,11 +1,6 @@ -import logging import secrets import string -from typing import Optional - -from common.database.database_base import DatabaseBase -from common.models.messages_af import TeamConfiguration def generate_assistant_id(prefix: str = "asst_", length: int = 24) -> str: @@ -21,22 +16,3 @@ def generate_assistant_id(prefix: str = "asst_", length: int = 24) -> str: # cryptographically strong randomness random_part = "".join(secrets.choice(alphabet) for _ in range(length)) return f"{prefix}{random_part}" - - -async def get_database_team_agent_id( - memory_store: DatabaseBase, team_config: TeamConfiguration, agent_name: str -) -> Optional[str]: - """Retrieve existing team agent from database, if any.""" - agent_id = None - try: - currentAgent = await memory_store.get_team_agent( - team_id=team_config.team_id, agent_name=agent_name - ) - if currentAgent and currentAgent.agent_foundry_id: - agent_id = currentAgent.agent_foundry_id - - except ( - Exception - ) as ex: # Consider narrowing this to specific exceptions if possible - logging.error("Failed to initialize Get database team agent: %s", ex) - return agent_id diff --git a/src/backend/v4/callbacks/response_handlers.py b/src/backend/v4/callbacks/response_handlers.py index 88c6085f5..f034e4168 100644 --- a/src/backend/v4/callbacks/response_handlers.py +++ b/src/backend/v4/callbacks/response_handlers.py @@ -8,7 +8,7 @@ import re from typing import Any -from agent_framework import ChatMessage, AgentRunUpdateEvent +from agent_framework import ChatMessage from v4.config.settings import connection_config from v4.models.messages import ( @@ -108,7 +108,7 @@ def agent_response_callback( async def streaming_agent_response_callback( agent_id: str, - update, # AgentRunUpdateEvent.data or similar streaming update object + update, # Streaming update object (e.g. AgentResponseUpdate, ChatMessage) is_final: bool, user_id: str | None = None, ) -> None: @@ -119,7 +119,7 @@ async def streaming_agent_response_callback( return try: - # Handle both AgentRunUpdateEvent.data and raw text updates + # Handle various streaming update object shapes chunk_text = getattr(update, "text", None) # If text is None, don't fall back to str(update) as that would show object repr diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index af9dcb846..c9093c318 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -15,10 +15,9 @@ from azure.ai.agents.aio import AgentsClient from azure.identity.aio import DefaultAzureCredential from common.database.database_base import DatabaseBase -from common.models.messages_af import CurrentTeamAgent, TeamConfiguration +from common.models.messages_af import TeamConfiguration from common.utils.utils_agents import ( generate_assistant_id, - get_database_team_agent_id, ) from v4.common.services.team_service import TeamService from v4.config.agent_registry import agent_registry @@ -148,13 +147,12 @@ async def _after_open(self) -> None: """Subclasses must build self._agent here.""" raise NotImplementedError - def get_chat_client(self, chat_client) -> AzureAIClient: + def get_chat_client(self) -> AzureAIClient: """Return the underlying ChatClientProtocol (AzureAIClient). - Uses agent_name with use_latest_version=True to get the latest agent version + Uses agent_name with use_latest_version=True to get the latest agent version. + Agent reuse is handled automatically by the SDK via agent_name. """ - if chat_client: - return chat_client if ( self._agent and self._agent.chat_client @@ -173,176 +171,16 @@ def get_chat_client(self, chat_client) -> AzureAIClient: ) return chat_client - async def resolve_agent_id(self, agent_id: str) -> Optional[str]: - """Resolve agent ID via Projects SDK first (for RAI agents), fallback to AgentsClient. - - Args: - agent_id: The agent ID to resolve - - Returns: - The resolved agent ID if found, None otherwise - """ - # Try Projects SDK first (RAI agents were created via project_client) - try: - if self.project_client: - agent = await self.project_client.agents.get_agent(agent_id) - if agent and agent.id: - self.logger.info( - "RAI.AgentReuseSuccess: Resolved agent via Projects SDK (id=%s)", - agent.id, - ) - return agent.id - except Exception as ex: - self.logger.warning( - "RAI.AgentReuseMiss: Projects SDK get_agent failed (reason=ProjectsGetFailed, id=%s): %s", - agent_id, - ex, - ) - - # Fallback via AgentsClient (endpoint) - try: - if self.client: - agent = await self.client.get_agent(agent_id=agent_id) - if agent and agent.id: - self.logger.info( - "RAI.AgentReuseSuccess: Resolved agent via AgentsClient (id=%s)", - agent.id, - ) - return agent.id - except Exception as ex: - self.logger.warning( - "RAI.AgentReuseMiss: AgentsClient get_agent failed (reason=EndpointGetFailed, id=%s): %s", - agent_id, - ex, - ) - - self.logger.error( - "RAI.AgentReuseMiss: Agent ID not resolvable via any client (reason=ClientMismatch, id=%s)", - agent_id, - ) - return None - - def get_agent_id(self, chat_client) -> str: - """Return the underlying agent ID or generate a new one. + def get_agent_id(self) -> str: + """Generate a local agent ID for the ChatAgent wrapper. - Note: The new AzureAIClient doesn't expose agent_id directly. - We generate a new ID if not available. + The new AzureAIClient identifies agents by name (not ID) on the server side. + This ID is only used locally for the ChatAgent wrapper instance. """ - # Generate a new agent ID since AzureAIClient doesn't expose agent_id id = generate_assistant_id() self.logger.info("Generated new agent ID: %s", id) return id - async def get_database_team_agent(self) -> Optional[AzureAIClient]: - """Retrieve existing team agent from database, if any. - - NOTE: Agent reuse is currently DISABLED to ensure fresh agents are created - with the correct Azure AI Search configuration. - This prevents issues with stale agents that may not have the search tool configured. - - To re-enable agent reuse, set ENABLE_AGENT_REUSE=true in environment. - """ - import os - - # DISABLED: Always create fresh agents to ensure Azure AI Search tool is configured - enable_reuse = os.environ.get("ENABLE_AGENT_REUSE", "false").lower() == "true" - if not enable_reuse: - self.logger.info( - "Agent reuse DISABLED: Creating fresh agent with search tools (agent_name=%s)", - self.agent_name, - ) - return None - - chat_client = None - try: - agent_id = await get_database_team_agent_id( - self.memory_store, self.team_config, self.agent_name - ) - - if not agent_id: - self.logger.info( - "RAI reuse: no stored agent id (agent_name=%s)", self.agent_name - ) - return None - - # Use resolve_agent_id to try Projects SDK first, then AgentsClient - resolved = await self.resolve_agent_id(agent_id) - if not resolved: - self.logger.error( - "RAI.AgentReuseMiss: stored id %s not resolvable (agent_name=%s)", - agent_id, - self.agent_name, - ) - return None - - # Create client with resolved ID - if self.agent_name == "RAIAgent" and self.project_client: - chat_client = AzureAIClient( - project_endpoint=self.project_endpoint, - agent_id=resolved, - credential=self.creds, - ) - self.logger.info( - "RAI.AgentReuseSuccess: Created AzureAIClient (id=%s)", - resolved, - ) - else: - chat_client = AzureAIClient( - project_endpoint=self.project_endpoint, - agent_id=resolved, - model_deployment_name=self.model_deployment_name, - credential=self.creds, - ) - self.logger.info( - "Created AzureAIClient via endpoint (id=%s)", resolved - ) - - except Exception as ex: - self.logger.error( - "Failed to initialize Get database team agent (agent_name=%s): %s", - self.agent_name, - ex, - ) - return chat_client - - async def save_database_team_agent(self) -> None: - """Save current team agent to database (only if truly new or changed).""" - try: - if self._agent is None or self._agent.id is None: - self.logger.error("Cannot save database team agent: agent or agent_id is None") - return - - # Use the agent ID from ChatAgent (set during creation) - agent_id = self._agent.id - - # Check if stored ID matches current ID - stored_id = await get_database_team_agent_id( - self.memory_store, self.team_config, self.agent_name - ) - if stored_id == agent_id: - self.logger.info( - "RAI reuse: id unchanged (id=%s); skip save.", agent_id - ) - return - - currentAgent = CurrentTeamAgent( - team_id=self.team_config.team_id, - team_name=self.team_config.name, - agent_name=self.agent_name, - agent_foundry_id=agent_id, - agent_description=self.agent_description, - agent_instructions=self.agent_instructions, - ) - await self.memory_store.add_team_agent(currentAgent) - self.logger.info( - "Saved team agent to database (agent_name=%s, id=%s)", - self.agent_name, - agent_id, - ) - - except Exception as ex: - self.logger.error("Failed to save database: %s", ex) - async def _prepare_mcp_tool(self) -> None: """Translate MCPConfig to a HostedMCPTool (agent_framework construct).""" if not self.mcp_cfg: diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index f44523fa5..6d3974010 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -115,7 +115,7 @@ async def _collect_tools(self) -> List: # ------------------------- # Azure Search helper # ------------------------- - async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional[AzureAIClient]: + async def _create_azure_search_enabled_client(self) -> Optional[AzureAIClient]: """ Create a server-side Azure AI agent with Azure AI Search tool using create_version. @@ -132,10 +132,6 @@ async def _create_azure_search_enabled_client(self, chatClient=None) -> Optional Returns: AzureAIClient | None """ - if chatClient: - self.logger.info("Reusing existing chatClient for agent '%s' (already has Azure Search configured)", self.agent_name) - return chatClient - if not self.search: self.logger.error("Search configuration missing.") return None @@ -244,8 +240,6 @@ async def _after_open(self) -> None: temp = 0.1 try: - chatClient = await self.get_database_team_agent() - if self._use_azure_search: # Azure Search mode (skip MCP + Code Interpreter due to incompatibility) self.logger.info( @@ -253,7 +247,7 @@ async def _after_open(self) -> None: self.agent_name, getattr(self.search, "index_name", "N/A") if self.search else "N/A" ) - chat_client = await self._create_azure_search_enabled_client(chatClient) + chat_client = await self._create_azure_search_enabled_client() if not chat_client: raise RuntimeError( "Azure AI Search mode requested but setup failed." @@ -261,8 +255,8 @@ async def _after_open(self) -> None: # In Azure Search raw tool path, tools/tool_choice are handled server-side. self._agent = ChatAgent( - id=self.get_agent_id(chat_client), - chat_client=self.get_chat_client(chat_client), + id=self.get_agent_id(), + chat_client=chat_client, instructions=self.agent_instructions, name=self.agent_name, description=self.agent_description, @@ -272,12 +266,12 @@ async def _after_open(self) -> None: default_options={"store": False}, # Client-managed conversation to avoid stale tool call IDs across rounds ) else: - # use MCP path + # MCP path (also used by RAI agent which has no tools) self.logger.info("Initializing agent in MCP mode.") tools = await self._collect_tools() self._agent = ChatAgent( - id=self.get_agent_id(chatClient), - chat_client=self.get_chat_client(chatClient), + id=self.get_agent_id(), + chat_client=self.get_chat_client(), instructions=self.agent_instructions, name=self.agent_name, description=self.agent_description, @@ -314,12 +308,7 @@ async def invoke(self, prompt: str): messages = [ChatMessage(role=Role.USER, text=prompt)] - agent_saved = False async for update in self._agent.run_stream(messages): - # Save agent ID only once on first update - if not agent_saved: - await self.save_database_team_agent() - agent_saved = True yield update # ------------------------- From 60c894e9bca957647d6d760fd6918c30a164d2bd Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 20 Feb 2026 19:59:24 +0530 Subject: [PATCH 12/13] Remove description and instructions from MagenticManager ChatAgent initialization --- src/backend/v4/orchestration/orchestration_manager.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index fa5801ae5..6d6811850 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -144,8 +144,6 @@ async def init_orchestration( manager_agent = ChatAgent( chat_client=chat_client, name="MagenticManager", - description="Orchestrator that coordinates the team to complete complex tasks efficiently.", - instructions="You coordinate a team to complete complex tasks efficiently.", default_options={"store": False}, # Client-managed conversation to avoid stale tool call IDs across rounds ) From 6a0462bb898395c7374f4933c95b60208788913a Mon Sep 17 00:00:00 2001 From: Abdul-Microsoft Date: Fri, 27 Feb 2026 11:54:44 +0530 Subject: [PATCH 13/13] resolved pylint issues --- src/backend/app.py | 4 +-- src/backend/v4/callbacks/response_handlers.py | 2 +- .../v4/magentic_agents/common/lifecycle.py | 4 +-- .../v4/magentic_agents/foundry_agent.py | 1 - .../orchestration/human_approval_manager.py | 4 +-- .../v4/orchestration/orchestration_manager.py | 30 +++++++++---------- 6 files changed, 22 insertions(+), 23 deletions(-) diff --git a/src/backend/app.py b/src/backend/app.py index 65381236a..2cf7d6a6b 100644 --- a/src/backend/app.py +++ b/src/backend/app.py @@ -10,7 +10,7 @@ from azure.monitor.opentelemetry import configure_azure_monitor -#from common.config.app_config import config +# from common.config.app_config import config from common.models.messages_af import UserLanguage # FastAPI imports @@ -66,7 +66,7 @@ async def lifespan(app: FastAPI): ) # Configure logging levels from environment variables -#logging.basicConfig(level=getattr(logging, config.AZURE_BASIC_LOGGING_LEVEL.upper(), logging.INFO)) +# logging.basicConfig(level=getattr(logging, config.AZURE_BASIC_LOGGING_LEVEL.upper(), logging.INFO)) # Configure Azure package logging levels azure_level = getattr(logging, config.AZURE_PACKAGE_LOGGING_LEVEL.upper(), logging.WARNING) diff --git a/src/backend/v4/callbacks/response_handlers.py b/src/backend/v4/callbacks/response_handlers.py index f034e4168..0a817ef94 100644 --- a/src/backend/v4/callbacks/response_handlers.py +++ b/src/backend/v4/callbacks/response_handlers.py @@ -121,7 +121,7 @@ async def streaming_agent_response_callback( try: # Handle various streaming update object shapes chunk_text = getattr(update, "text", None) - + # If text is None, don't fall back to str(update) as that would show object repr # Just skip if there's no actual text content if chunk_text is None: diff --git a/src/backend/v4/magentic_agents/common/lifecycle.py b/src/backend/v4/magentic_agents/common/lifecycle.py index c9093c318..b38e31eed 100644 --- a/src/backend/v4/magentic_agents/common/lifecycle.py +++ b/src/backend/v4/magentic_agents/common/lifecycle.py @@ -149,7 +149,7 @@ async def _after_open(self) -> None: def get_chat_client(self) -> AzureAIClient: """Return the underlying ChatClientProtocol (AzureAIClient). - + Uses agent_name with use_latest_version=True to get the latest agent version. Agent reuse is handled automatically by the SDK via agent_name. """ @@ -173,7 +173,7 @@ def get_chat_client(self) -> AzureAIClient: def get_agent_id(self) -> str: """Generate a local agent ID for the ChatAgent wrapper. - + The new AzureAIClient identifies agents by name (not ID) on the server side. This ID is only used locally for the ChatAgent wrapper instance. """ diff --git a/src/backend/v4/magentic_agents/foundry_agent.py b/src/backend/v4/magentic_agents/foundry_agent.py index 6d3974010..38fd0cc6b 100644 --- a/src/backend/v4/magentic_agents/foundry_agent.py +++ b/src/backend/v4/magentic_agents/foundry_agent.py @@ -174,7 +174,6 @@ async def _create_azure_search_enabled_client(self) -> Optional[AzureAIClient]: f"{self.agent_instructions} " "Always use the Azure AI Search tool and configured index for knowledge retrieval." ) - azure_agent = await self.project_client.agents.create_version( agent_name=self.agent_name, # Use original name diff --git a/src/backend/v4/orchestration/human_approval_manager.py b/src/backend/v4/orchestration/human_approval_manager.py index 00850ec26..7f33c9ac4 100644 --- a/src/backend/v4/orchestration/human_approval_manager.py +++ b/src/backend/v4/orchestration/human_approval_manager.py @@ -85,7 +85,7 @@ def __init__(self, user_id: str, agent, *args, **kwargs): ORCHESTRATOR_TASK_LEDGER_PLAN_UPDATE_PROMPT + plan_append ) kwargs["final_answer_prompt"] = ORCHESTRATOR_FINAL_ANSWER_PROMPT + final_append - + # Override progress ledger prompt to discourage re-calling agents from agent_framework._workflows._magentic import ORCHESTRATOR_PROGRESS_LEDGER_PROMPT kwargs["progress_ledger_prompt"] = ORCHESTRATOR_PROGRESS_LEDGER_PROMPT + progress_append @@ -319,4 +319,4 @@ def plan_to_obj(self, magentic_context: MagenticContext, ledger) -> MPlan: task=task_text, ) - return return_plan \ No newline at end of file + return return_plan diff --git a/src/backend/v4/orchestration/orchestration_manager.py b/src/backend/v4/orchestration/orchestration_manager.py index 6d6811850..d38748d83 100644 --- a/src/backend/v4/orchestration/orchestration_manager.py +++ b/src/backend/v4/orchestration/orchestration_manager.py @@ -50,7 +50,7 @@ def __init__(self): def _extract_response_text(self, data) -> str: """ Extract text content from various agent_framework response types. - + Handles: - ChatMessage: Extract .text - AgentResponse: Extract .text @@ -59,15 +59,15 @@ def _extract_response_text(self, data) -> str: """ if data is None: return "" - + # Direct ChatMessage if isinstance(data, ChatMessage): return data.text or "" - + # Has .text attribute directly (AgentResponse, etc.) if hasattr(data, "text") and data.text: return data.text - + # AgentExecutorResponse - has agent_response and full_conversation if hasattr(data, "agent_response"): # Try to get text from agent_response first @@ -79,7 +79,7 @@ def _extract_response_text(self, data) -> str: last_msg = data.full_conversation[-1] if isinstance(last_msg, ChatMessage) and last_msg.text: return last_msg.text - + # List of items - could be AgentExecutorResponse, ChatMessage, etc. if isinstance(data, list) and len(data) > 0: texts = [] @@ -91,7 +91,7 @@ def _extract_response_text(self, data) -> str: if texts: # Return the last non-empty response (most recent) return texts[-1] - + return "" # --------------------------- @@ -195,12 +195,12 @@ async def init_orchestration( # Assemble workflow with callback storage = InMemoryCheckpointStorage() - + # New SDK: participants() accepts a Sequence (list) of agents # The orchestrator uses agent.name to identify them participant_list = list(participants.values()) cls.logger.info("Participants for workflow: %s", list(participants.keys())) - + builder = ( MagenticBuilder() .participants(participant_list) # New SDK: pass as list @@ -241,7 +241,7 @@ async def get_current_or_new_orchestration( """ current = orchestration_config.get_current_orchestration(user_id) needs_rebuild = current is None or team_switched or force_rebuild - + if needs_rebuild: if current is not None and (team_switched or force_rebuild): reason = "team switched" if team_switched else "force rebuild for new task" @@ -387,7 +387,7 @@ async def run_orchestration(self, user_id: str, input_task) -> None: event_type_name = type(event).__name__ if event_type_name != "AgentRunUpdateEvent": self.logger.info("[EVENT] %s", event_type_name) - + # Handle orchestrator events (plan, progress ledger) if isinstance(event, MagenticOrchestratorEvent): self.logger.info( @@ -403,7 +403,7 @@ async def run_orchestration(self, user_id: str, input_task) -> None: elif isinstance(event, AgentRunUpdateEvent): message_id = event.data.message_id if hasattr(event.data, 'message_id') else None executor_id = event.executor_id - + # Stream the update try: await streaming_agent_response_callback( @@ -417,7 +417,7 @@ async def run_orchestration(self, user_id: str, input_task) -> None: "Error in streaming callback for agent %s: %s", executor_id, e ) - + # Track message for formatting if message_id != last_message_id: last_message_id = message_id @@ -427,14 +427,14 @@ async def run_orchestration(self, user_id: str, input_task) -> None: agent_name = event.participant_name agent_call_counts[agent_name] = agent_call_counts.get(agent_name, 0) + 1 call_num = agent_call_counts[agent_name] - + self.logger.info( "[REQUEST SENT (round %d)] to agent: %s (call #%d)", event.round_index, agent_name, call_num ) - + if call_num > 1: self.logger.warning("Agent '%s' called %d times", agent_name, call_num) @@ -448,7 +448,7 @@ async def run_orchestration(self, user_id: str, input_task) -> None: # Send the agent response to the UI if event.data: response_text = self._extract_response_text(event.data) - + if response_text: self.logger.info("Sending agent response to UI from %s", event.participant_name) agent_response_callback(