From 5de438066102d463d38a19bcdd4a6e4f77dcfa5a Mon Sep 17 00:00:00 2001 From: casaman Date: Tue, 2 Sep 2025 12:45:02 -0600 Subject: [PATCH 01/21] Initial commit --- README.md | 1 + 1 file changed, 1 insertion(+) create mode 100644 README.md diff --git a/README.md b/README.md new file mode 100644 index 0000000..62912e3 --- /dev/null +++ b/README.md @@ -0,0 +1 @@ +# DatabricksBotService \ No newline at end of file From 2d7e06b9b88c8a65c15cabe2f5208d84f0641e2e Mon Sep 17 00:00:00 2001 From: manffred-calvosanchez_data Date: Mon, 8 Sep 2025 17:00:12 -0600 Subject: [PATCH 02/21] Adding all the logic to create a bot azure service that interacts with databricks agent endpoint using authentication on behalf of the user. --- README.md | 277 +++++++++++++++++- app.py | 118 ++++++++ appManifest/color.png | Bin 0 -> 3415 bytes appManifest/manifest.json | 42 +++ appManifest/outline.png | Bin 0 -> 407 bytes bots/__init__.py | 7 + bots/__pycache__/__init__.cpython-312.pyc | Bin 0 -> 311 bytes bots/__pycache__/auth_bot.cpython-312.pyc | Bin 0 -> 2156 bytes bots/__pycache__/dialog_bot.cpython-312.pyc | Bin 0 -> 6308 bytes bots/auth_bot.py | 43 +++ bots/dialog_bot.py | 111 +++++++ client/__init__.py | 0 client/__pycache__/__init__.cpython-312.pyc | Bin 0 -> 186 bytes .../databricks_client.cpython-312.pyc | Bin 0 -> 3725 bytes client/databricks_client.py | 77 +++++ config.py | 20 ++ dialogs/__init__.py | 7 + dialogs/__pycache__/__init__.cpython-312.pyc | Bin 0 -> 326 bytes .../__pycache__/logout_dialog.cpython-312.pyc | Bin 0 -> 3194 bytes .../__pycache__/main_dialog.cpython-312.pyc | Bin 0 -> 5744 bytes dialogs/logout_dialog.py | 42 +++ dialogs/main_dialog.py | 144 +++++++++ helpers/__init__.py | 6 + helpers/__pycache__/__init__.cpython-312.pyc | Bin 0 -> 258 bytes .../__pycache__/dialog_helper.cpython-312.pyc | Bin 0 -> 2852 bytes helpers/dialog_helper.py | 43 +++ requirements.txt | 4 + 27 files changed, 940 insertions(+), 1 deletion(-) create mode 100644 app.py create mode 100644 appManifest/color.png create mode 100644 appManifest/manifest.json create mode 100644 appManifest/outline.png create mode 100644 bots/__init__.py create mode 100644 bots/__pycache__/__init__.cpython-312.pyc create mode 100644 bots/__pycache__/auth_bot.cpython-312.pyc create mode 100644 bots/__pycache__/dialog_bot.cpython-312.pyc create mode 100644 bots/auth_bot.py create mode 100644 bots/dialog_bot.py create mode 100644 client/__init__.py create mode 100644 client/__pycache__/__init__.cpython-312.pyc create mode 100644 client/__pycache__/databricks_client.cpython-312.pyc create mode 100644 client/databricks_client.py create mode 100644 config.py create mode 100644 dialogs/__init__.py create mode 100644 dialogs/__pycache__/__init__.cpython-312.pyc create mode 100644 dialogs/__pycache__/logout_dialog.cpython-312.pyc create mode 100644 dialogs/__pycache__/main_dialog.cpython-312.pyc create mode 100644 dialogs/logout_dialog.py create mode 100644 dialogs/main_dialog.py create mode 100644 helpers/__init__.py create mode 100644 helpers/__pycache__/__init__.cpython-312.pyc create mode 100644 helpers/__pycache__/dialog_helper.cpython-312.pyc create mode 100644 helpers/dialog_helper.py create mode 100644 requirements.txt diff --git a/README.md b/README.md index 62912e3..f103349 100644 --- a/README.md +++ b/README.md @@ -1 +1,276 @@ -# DatabricksBotService \ No newline at end of file +# Databricks AI Agent Bot for Microsoft Teams + +A sophisticated Microsoft Teams bot that provides AI agent capabilities through Databricks serving endpoints, featuring secure OAuth authentication and intelligent conversation management. + +## Overview + +This bot integrates Microsoft Teams with Databricks AI/ML serving endpoints to provide users with an intelligent conversational AI agent directly within Teams. The bot handles secure authentication using OAuth with token exchange, maintains conversation history, and supports advanced features like tool calling with visual feedback through adaptive cards. + +## Key Features + +* **šŸ” Secure Authentication**: OAuth integration with Microsoft identity providers and token exchange to Databricks +* **šŸ¤– AI Agent Integration**: Direct connection to Databricks AI/ML serving endpoints +* **šŸ’¬ Conversation Management**: Maintains conversation history across interactions +* **šŸ› ļø Tool Calling Support**: Advanced function calling capabilities with visual adaptive card feedback +* **šŸ‘„ Teams Integration**: Full Microsoft Teams bot experience with personal and team scopes +* **šŸ”„ Real-time Processing**: Asynchronous handling of AI responses and tool executions + +## Architecture + +### Authentication Flow +1. User authenticates with Microsoft OAuth via Teams +2. Bot exchanges Microsoft token for Databricks token using OIDC token exchange +3. Databricks token is used to authenticate with serving endpoints + +### Component Overview +- **`AuthBot`**: Main bot class handling Teams interactions and member management +- **`MainDialog`**: Core dialog managing authentication and AI interactions +- **`DatabricksClient`**: Client for Databricks API interactions and token management +- **`LogoutDialog`**: Handles user logout functionality +- **Adaptive Cards**: Visual representation of tool calls and results + +## Prerequisites + +### Required Accounts & Services +- **Microsoft Teams**: Account with permissions to upload custom apps (not a guest account) +- **Databricks Workspace**: Access to a Databricks workspace with AI/ML serving endpoints +- **Azure Bot Service**: Bot Framework registration for Teams integration +- **Microsoft Entra ID**: App registration for OAuth authentication + +### Development Environment +- **Python 3.11+**: [Download Python](https://www.python.org/downloads/) +- **Tunneling Solution**: [dev tunnel](https://learn.microsoft.com/en-us/azure/developer/dev-tunnels/get-started?tabs=windows) or [ngrok](https://ngrok.com/) for local development +- **Visual Studio Code**: Recommended with Microsoft 365 Agents Toolkit extension + +### Required Environment Variables +```bash +MicrosoftAppId= +MicrosoftAppPassword= +MicrosoftAppType= +MicrosoftTenantId= +ConnectionName= +DATABRICKS_HOST= +SERVING_ENDPOINT_NAME= +``` + +## Setup and Installation + +### 1. Databricks Configuration + +#### Create a Serving Endpoint +1. In your Databricks workspace, navigate to **Serving** +2. Create a new serving endpoint or use an existing one that supports chat completions +3. Note the endpoint name for the `SERVING_ENDPOINT_NAME` environment variable +4. Ensure the endpoint supports the OpenAI chat completions format + +#### Configure OAuth Token Exchange +1. Set up OIDC token exchange in your Databricks workspace +2. Configure the workspace to accept Microsoft Entra ID tokens +3. Ensure the `/oidc/v1/token` endpoint is accessible + +### 2. Azure Bot Service Configuration + +#### Create Bot Registration +1. Go to [Azure Portal](https://portal.azure.com) and create a new Bot Service +2. Configure the messaging endpoint: `https:///api/messages` +3. Enable the Microsoft Teams channel +4. Note the App ID and App Password for environment variables + +#### Setup OAuth Connection +1. In the Bot Service, go to **Configuration** > **OAuth Connection Settings** +2. Create a new connection with: + - **Name**: Use for `ConnectionName` environment variable + - **Service Provider**: Microsoft Entra ID v2 + - **Scopes**: `openid profile User.Read` (or as required by your Databricks setup) + +### 3. Teams App Configuration + +#### Update Manifest +1. Edit `appManifest/manifest.json`: + - Replace `botId` with your Bot Service App ID + - Update `validDomains` with your hosting domain +2. Create a ZIP file with `manifest.json`, `color.png`, and `outline.png` +3. Upload to Teams via **Apps** > **Manage your apps** > **Upload an app** + +### 4. Local Development Setup + +> Note: For local development, you need a tunneling solution as Teams needs to call into your bot. + +#### Clone and Setup Project + ```bash +# Clone the repository +git clone +cd DatabricksBotService + +# Create and activate virtual environment +python -m venv env +source env/bin/activate # On Windows: env\Scripts\activate + +# Install dependencies +pip install -r requirements.txt +``` + +#### Configure Environment Variables +Create a `.env` file or export the following environment variables: + ```bash +export MicrosoftAppId="" +export MicrosoftAppPassword="" +export MicrosoftAppType="MultiTenant" +export MicrosoftTenantId="" +export ConnectionName="" +export DATABRICKS_HOST="https://.cloud.databricks.com" +export SERVING_ENDPOINT_NAME="" +``` + +#### Setup Tunneling +**Using ngrok:** + ```bash +ngrok http 8000 --host-header="localhost:8000" +``` + +**Using dev tunnels:** +```bash +devtunnel host -p 8000 --allow-anonymous +``` + +#### Run the Application +```bash +python app.py +``` + +The bot will start on port 8000 and be accessible at `/api/messages`. + +## Usage + +### First Time Setup +1. **Install the Bot**: Upload your app manifest to Microsoft Teams +2. **Start a Conversation**: Open the bot in Teams (personal or team scope) +3. **Authentication**: On first interaction, you'll be prompted to sign in with your Microsoft account + +### Bot Commands and Interactions + +#### Authentication +- **First Message**: Triggers OAuth authentication flow +- **`logout`**: Signs you out and clears your session + +#### AI Agent Interactions +Once authenticated, you can: +- **Ask Questions**: Send any message to interact with the Databricks AI agent +- **Tool Usage**: The agent can call tools/functions, displayed as adaptive cards +- **Conversation Context**: Your conversation history is maintained across interactions + +#### Example Interactions +``` +User: "What can you help me with?" +Bot: [Connects to Databricks agent and provides AI-powered response] + +User: "Analyze the sales data from last quarter" +Bot: [Agent processes request, may call tools, returns analysis] + +User: logout +Bot: "You have been signed out." +``` + +### Features in Action + +#### Authentication Flow +1. **Initial Prompt**: Bot requests Microsoft authentication +2. **OAuth Process**: User authenticates via Microsoft login +3. **Token Exchange**: Microsoft token is exchanged for Databricks token +4. **Ready State**: Bot is ready for AI agent interactions + +#### Tool Calling with Adaptive Cards +When the AI agent uses tools or functions: +- **Tool Call Card**: Shows tool name, parameters, and results +- **Visual Feedback**: Adaptive cards provide rich formatting +- **Multiple Tools**: Supports multiple tool calls in a single conversation + +#### Conversation Management +- **History Persistence**: Conversation context maintained per user +- **State Management**: User and conversation state handled separately +- **Error Handling**: Graceful handling of authentication and API failures + +## Project Structure + +``` +DatabricksBotService/ +ā”œā”€ā”€ app.py # Main application entry point +ā”œā”€ā”€ config.py # Configuration and environment variables +ā”œā”€ā”€ requirements.txt # Python dependencies +ā”œā”€ā”€ appManifest/ # Teams app manifest and icons +│ ā”œā”€ā”€ manifest.json # Teams app configuration +│ ā”œā”€ā”€ color.png # App icon (color) +│ └── outline.png # App icon (outline) +ā”œā”€ā”€ bots/ # Bot implementation +│ ā”œā”€ā”€ auth_bot.py # Main bot class with Teams integration +│ └── dialog_bot.py # Base dialog bot class +ā”œā”€ā”€ client/ # External service clients +│ └── databricks_client.py # Databricks API client and token exchange +ā”œā”€ā”€ dialogs/ # Dialog implementations +│ ā”œā”€ā”€ main_dialog.py # Core conversation and AI logic +│ └── logout_dialog.py # Logout functionality +└── helpers/ # Utility classes + └── dialog_helper.py # Dialog execution helpers +``` + +## Deployment + +### Azure App Service Deployment +1. **Create App Service**: Create a new App Service in Azure Portal +2. **Configure Environment**: Set all required environment variables in App Service Configuration +3. **Deploy Code**: Use GitHub Actions, Azure DevOps, or direct deployment +4. **Update Bot Registration**: Update the messaging endpoint to your App Service URL +5. **SSL Certificate**: Ensure HTTPS is enabled (required for Teams) + +### Docker Deployment +```dockerfile +FROM python:3.11-slim +WORKDIR /app +COPY requirements.txt . +RUN pip install -r requirements.txt +COPY . . +EXPOSE 8000 +CMD ["python", "app.py"] +``` + +## Troubleshooting + +### Common Issues + +#### Authentication Problems +- **OAuth Timeout**: Check that `timeout=300000` is sufficient for your use case +- **Token Exchange Fails**: Verify Databricks OIDC configuration and Microsoft token scopes +- **Login Loop**: Clear browser cache or try incognito mode + +#### Databricks Connection Issues +- **Endpoint Not Found**: Verify `SERVING_ENDPOINT_NAME` matches exactly (case-sensitive) +- **403 Forbidden**: Check Databricks workspace permissions and token exchange setup +- **Timeout Errors**: Increase `request_timeout` in DatabricksClient initialization + +#### Teams Integration Issues +- **Bot Not Responding**: Check Bot Service messaging endpoint and ensure it's accessible +- **Manifest Upload Fails**: Verify JSON syntax and that app upload is enabled in Teams admin +- **Permission Denied**: Ensure proper OAuth scopes in Bot Service configuration + +### Logging and Debugging +- Set logging level to DEBUG in `main_dialog.py` +- Check Azure App Service logs for production issues +- Use Bot Framework Emulator for local testing +- Monitor Databricks serving endpoint logs + +## Security Considerations + +- **Token Storage**: Tokens are stored in memory only, not persisted +- **HTTPS Required**: All communications must use HTTPS in production +- **Scope Limitation**: OAuth tokens have limited scopes for security +- **Token Expiration**: Implement proper token refresh handling +- **Input Validation**: Sanitize user inputs before sending to AI models + +## Further Reading + +- **[Bot Framework Documentation](https://docs.botframework.com)** - Core bot development concepts +- **[Databricks AI/ML Documentation](https://docs.databricks.com/machine-learning/)** - AI model serving and endpoints +- **[Microsoft Teams Platform](https://docs.microsoft.com/microsoftteams/platform/)** - Teams app development guide +- **[OAuth 2.0 Token Exchange](https://datatracker.ietf.org/doc/html/rfc8693)** - Token exchange specification +- **[Azure Bot Service](https://docs.microsoft.com/azure/bot-service/)** - Bot hosting and management +- **[Adaptive Cards](https://adaptivecards.io/)** - Rich card formatting for Teams \ No newline at end of file diff --git a/app.py b/app.py new file mode 100644 index 0000000..2f3f9c5 --- /dev/null +++ b/app.py @@ -0,0 +1,118 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +import sys +from datetime import datetime +from http import HTTPStatus +from aiohttp import web +from aiohttp.web import Request, Response +from botbuilder.core import ( + ConversationState, + MemoryStorage, + TurnContext, + UserState, +) +from botbuilder.integration.aiohttp import CloudAdapter, ConfigurationBotFrameworkAuthentication +from botbuilder.core.integration import aiohttp_error_middleware +from botbuilder.schema import Activity, ActivityTypes + +from bots import AuthBot +import logging +import traceback + +# Create the loop and Flask app +from config import DefaultConfig +from dialogs import MainDialog + +CONFIG = DefaultConfig() + +# Create adapter. +# See https://aka.ms/about-bot-adapter to learn more about how bots work. +ADAPTER = CloudAdapter(ConfigurationBotFrameworkAuthentication(CONFIG)) + + +# Catch-all for errors. +async def on_error(context: TurnContext, error: Exception): + # Handles uncaught exceptions during turn execution + print(f"\n [on_turn_error] unhandled error: {error}", file=sys.stderr) + traceback.print_exc() + + # Inform user that an error occurred + await context.send_activity("The bot encountered an error or bug.") + await context.send_activity( + "To continue to run this bot, please fix the bot source code." + ) + + # Send a trace activity if using Bot Framework Emulator + if context.activity.channel_id == "emulator": + # Create a trace activity for debugging + trace_activity = Activity( + label="TurnError", + name="on_turn_error Trace", + timestamp=datetime.utcnow(), + type=ActivityTypes.trace, + value=f"{error}", + value_type="https://www.botframework.com/schemas/error", + ) + # Send trace activity to emulator + await context.send_activity(trace_activity) + + +# Assign the global error handler to the adapter +ADAPTER.on_turn_error = on_error + +# Create MemoryStorage and state management objects +MEMORY = MemoryStorage() +USER_STATE = UserState(MEMORY) +CONVERSATION_STATE = ConversationState(MEMORY) + +# Create dialog instance +DIALOG = MainDialog(CONFIG.CONNECTION_NAME, + CONFIG.DATABRICKS_HOST, + CONFIG.SERVING_ENDPOINT_NAME, + USER_STATE, + CONVERSATION_STATE) + +# Create the main bot instance +BOT = AuthBot(CONVERSATION_STATE, USER_STATE, DIALOG) + + +# Listen for incoming requests on /api/messages. +async def messages(req: Request) -> Response: + # Deserialize incoming request to Activity object + if "application/json" in req.headers["Content-Type"]: + body = await req.json() + else: + return Response(status=HTTPStatus.UNSUPPORTED_MEDIA_TYPE) + + activity = Activity().deserialize(body) + + # Log incoming activity type and name + logging.info(f"Incoming activity type: {activity.type}") + logging.info(f"Activity name: {getattr(activity, 'name', None)}") + + try: + # Process activity with the bot adapter and bot logic + response = await ADAPTER.process(req, BOT) + return response + + except Exception as e: + # Log unexpected errors during activity processing + logging.error(f"Exception in processing activity: {e}") + traceback.print_exc() + return Response( + status=HTTPStatus.INTERNAL_SERVER_ERROR, + text="Internal server error while processing the activity." + ) + + +# Create aiohttp app and register message route +APP = web.Application(middlewares=[aiohttp_error_middleware]) +APP.router.add_post("/api/messages", messages) + +# Run aiohttp web server +if __name__ == "__main__": + try: + web.run_app(APP, host="0.0.0.0", port=CONFIG.PORT) + except Exception as error: + raise error diff --git a/appManifest/color.png b/appManifest/color.png new file mode 100644 index 0000000000000000000000000000000000000000..b8cf81afbe2f5bafd8563920edfadb78b7b71be6 GIT binary patch literal 3415 zcmb_f_cz=97yl$yB&9JzRh6h2tH#4qGlGguP@5VZ)TmuMREiEYsmAqpTZ7ZnE>F-ih-`S z)jiPabibc~4T5Do@MgZ}C5dq?7H{rvYr!LtVV;haHWm>H5pk+~G>pJtSPwz9!%QIL z?J6p?*$Q$^sbaC}3#mquX(;945bnpoc+%>4bmj2j*4KG@ZlhvIK1EKveQp-tp;sflS z4}SX;$jwoVae}M%3TBb@f-(BCG-m~}LW z311k8hKz8Ecm+M)P%mwS`Qda^pus{!e?Y+KDQD2B zWjuLo3{6=k`fmQI5d@(}*Q181Mj`he_jbr58C>@^+LzKri!pF}V7#<_PpQz&%C;U{ zmw+W{t0J1#nQ=&npU~H@5560!cFBrXbr9|2B0^~cU|iuMlNCdQc=W{4l5?D+6VaEh zTMw4Le|CpisEssdz5I_WB6-(_;8BOb0Ov8s8pGkEy3dRw%({?pOI-F=klY?eZ? zUVhJNclMhOiaUeo1=K6XJM&%_W3cuMl0&!|dZ*m;OnJ@X0hcbckvNZBg(+D^|Ij*W z^k!?ARMd55LmON%i4$H$oX@f6BX!4A;^vP8 z8cz4BuYM-<o;D&UDP5xiVZj*vOwL(Xgi^WuW~qbXAKq2Luow#G(c({?o;I6o^aPh zY8-5*rVevAtn+kvbMgF0e2aRCg<-9As)UjYZ6KflvEXw~s4oA9`rIcL$EwC#Nl4!Y z{Ra>{I}!nf;fS&)z+jL655PntETI$6U8Y}Ig2{rj%v@0jcn*%`A)a!{%}s7NBl@YZ zF=5*reV$RHd3{o<&n#+Q@`qDF353xaQpB`4xV}riJ9I9)n@3Z)XG}5(V{Q&3aR3@U zfvScEs@b=w&t&>>-{+3xqK!b>z!qBbNS|r5c*fsepeyv}`T2T3^Rl^VEuDJ791>m# z2v4z4^&I6;*?N?Y>{&QA68>t1^-&FL3ENmAhPS{0r|=(*lqbEP>9cOMLGp_HYhQZg z5|nV2{_Izd_;#CdtTqsobR}=S-qFTrJ-x;iS2#i#z#&uT!%~by2H7SHE59gi?MRJ@ z&uPeey)XN;6>?uj&+koIuhrru!~8?iOjP)pOk zZS*!=6WN?lHJ?`i{nB-e%fBUOPJ{yj=4Qw0yy+VSJ~h!ic41=jIWl86;2wQpJ$|c; zR^8lfv6@E+Ml{RZa7=y6$Fm2e{S_LC&C&1z_6HAE5R)AY98`77m2}Wv?2u>t#n znVG&}p_ND4RUXyAe0eXPm~gRFy97$f;5uNp5E%g15TTUE!!9}f9|!fPptQ}hXUJ-Lf~U%GJe zsq^FU`Ls)2UH98$x8x$=Tx0Fa`MacR@Y*8VNB4KDI$rXuP3tLT~d$yTUmB8m)7qg;fcbUj22v9YhPg)l!VIN8UIm#P<%(f!Xxw-=tty8Y31-^i)60)F`@KU!EX(mkf zQ)GeUGN)evp^?tyIxI4pQA!m=31izfrrvagzaMa~$#cu04I6IB;GGvc4WT-%YB+-dV^gTZZh%XO`b}DECWpOoZjqt9 zqktOLcvhMktKKW=LeH#wDjj)gZTsybRlro)>};szu4ZDya*m$j46iaD|7AtPR&)iG z*~&F{db|zcArblJB^#hfDfNHcBoXPrl|fJ_nY6|4PZvm8y%nhrBrMds%ST0DAoy9= zfGS2J3)T=H-9zf)Va%IxUrlHoa+k}BTWY5cQm5cg1m;kyx6jIVo} zncTNdzEOT^iXh`mZlRk{pWp?fwB`;UK8j^m!oH0&482 zLtYN=)+aYNZ4sk7|&V_eX z>Q)oVz#n+pJ})Bur(co;;PZGpQTW%-s;*VNl8sfFGp0FfZcJIui)lqu)fus9RW8x5>XRi#eKcG&_};xJr8+Kr5*T z`xf#w6!*t}>W)r?K}`cUBF1xChxm1CeQ~Iv!hpZ*aAfA2Oj+4dO7$ZY#HUkTBv7VZ z9{ummlF5yEz#3Q3qr@tUyEH39^e^h#n-ossc?E}3wwVM06<*ub6=g#PU8^A^X*rp* zHdbNBWv)qo)pwXWCP(eOSERnk<+Lwz$c=q_b{Oy9D-rhbvBhiC9BkT4BP$o|ked-g z13lVezZV!hdr*Cp&gcWv1m>P7>o8p1rPUe)cvFI#EF&G+lUbFSDxq3w?&ORaa)Y!@?0&a>GT8psQ{JX#@_+az{5K+M YJx2difYK9bhlEpZpl7Q49&GP9wA4-6No2JPavK^y+J&IdIIqnt|)iz#;q%0#|~})uPXtHpGg|3DT=Cm zRbOQmZzjp~Oa~|w3J0d4$UMjUP`eo9-%ZEed<9c*o{#frSUWpe$h)9<7f||JElr8%Q+a+LHNJ~kNO5B zlRv;1hxJ`;YEbQ%GiTGTR{shYbEe%;Xrq2t9*a`EVNoJ89P+!W;^dkhG3QK~lh@uy z_@!DknGSuYuSg%;OK8pl!P9F+PR@yY6bgl7VhU4=M!!cg{}TWJ002ovPDHLkV1nXO Bp2+|J literal 0 HcmV?d00001 diff --git a/bots/__init__.py b/bots/__init__.py new file mode 100644 index 0000000..d6506ff --- /dev/null +++ b/bots/__init__.py @@ -0,0 +1,7 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +from .dialog_bot import DialogBot +from .auth_bot import AuthBot + +__all__ = ["DialogBot", "AuthBot"] diff --git a/bots/__pycache__/__init__.cpython-312.pyc b/bots/__pycache__/__init__.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..af519f34aa62ce9d1ab513097597c9f71fa643d2 GIT binary patch literal 311 zcmX@j%ge<81Uo#pXB-C7k3k$5V1hC}s{k3(8B!Qh7;_kM8KW2(L2RZRrd;MIW+0n6 zg(aOSilvfOlkFuVP^l*4El!ur#GL$er~DEihuyKXBm>ASVg^bUu>c7_P1ajnDPSe> zNl=Mf9Em_>U@l0>E%x~M#GIV?_>~Nwfn0`P;rgM)sYS*5xrupcX+^0iddZ17W%c*Ou^=H#ip+ZYAZL~_-ytGmnD8pptCYf}8)_Z4T zH?C0HpeaP^gIXWOH>rY8Qs2z8FD^wA1`2{c*tc!BAtg_qduO+a8ZYd*-~E|$&pF>Y zXMZUavIxdEt)CWN*AV)HFFHeQgVrK|+sH*O_ECUMj5(HkDUeM$P)sFIO;zm4z7}N6 zjKGSo2Zm_~toqp?XXXHFZpP0C1+##Wgw7&YUqP-x3U}o6EwihZE#)5YMF}U$IgiC9 zIq5wUg-e7oJNBaRLTtw*G2UZ@rZ9K0N<+92ldCa&q)mCYAI$^mIv#%XeixySke}99iERcZa{T?2$p^+-IhOARj_PrS^?rtpta2R@&~vi%`Bh2 zM0_U-NHLCz7l|FPB99lu+jB&b6d}OHx31bj#V6(B#ae|Fk3vyI)%Ym?gn9E}5fwO1AO;j6k7kF z;Hie?kq)Q-dxh&|mq7f*X>RIh@TEscGx}EFZfdA!;8yKs?W=Pe!{d$N@r~h=jp36U z!xN3+iF@4>n*%Ry3><3=9NX-_+>)iizHKD+_B|BB01ZtYmOG4_c>VZfKmM^F#CjG_ z_G#-m9k8>PSS7L1@1msH6R|0E!PScKQft>WoFBgLda#{yW9`-#idG zptf@$$)$ZuG{KTexy~d@tD%*43{kqGl9O(S5@?!Mg+@^Hajhh!T#9gWzdPg&0yA-R zg@hI*tP+KcSY!!uVflqU)a`d+6qK-2AOvxLc0Ie%HQMMJ-OTsgn!Y)`nI8hQkw4tX zAKu6xZ{&~vaQM8hvW_*tr2=nIp&by*O0t#} zL~hmRIB!|+S8YE%p*#yI?@pAHCz=T5}K1-CRQv>Td3d>`5#rUzh`afaU% zSuFaZ5WN2}{txhE1vMp2ox#5kzt&VafRsG&d^01Uj(P{T3;}zxYTxFek*~{*L*q@E zFDB2m_}m=Q)a>0?wh>Hp5LAi>W=h?O7S}3XI4|r?x}gouRz2S(wCqHb?D33yk@W33 zPnl-I<)nAd0fQzS*uwYZz=HU@W99U=l`h-)(%yqlDg0ZcCVh)i^LVlB=%>PdUG=sQOhVt^L{tuQ literal 0 HcmV?d00001 diff --git a/bots/__pycache__/dialog_bot.cpython-312.pyc b/bots/__pycache__/dialog_bot.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..982f48ae646b0b7cd4c297de07a72addc534c634 GIT binary patch literal 6308 zcmd5=U2Gf25#Ht9QKWbhDUy;bSwAORKW1c+u`Sz)WXqBD6T6aKyRy-UK|)ZxlT4XE z%H7GbSUL@Ar?rcui3>DJP!uT8hd4-6_^krkhx$iGfWDBCnj;;}&SCV}QDftfb*^d;JF<{6=4+H@IBW8Qp; z^;`5hOpf+ji)0p%RbEEtvEMdqNB;yTO&cVpA_=uYLKR7H5(V#`eqfYxYcC(`EW~+d3#lDl3CW1z@3w zONH{w4I&Z_F2cM1OV|>>!ea=-hO;6WSo(7Xw#4huDw*GQ7a>6W2sX)E)DrbmhJhYQ zG+VrVb3JD(3>}4U62tb*{lBapOQEen{r7qn(SG#WF?hS-wQfhU77mHNb|jv-V3LW= zwP%)Wjgp5qB~8CZzsWljMskhOWER3{>GOV$0d~N%wWU&o_`LIJGX_h&MH9>I2jgQj z3D&$|dfBv?`!3)Gbe@_-3-|+S0fPgMCY&Rp7#F#4jElztWF`B!MDIjg6c=}LQXIVS z4DrYX*qz*h7$?pt#zTVcr;G7%gg0UGv z(p|;@^H3`wn9hK1l@Z7YRI4loL&8)rbXK*_iQ$+W5YC6xDoKd(fnY8^zekJ2S|Txo ziXpBC0-;Dyk^%w4)m{;a5ydS~Bx(x&kQ-4Ou%!>6c=S*72wZ}$vr1gda0yqv;`F4Q9>uvY?c8@g`IpnH&M}1@OR-}aXWf$Ts!wrx(@yWTd8mYyJ{Ri%CJLUKCo~37ox~NSl1iJt{h7>d9S^6EtG29y~ggj%hoDv zYnp9cW7{&$7R9+U?cAB_39UMLh2=kH`G03CNZ+reD2(_b9;(2vItRPa5+3ZJ-`)K?iP7mW3$fDG;sYIPfJw=Jce2({e;_qE=U^ zS2?3;O5G^!hdtkA@!CqlgwR_hr`?97+>m<&&ImPWsyFH5An|6CYa`#@} zrnvgkuKrcmLB%zcb`7n%MlKI$+IJ}J{pt38rF|^jKK9|U)pmc1ZM$!Gy}9u7AK!J< zFR@qIRI`80@qDJEPw5y)cML2iS3Aa)*6}ylzq=qu^eK(|(v5(ctBu2odpPADzU$t) zG=6nF)%Mhy`{_($r_$J)ZtP9%4X!p$DekG1dn#)|HT4e}RNJIC99nY^7d3ogwedy8 z{bI`fV#ZPTfS@Ho=ZwR>WV>q1IGQq!j`a%KRs98`tExXEMYbMs$1p+T+7bNdzI#d| z1B3(8FCVhV(Llg?vH;F~%e$7hry2*=*h8Cu^AA^@&nfJ4AG6PC;5-T7{0A`nPwGZH z@jHWuMk>+GEqDacHyauWy$6q&=$pMgK;LS^BQ$+$8%^j=4DD}qd4Rs{z$0e*b`9a* zt|k22b?t=i!y^{@cE83yMEENR1AoP&m9$oxwbG)MmDbVi)QW2mjkZvCTzIt3e8;iDCOr8%ohGt2WD0#}^KcCMlHb*`=pZoLX-{&!XVzcb3 zjJS-K~8MHgVzJE1+s;F-T~RdVK}iB_GCVfFtkIa zSow_;TdN_RO^hWqJE7e1J>O3?oLFN|<{G?x=-L6L`q*!3WZYYp#;%S5Gy|o$yVLG& z#oeEF_b(q?b$<^4t=?0jQ9HU zUm*B=Mhekw{KI2c?QmB(H6M=faHSfGi$c*%$oSRT61i6L=%90@dlq$7kkcmc+;F6;%G>u*uC&Da}c>D`ExO#0bR2AjU^#Mn=XWW*`dyO}aAZ literal 0 HcmV?d00001 diff --git a/client/__pycache__/databricks_client.cpython-312.pyc b/client/__pycache__/databricks_client.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..8f861e531ec10c0ecf7c1ab20266d7670aae937d GIT binary patch literal 3725 zcmai1U2N3Y6~6Y^GoJq>lb;!qkO3C54v-0d0;anQfoREgOI9Up+YKtNV>^k5i9Kt3 z0-1VRw2M}Qx-C>IHCX{#9{NBFmGYRjFG$-5q`r7Wq>0?Elxlfs-$JueTb_EZJ)Q|6 zRj)GV-s^Mjz2}~v@7(-38Vw<6@v%S7j06$-8c)1N)dw3#VX%NiBocWv;r7G?0dq1> z&iW>N1R~KnB>HY3k$OnFt0t&-POg8i;XYbjH(-Uhc0PzvNFv{idax(#o!cvRo`taXH2i$<+c1GFAd@f=2P3zcB+LkJW?)&T<^0~L*>B=9&9C0b z^gXQa^N8?9uy~;DNf1tY5=tXaOKLJ!2S!Yj${8pI3zX5d)S>2n( z{qCiU zBt@u4FP3mT`Z4qyzXkFxBCuCc4s*7&JCGUo$TDdXA}RVrN~G^nWgp0H-mShS*YH$x z6NnDqBfpn9A47)y7$WIcHq;mMOTiV0=TnwaOG>(p=PrH8SyGpRkSSglK zWl_k?9m+V@D!o?Jl&mZnQ`rKq@w0l?xpUTG9yI0)QZ{$hC}p;nn$~#5aHZS@(lud* zSEi*B!{_q{`GTxN&_wm(WKI$wIj|w^wwIxu_L@7NV@JW zW+^xLVsI~;D&UNya=NOpv=)a?S^|If6}7yEZ*DW1q!(00mvn4f9HVKi1v6rvqVBe( zzE@)DWdVGiJ_+RS=-=1TT5CtGb#JwGuhp{ezGB6WRebAywDa)l?$Om9@2+#t<)cqFtRB_x@Opc8Cf?}ZQl0lh9>c|+oQ|`tY1Jz15-@WLjxyX zFcNKPB7x)cZbOa$#5O6JDAT4-gP%!@)EltIq{k6UkJ(fa(wrr2N>lTa9S~H-fV{S8 zSt%3^Z8u0_W*(Ha{d}PSPzhM$_$LiKZ5Jk3%4TN4v;4HA+kpZ%x4#$EE&0JlyH1LYk{-wA_l{jNk8E8mT5vt}xlP z#E!*Lt8H&3@V9X5YIsD}XM0PGvy#HbIU27!kkkq_(Vml+b7 zfn4NO$s7*-ElrF-e4rf8q%BP*Oy<|bFNku`z^Ul8T)Q`F_9A5l@1T2|B5wQp1Afxm zp9CELJ*pfM10rM6crP({mn?@ty0dwXH_Bl+*wW09$(Z3sA5G<7x&aH{Zf+AX5lbSaUkzhVx0uNXa3iezMEP8E1V2EZ(}K;GtM zcb35jTMC?!cumsMELn=2bu<3pMU36iVJMQSCYPM0EUERtEdZ+|9!jWQ>c3FOU{@Do zMoL4*D@IyY(#-_UrhkLj77ZIf!!hZ=yxm&Yv+>%}?Y3#j;P8bwUamh2v;+9Spm%2- z-YX7fxG$SPUY(wX7n@DV%9N_%aMrL_?G}xfb%}#-oNn+&QFj&h0k*QSO>l(+c@t*E z+(&^pSubZTSH1UZT^&2>+S}7^KVH8RpBQw!!{PFD7|)~Fu$-s29mOEunCe(l7YRM; z)~*D+s`8w>mFK_GYfV$uA4A(nVNLWwfGD?$_3f$k9j^8rwvL>8+BaHDjaFjMx(Al1 z+R#XKXynP*O7|s;NvIC zja33`i9@&FtR)7jiGf<;P!<1OUZ!d%E>urkuo9Q8@BI)yRG4Q>{8r><#OgY_^n;~u zTOFrXnA2-r-Lg#wdG49}djoDRYcC9db){^NjqMt|a#4FLp>2EEUmsP7{ zY=wF6S+aW(ZZ-fni}zSe&-yX6;~nBZ>uJ<}k$9o+0lR(FF>;JpAV!9$#j`OOez}hr zIYNEeKOBO|pNkX^$6u*QRn0q*t)0P~BbeYKuizeAMZlv~S?q^PV2M7jvL{0{cK`z5jvR3)A1= zxvK&`9iCjcplV?tcC1;R2X*m^o5GHcFYEn+dBZiuTiQFY4p&Qk2at6iK@fjIv9D3& SuV~<3)G$Hx{u5#99Qa>GBBNCR literal 0 HcmV?d00001 diff --git a/client/databricks_client.py b/client/databricks_client.py new file mode 100644 index 0000000..30465af --- /dev/null +++ b/client/databricks_client.py @@ -0,0 +1,77 @@ +import logging + +import httpx +from databricks.sdk import WorkspaceClient + +class DatabricksClient: + def __init__(self, databricks_host: str, serving_endpoint_name:str, request_timeout: float = 300): + self.databricks_host = databricks_host + self.serving_endpoint_name = serving_endpoint_name + self.client = httpx.AsyncClient( + timeout=httpx.Timeout(request_timeout), + ) + + async def exchange_token(self, provider_oauth_token: str): + + url = f"{self.databricks_host}/oidc/v1/token" + + data = { + "subject_token": provider_oauth_token, # replace with your JWT token variable + "subject_token_type": "urn:ietf:params:oauth:token-type:jwt", + "grant_type": "urn:ietf:params:oauth:grant-type:token-exchange", + "scope": "all-apis" + } + + response = await self.client.post(url, data=data) + + return response.json()['access_token'] + + def create_databricks_payload(self, text:str, history: list, stream: bool = False): + + messages = list(history) + + messages.append({"role": "user", "content": text}) + + payload = {"input": messages} + + if stream: + payload["stream"] = stream + + return payload + + def parse_model_output(self, response): + parsed_output = [] + + logging.info(f"Response from openai client: {response}") + logging.info(f"Response output from openai client: {response.output}") + + for item in response.output: + logging.info(f"Item type: {item.type}") + if item.type == "message": + parsed_output.append({"type": "message", "text": "".join([e.text for e in item.content])}) + if item.type == "function_call": + parsed_output.append({"type": "tool_call", "arguments": item.arguments, "name": item.name, "call_id": item.call_id}) + if item.type == "function_call_output": + parsed_output.append( + {"type": "tool_result", "call_id": item.call_id, "output": item.output}) + + logging.info(f"Response parsed from openai client: {parsed_output}") + return parsed_output + + async def call_model_endpoint(self, text:str, provider_oauth_token: str, history:list, stream: bool = False): + + oauth_db_token = await self.exchange_token(provider_oauth_token) + + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + + openai_client = workspace_client.serving_endpoints.get_open_ai_client() + + logging.info(f"Actual history in the chatbot: {history}") + + payload = self.create_databricks_payload(text, history, stream=stream) + + response = openai_client.responses.create(model=self.serving_endpoint_name, **payload) + + parsed_output = self.parse_model_output(response) + + return parsed_output \ No newline at end of file diff --git a/config.py b/config.py new file mode 100644 index 0000000..753007a --- /dev/null +++ b/config.py @@ -0,0 +1,20 @@ +#!/usr/bin/env python3 +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +import os + +""" Bot Configuration """ + + +class DefaultConfig: + """ Bot Configuration """ + + PORT = 8000 + APP_ID = os.environ.get("MicrosoftAppId", "") + APP_PASSWORD = os.environ.get("MicrosoftAppPassword", "") + APP_TYPE = os.environ.get("MicrosoftAppType", "") + APP_TENANTID = os.environ.get("MicrosoftTenantId", "") + CONNECTION_NAME = os.environ.get("ConnectionName", "") + DATABRICKS_HOST = os.environ.get("DATABRICKS_HOST", "") + SERVING_ENDPOINT_NAME = os.environ.get("SERVING_ENDPOINT_NAME", "") diff --git a/dialogs/__init__.py b/dialogs/__init__.py new file mode 100644 index 0000000..ab5189c --- /dev/null +++ b/dialogs/__init__.py @@ -0,0 +1,7 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +from .logout_dialog import LogoutDialog +from .main_dialog import MainDialog + +__all__ = ["LogoutDialog", "MainDialog"] diff --git a/dialogs/__pycache__/__init__.cpython-312.pyc b/dialogs/__pycache__/__init__.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..43485530f917cf702e30409e3ef53b8ed8a68951 GIT binary patch literal 326 zcmX@j%ge<81Uo#pXIur+k3k$5V1hC}s{k3(8B!Qh7;_kM8KW2(L2RZRrd;MIW+0n6 zg(aOSilvfOlkFuVP^l*4Egqlz^!(Bim(0YR{B$6n%QrDI55g;A1}Z6H0TOGcq!M^!v KM)o33pd0`t|67^> literal 0 HcmV?d00001 diff --git a/dialogs/__pycache__/logout_dialog.cpython-312.pyc b/dialogs/__pycache__/logout_dialog.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..46615e6d14624d01c4c05e664e7e70bd5039402d GIT binary patch literal 3194 zcmbVOUu+vm8K2od_QsCu#NNc$rXjvduhfv(SLu9BoIm}O1atyQd&vo~gqDqWoY-A^ z!|b|kf{&|c1wlHb-ow$}+dbw~PO8Kqod5|Q$`dbH0$Q`WlTLyMc(W9$Mt$O&T{}&- zt#t5N{$}Qz*>7fk-@o~DEEYjP((gZ6{X-a`f3QK9#7<}TYtXraG^Al2)o>BlxFT2M zi#%pBP8VuoQS|z}9;$_lVT?HRb)*TmkR}rD9`EZZO6#Fa_}^@h;oZLZimDqcOAT6o zgP0B7a^rJGZOy2Yy5&!~F~6NN>K3_fWw0B6(jD0aSNRjyX7^`-SS zVwU`!d>-lyc7Fh!pTPwPDq;;4ISmtD;|O0CHc=UC{H@-ipb12T!-ljF3712jJPdNl zlQWsWRsvZMla)EG=ax{6Y@+M97+pk}h}-*3W5sB658tDC_fsmG8!s94I)QT zNT~yecRto<|J?$x_wXHLd1A&auG?**UqlgV;(OSaVASNANZHr3jy}W}GlDCcjWt4P z53F(}MXA&)mZDJj6v3om61_~Bxjl*7qbMa^HBCh^1rVmw=^Kxg9D~@*)zo^qOo^5) zsrpsJRO_WxvXPrNN{t%eZss)AQkQ9^bj8fg?{_a4)*_)-D-u%+dU^cwRsHK-VO9mDfW(jcixu1vNicvDgXIMc2oG9bbJ?P;bs+plPX%t z4I66=RnrYoVu2?eLY7_XCj57Q4D3Dp4TQl5nyA{r5QO{m1lgP5(mw)Rf63k9e=3UT zExyT9uE{SRu*S~?um_yBiJRQLj*WufEuu5%oizMIAMw{Pjlovn^bs(L1qN!YS@Z~q z+&H+zGFhqAI}l|=+6QtfGd93hz|B~6(0&-0EF!^9=^He`790h(ar_XN0euW^^E|M< ze=%4;PYvC9``x!cxW1Em#p!=#GxYape=B)(^IM+|9DDaW_Q2TIckf@@893|2&u&X+ z+hLSAz1I$*zH|7o`2uLUIWU9o;z{BCp&9%!o)Uf|&fwqS8C=@;R9M8k(!;$8o&Otm zuE_!J+#%fgF7DkpcqZy20n{lwxes>&;DKPYD)k2GA}$<2oOz#*bT=Y=ppS!mVS(UL8=}!PiL~qZWth|n=(;V3QWQ9$|;0yV|t|d#rN=P%VAAb7778n*BQSB z2G=m#hq}+%g+)z(sw^}DWU=`0sLDbqKpN_XEer2zDliSh=j{G09GQ0osLK2(K#+Kt z7X|2l*yl@uPhr@ttidpV^6+dMz4){ajeA}&d0IxF{-SHG;I4~k9exme=R2t{!ct#` zzvzbiGO%&8@ZwQK#@ML2kH4 zOjBJUAY*Gp-MB`m8(~RbF)h_1F26!7${d(gK2~NG#f`Bo6jp&6P}ba7nHn`^jT)fU zT6ejM=2yZ+Gq!J9nq?v@O7&iDm;{K8K3 zLMu7sBuDM!sFNJClVeWuw4FTdB+uB%Gds!2&G}Y5dHecb;v=p6jFVrm^9xS?lAXWg zvMBahkg*p{}-bFU?~`(A|fo;u|W=Iz0JOFr(%Cv5qIBahqixFb*5@{}Xb z+VX5me!-DX+48BU)=b#)#BNwfB_1K6KhgGB1f@ot!ISpjNoVj?d+^om{H2}2%TC|r z?Y_&c;g_A^343^AXSm>`3fp~!_5kWjw?))DhugiNW3L@S{cqsCRtl2gYXHtq=Fvg~ zKj83!AUxn-V*Qu#LP&UUGR68cctH{#%!Z-=rx0dbVm#AJzs!siQIwjYHFV|=y^8X~ zhN^eGgEuYeJxD0aq;AMquEHyg@-Xm9D`f%eRS!4Tc`Jb{=!%lTevx1=iF3Y%+2h4~ zcd^82z6osa7HV^XnEPN9KoN7F@DEwj?iIyzc&pY%to>uzZ?`Y24?RKqh$t5B4>M_D zdv1{l3+*E@acXP$5rT#$yZcaZNyj(o%f(gnp`Nu)`xy1TM0 zl@5Z$Xcg2&T@;O7q-YbMKwQ{G`%x4M@=@DKe4s$V4IM1AbpQfw(e$GLQ8|e0{`Ad~ zOUg1`80Y}|=FRcuy*D%OnEjK@R)L@lAN*+U7fyu!N*Y=*6p*!5KrSK%F*t^Tq}Ii7 zTo=^E^+7!*ZGDW28-j+oF=$lVRLm4N2hDLy(4w{tv5I(Quu{d1F>Blww5hl$W{*39 z4i(2SXV9rbtTWoG2~et`x4qQTu&d@PIFDnbL!*eXoI^|n>$sxN_Xgcd*kAci(wOw? z6w5Jz<)_4u$SPZgxx{If7eZo$OQ>Bi?l=KM{7fho+aC$VxLKvD*fAxt^FSi9XGEA- zd1&u~ICq%m;`5^7-q4t0MKO|?75rGS?hT8P(-Cp$=+Zn3ZRh@w7@FoI;ZwqJEW##4 zGUp&S%Poj`ZqPQw*wbt*oEJV1)zblV?I3{f!zr*Rh#3^rF__geI{5VPQ7pw6Si?-k z3Yx);@tieiWK67SM#q@XQ9*M7w*YQgL1%EVVhZ^yl!})_kwigNN0>__SP+>@(1}o- zRUAxFUV4rbM5P)w?=&b!vk7LNizLMSyov?rrv-IaU9d=TSMz;g;|6UkfRm$HtI{F# zQ+yGL#X%8z@+FCWScy=n9o@KsHV0oR%*QB6B@s;(sSVK5E-p%9Xcg&`C~DT`mFlRY zgliaswTlJxMH-;jjzgOWby<4d2702kTEyxiR;6L(v=cqrXc9AUxQyYd@k&8MO>dCP zs?8GXH7u!1>Nm`=7yD7PS;I>;=u=y2X_zKC>}**ou|=E7m|#X3V};%_#yYFhBr9J3bghGCbC+DnC#Do=t7!hNvViF^977S1^pBO0^ zVU=PK7Uo%=Cwr%uX%HbI(lo({StX^gPc2nYg65Rf7NpA_+Pa z4zq&5@q7)LZz6!)CdJByz=04N@E|<7RLYhO_7+whgi;Y=7@ARC5l^nP;$joR0?*O{ zYyqq;fKyUzq4@|M2A`F0ZXxmlNr8@zFn*Il32bae6_*DekN7PF6!lOnG#!qG1c9an zBBD<%Z#;ahm-tJeHy%pN%_7&>9Bh%WyLfthtXPX10ZpgNEelYa@kleOUYTI{hDBU(ix_uc>z3d4{o`CGx zBYF0$E~P!=KyLk8L+3qbz3gn4obC6VZJG8R8SjBiU0bH1^}fZh#kFQIH`>Vx-uo`p z+V{KO)bOk6mXoBXDQmX=$nm!05+&FArCR@8b626~p0z&Xtjku_%2nM`RdSA|-8R;bU-O>3Yl+zE1=GUf(-u7O&T5N8;q#Z%!J z@GZJSZ4S6VaM+tmdA>~132f#J?gmF~n^%EAP;_EZa(6eCC+RUocIf|;wCG67N?LSU zC4rJfr@xtE3Js$3zlBeP?+|=#@L7MOU&MZD5}XlticO3h^;hsdSWU6dvLdZI5p_T@ z74C#$f`O1I@;ve7ie6wv#X8LrKT~iLs)K1HgH(cKDtf*H8hn?!s=64>v%)-=5Lkg+ z6)<#u8=27tV7YOFEk5pO_P`Y3BLH6!yUaGX$<5oP=IwIxkkmYM%__fiRC?)Xy7`#w zJ+@;1!qNDv6PLfedg2;$ck4vPRU^B)Bv+U03P`TN<)qv@EcFgwJC=4$$kvIJbt3EC zDtr4RZ{P1$kiQEb;oh5K$3V(FkaM8A<9I!5omkJh$JR6Ek@c)&;Gu8=SK?A;EkBTW)-%m zu#Lb+ES97kZ^5_r2e7?@#o}@n3-*Sh8*LqkNQo9xn0%W+u#aLYQ|g6Or9i*=)}7UZ zEiWc@z?+iT!QTmY2fE^7RY`j7Cv4c{z$N7|6sl;1Ig+7Zv5HI-7~a%u2}lQ3!0YPBlm z)7l56QK>r&9_1a1+>Aj;44z8cM<~L&lMwUInqD_e73xL2h|cIwqD2f^z7O9I)W~F< z967pNbCO%|g&@sNaErctQtpc+0?QqHXCe3VMFbcSeWBB#NGvoRV|^hJ@^9#kb0jwm zEZ2~SGGLLg`qK8zz(mL-{Vv4_k!T?%7GfDhHa-BG%l8nl9e`pJ;Gqc_bp92sRFGCa z&M9WCmnSbJcs6s3kg=;dHhG*VZZ#i{a||0RzE&Y#io9a^)|oInPhPo-iKqjw2E_oh z9Pc-mf+fF$NI}d!&%h4+79giCJVg|<7DW}z3M9)=Gv;D6ZHJ!a+6~e4kraI#MumSt zCZ`6{42gU_lDB8Yo@r{8n+Bw&0lDdUsp8!TFtfx)(bW5IY+0!q1`en~9$+JuL3`w4$v}f-H zD%08hLGt~i+&L(94z60|od>0z2X9ZOJCDdMM=sbiEnDT5L8)a>Zkdo;CT{P)`^vG@ zaYjBKm5xW#ubfK7xpd3CTt9!ooHHWeRnsNabjdY4rJ9|qW2?`nYer?~Xv#U7Y4ppD z{ZeCpx^Y1E45XX`IV*BEKS?olKOa#n_bl?hiXQ+M+XQ})wN5_By89mFDp2bycs)}; zzYZf?#sKvUJh_&3tq2JbmN7i~Yb{XG0$%eymoom(?Cts4bQ?Fyx9re)`}ZYy_wgP^`i$`my z8+9InKZ{4b)QvtD;2#_IdPdvPCk=SCf%>Fz5b#f{@MshDX>}XHcj8eW_317$=cW^n zwo*5@=m}oc48%8g;PGneWzL+b!cR z{jFzqkK6RO{dla$c)P0>@H;v@ZZO`_Qv^3*m~+QWR=#615Zpx(+(TCOSMq8`T0zrb zkP9(*LMv99{tnnv;X%C{c7`XhhzCc7)OR6IT%($m^2C@FTj7NoX5pPn`!U}My#m+} zdMTg4l1GC2WG8vKFbQD&9Lnjo8G0`3a|o(TOM6Zab#_~K&HyzWdQ)f4L@+bz82s8o zu%p;w7)(7kzJ{RA)zlewq_*$>20_jHH3t4FMaKz>W117E7a}o+f0Uj%{DGI`315 PG2U^Z;cG-{mCt_xj(P)) literal 0 HcmV?d00001 diff --git a/dialogs/logout_dialog.py b/dialogs/logout_dialog.py new file mode 100644 index 0000000..7fb418e --- /dev/null +++ b/dialogs/logout_dialog.py @@ -0,0 +1,42 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +from botbuilder.dialogs import DialogTurnResult, ComponentDialog, DialogContext +from botframework.connector.auth.user_token_client import UserTokenClient +from botbuilder.schema import ActivityTypes + +class LogoutDialog(ComponentDialog): + def __init__(self, dialog_id: str, connection_name: str): + # Initializes the LogoutDialog with a dialog ID and OAuth connection name. + super(LogoutDialog, self).__init__(dialog_id) + + self.connection_name = connection_name + + async def on_begin_dialog(self, inner_dc: DialogContext, options: object) -> DialogTurnResult: + # Intercepts the dialog at the beginning to check for logout command. + result = await self._interrupt(inner_dc) + if result: + return result + return await super().on_begin_dialog(inner_dc, options) + + async def on_continue_dialog(self, inner_dc: DialogContext) -> DialogTurnResult: + # Intercepts the dialog while it is running to check for logout command. + result = await self._interrupt(inner_dc) + if result: + return result + return await super().on_continue_dialog(inner_dc) + + async def _interrupt(self, inner_dc: DialogContext): + if inner_dc.context.activity.type == ActivityTypes.message: + text = inner_dc.context.activity.text.lower() + if text == "logout": + user_token_client: UserTokenClient = inner_dc.context.turn_state.get( + UserTokenClient.__name__, None + ) + await user_token_client.sign_out_user( + inner_dc.context.activity.from_property.id, + self.connection_name, + inner_dc.context.activity.channel_id, + ) + await inner_dc.context.send_activity("You have been signed out.") + return await inner_dc.cancel_all_dialogs() diff --git a/dialogs/main_dialog.py b/dialogs/main_dialog.py new file mode 100644 index 0000000..22b6631 --- /dev/null +++ b/dialogs/main_dialog.py @@ -0,0 +1,144 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +from botbuilder.core import UserState, ConversationState, CardFactory, MessageFactory +from botbuilder.dialogs import ( + WaterfallDialog, + WaterfallStepContext, +) +from botbuilder.dialogs.prompts import OAuthPrompt, OAuthPromptSettings + +from client.databricks_client import DatabricksClient +from dialogs import LogoutDialog +import logging +# Set the logging level to INFO +logging.basicConfig(level=logging.INFO) + +tool_card_placeholder = { + "$schema": "http://adaptivecards.io/schemas/adaptive-card.json", + "type": "AdaptiveCard", + "version": "1.5", + "body": [ + { + "type": "TextBlock", + "text": "Agent Tool Call", + "weight": "Bolder", + "size": "Medium" + }, + { + "type": "FactSet", + "facts": [ + ] + } + ] +} + +class MainDialog(LogoutDialog): + def __init__(self, connection_name: str, + databricks_host:str, + serving_endpoint_name: str, + user_state: UserState, + conversation_state: ConversationState): + + # Initializes the MainDialog with OAuthPrompt and WaterfallDialog. + super(MainDialog, self).__init__(MainDialog.__name__, connection_name) + + self.user_state = user_state + self.conversation_state = conversation_state + + self.connection_name = connection_name + + self.user_login_accessor = self.user_state.create_property('has_logged_in') + self.history = self.conversation_state.create_property('history') + + self.oauth_prompt = OAuthPrompt( + OAuthPrompt.__name__, + OAuthPromptSettings( + connection_name=connection_name, + text="Please Sign In Before Interacting with the Bot.", + title="Sign In", + timeout=300000 + ) + ) + self.databricks_client = DatabricksClient(databricks_host, serving_endpoint_name) + + self.add_dialog(self.oauth_prompt) + + self.add_dialog( + WaterfallDialog( + "WFDialog", + [ + self.ensure_signin_step, + self.api_call_step + ], + ) + ) + + self.initial_dialog_id = "WFDialog" + + def create_tool_call_card(self, tool_call_info): + + tool_card = tool_card_placeholder.copy() + + tool_card["body"][1]["facts"] = [{ "title": "Tool", "value": tool_call_info["name"]}, + { "title": "Params", "value": tool_call_info["arguments"]}, + { "title": "Result", "value": tool_call_info["output"] }] + + return tool_card + + async def ensure_signin_step(self, step_context: WaterfallStepContext): + # Try to retrieve the token + token_response = await self.oauth_prompt.get_user_token(step_context.context) + if not token_response or not getattr(token_response, "token", None): + await self.user_login_accessor.set(step_context.context, False) + # No valid token: begin OAuthPrompt + return await step_context.begin_dialog(OAuthPrompt.__name__) + await self.user_login_accessor.set(step_context.context, True) + # Token is valid, move to next step using token as result + return await step_context.next(token_response) + + async def send_response_activities(self, input_text, response, new_history, dc_context): + new_history.append({"role": "user", "content": input_text}) + tool_calls = dict() + for item in response: + if item["type"] == "message": + await dc_context.send_activity(item["text"]) + new_history.append({"role": "assistant", "content": item["text"]}) + elif item["type"] == "tool_call": + tool_calls[item["call_id"]] = item + elif item["type"] == "tool_result": + assert item[ + "call_id"] in tool_calls, f"Every tool call must have a tool result. Call id: {item['call_id']}" + tool_calls[item["call_id"]].update(item) + activity = MessageFactory.attachment( + CardFactory.adaptive_card(self.create_tool_call_card(tool_calls[item["call_id"]]))) + await dc_context.send_activity(activity) + return new_history + + async def api_call_step(self, step_context: WaterfallStepContext): + token_response = step_context.result + has_logged_in = await self.user_login_accessor.get(step_context.context, False) + if token_response and token_response.token and not has_logged_in: + # Set flag after first login completes + await self.user_login_accessor.set(step_context.context, True) + + await step_context.context.send_activity("You are now logged in.") + # Do NOT call API on first login + return await step_context.end_dialog() + elif token_response and token_response.token: + try: + # Call Databricks agent API. + input_text = step_context.context.activity.text.lower() + actual_history = await self.history.get(step_context.context, default_value_or_factory=list) + response = await self.databricks_client.call_model_endpoint(input_text, str(token_response.token), actual_history) + new_history = await self.send_response_activities(input_text, response, actual_history, step_context.context) + await self.history.set(step_context.context, new_history) + return await step_context.end_dialog() + except Exception as e: + logging.error(str(e)) + await step_context.context.send_activity("Agent is not available at this moment.") + return await step_context.end_dialog() + else: + await step_context.context.send_activity("Authentication failed.") + return await step_context.end_dialog() + diff --git a/helpers/__init__.py b/helpers/__init__.py new file mode 100644 index 0000000..a824eb8 --- /dev/null +++ b/helpers/__init__.py @@ -0,0 +1,6 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +from . import dialog_helper + +__all__ = ["dialog_helper"] diff --git a/helpers/__pycache__/__init__.cpython-312.pyc b/helpers/__pycache__/__init__.cpython-312.pyc new file mode 100644 index 0000000000000000000000000000000000000000..884e611498dac70b62305a0c9118ad186bace73e GIT binary patch literal 258 zcmX@j%ge<81Uo#pXUqW7k3k$5V1hC}ivbza8B!Qh7;_kM8KW2(8B&`Sl7&-6`tY#kd#*@P1#h`(9$Y&>~2a+RSHCbEE|w3DYcW(u9Y0Z1fW;DBWWpe zSDjr-CMy|1V-$8#z{n*|5CrM5g%iLz#XU7fV7EXIB`dWQHZEeIIW#xc3J}-LA#e6a zRAK`jU}xUE`QDrN9%sJcU%R`z5VY$jK3<4K5&AcqgpCh`t>1uf1?fn~WmI5eq>L+( zLZreKILzi;nXd>1p&}Ny}AgwrwM^#~Dkj)-%j&R^iK2GByZCMcpS`CI;slp$DG)cu$w2SO-neynax4N}m! zz?cmJd~d+FbsDtKBMke-krPx2p;AyA1`V5zJ5_X;z1LPFp>~a1-N83s+V!CCuwoB*7FLCl^dR2Z9qLNS zQ0k0}KSYR5KO!b>6_rvQn^1Zez4zT{_k}B+@!^(X{l1?AMrQ62vG5Pi@?p2+d-37Y zW1YA{>0x%du#D*7BjOOl958Yo554Kp4`QKy2G!6g`t5!A{sy0wMNTHf9YFN-nz&hV?ayBX+RIw~jEqN~wTwu1|$@La&T!KEidZD-VXg^_tx2i|&%XJ>L0t`0`8 zz&TwL&x^HST)`{o9QO)Z!5?xfSdsE|<%DTCa377jVb06jEFr6=E~~oEl>T)cst+() zG0ZB{sb)9l)WyC99faER)bi28N*?#w06&a!js2zQa@CogsmX#$+W@-)kH5 zreO{{KuPA*Ic-5T=Se*kXqO2!=4u}M)qiqoqUtQjOVp|?IkJa`uzUmAqVf)`3|{my z(TBeL3sesnPPDxp>**;<0ZOYT>=`CACR0W&^&ihzl-6>xZB>Y~z}8oY>BuY80udbe z$vF$#87n<*6bN@7pe}Eia~2iBkIO+V-Q5}`z}efBmtfon1tmql2y_oXDWfJ=n1Y0A zZp6@CX_kPu0OIndS|N-qq0EL;qcMh#Ly^dmV_2pu5SEP_cc`k7Syfwf<4e>qog$RQ z?Y4=j7eiYmMj5ftZq_k6al8CDiZ(nGy8}7o=O`t1wd~j~CX}6Po9!$I?S66iWt&iY zxT2bKa}=tqspVzMR!wbz)Q2Z6ty+P)*~7Z(sI%127VY85ZSlC}%n-V4Xk>W7dm;em zJrPBJI;iax--Sw$yX^g-b6q`X)D| zsoQ-|{O;NHzTCx=n^O96{!;$!XEvlK?qmc_iN$JJr~K zdL#Mswuq9MR^mW2aiEn@nhB+G_|<0O)o>Tj_3hp#e?PO)JJLvwv;~xW4Y&JH|MU1B z*!CX1*N&k*NB{R#3MKpQB0ea@iuBXnnq_h8&&i=n0XXfA74pjG7c={l|FdcP`#@z!2}$$bJC z-0Bx#`Bs(#Ij_XrSh2{?eX;0v6^j*1ua=oC6^p;9s^wrsV)xn5*gds$${v2o{>4#v zrV%{|-5<~v^oTMRrR)jzI(7x@5$GuI()B&PoB zp=d7jp>$LBkh+Ok%bBejWu4F=&7#ELn?>QKb}ada+xBB*^x)t5Jv$&a+?$H(-|k(r zs%>aMnR5nk%8C5+DL?ujF-tE7meKv#xL=rK^l6x~zex78(6u>?@m37s#OLVP=jeD7 o9sd`~!6CtWUyrs>PZRaLedzt%JGppjok0t%I@-T(jq literal 0 HcmV?d00001 diff --git a/helpers/dialog_helper.py b/helpers/dialog_helper.py new file mode 100644 index 0000000..7c211ab --- /dev/null +++ b/helpers/dialog_helper.py @@ -0,0 +1,43 @@ +# Copyright (c) Microsoft Corporation. All rights reserved. +# Licensed under the MIT License. + +import logging +from botbuilder.core import StatePropertyAccessor, TurnContext +from botbuilder.dialogs import Dialog, DialogSet, DialogTurnStatus +from botbuilder.schema import InvokeResponse + +# Set the logging level to INFO +logging.basicConfig(level=logging.INFO) + +class DialogHelper: + # Runs the provided dialog, handling both normal and OAuth invoke activities. + @staticmethod + async def run_dialog( + dialog: Dialog, turn_context: TurnContext, accessor: StatePropertyAccessor + ): + try: + logging.info("Initializing DialogSet and adding dialog.") + dialog_set = DialogSet(accessor) + dialog_set.add(dialog) + + dialog_context = await dialog_set.create_context(turn_context) + + logging.info("Continuing any existing dialog.") + results = await dialog_context.continue_dialog() + + if results.status == DialogTurnStatus.Empty: + logging.info(f"No active dialog. Starting new dialog: {dialog.id}") + await dialog_context.begin_dialog(dialog.id) + else: + logging.info(f"Continuing dialog. Status: {results.status.name}") + + # Special handling for OAuth invoke activities + if turn_context.activity.name in ["signin/tokenExchange", "signin/verifyState"]: + logging.info("OAuth prompt token exchange or verifyState handled.") + return InvokeResponse(status=200) + + except Exception as e: + logging.error(f"Error running dialog: {e}") + import traceback + traceback.print_exc() + await turn_context.send_activity("Sorry, something went wrong starting the dialog.") diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..58a57ca --- /dev/null +++ b/requirements.txt @@ -0,0 +1,4 @@ +botbuilder-integration-aiohttp>=4.17.0 +botbuilder-dialogs>=4.16.2 +databricks-sdk[openai]==0.58.0 +httpx From 9e43efd91cf1fb05f79d2d336b77cd39f4998501 Mon Sep 17 00:00:00 2001 From: manffred-calvosanchez_data Date: Mon, 8 Sep 2025 18:02:29 -0600 Subject: [PATCH 03/21] Update README to clarify that we are using auth on behalf of the user. --- README.md | 28 ++++------------------------ 1 file changed, 4 insertions(+), 24 deletions(-) diff --git a/README.md b/README.md index f103349..086d800 100644 --- a/README.md +++ b/README.md @@ -1,14 +1,14 @@ # Databricks AI Agent Bot for Microsoft Teams -A sophisticated Microsoft Teams bot that provides AI agent capabilities through Databricks serving endpoints, featuring secure OAuth authentication and intelligent conversation management. +A sophisticated Microsoft Teams bot that provides AI agent capabilities through Databricks serving endpoints, featuring secure OAuth authentication on behalf of the user and intelligent conversation management. ## Overview -This bot integrates Microsoft Teams with Databricks AI/ML serving endpoints to provide users with an intelligent conversational AI agent directly within Teams. The bot handles secure authentication using OAuth with token exchange, maintains conversation history, and supports advanced features like tool calling with visual feedback through adaptive cards. +This bot integrates Microsoft Teams with Databricks AI/ML serving endpoints to provide users with an intelligent conversational AI agent directly within Teams. The bot handles secure authentication using OAuth with token exchange on behalf of the user, maintains conversation history, and supports advanced features like tool calling with visual feedback through adaptive cards. ## Key Features -* **šŸ” Secure Authentication**: OAuth integration with Microsoft identity providers and token exchange to Databricks +* **šŸ” Secure Authentication**: OAuth integration with Microsoft identity providers and token exchange to Databricks on behalf of the user * **šŸ¤– AI Agent Integration**: Direct connection to Databricks AI/ML serving endpoints * **šŸ’¬ Conversation Management**: Maintains conversation history across interactions * **šŸ› ļø Tool Calling Support**: Advanced function calling capabilities with visual adaptive card feedback @@ -19,7 +19,7 @@ This bot integrates Microsoft Teams with Databricks AI/ML serving endpoints to p ### Authentication Flow 1. User authenticates with Microsoft OAuth via Teams -2. Bot exchanges Microsoft token for Databricks token using OIDC token exchange +2. Bot exchanges Microsoft token for Databricks token using OIDC token exchange on behalf of the user 3. Databricks token is used to authenticate with serving endpoints ### Component Overview @@ -213,26 +213,6 @@ DatabricksBotService/ └── dialog_helper.py # Dialog execution helpers ``` -## Deployment - -### Azure App Service Deployment -1. **Create App Service**: Create a new App Service in Azure Portal -2. **Configure Environment**: Set all required environment variables in App Service Configuration -3. **Deploy Code**: Use GitHub Actions, Azure DevOps, or direct deployment -4. **Update Bot Registration**: Update the messaging endpoint to your App Service URL -5. **SSL Certificate**: Ensure HTTPS is enabled (required for Teams) - -### Docker Deployment -```dockerfile -FROM python:3.11-slim -WORKDIR /app -COPY requirements.txt . -RUN pip install -r requirements.txt -COPY . . -EXPOSE 8000 -CMD ["python", "app.py"] -``` - ## Troubleshooting ### Common Issues From c7e8c820abd3291d7d5a562083ee62bdea244e97 Mon Sep 17 00:00:00 2001 From: manffred-calvosanchez_data Date: Mon, 8 Sep 2025 19:42:04 -0600 Subject: [PATCH 04/21] Update README to clarify some steps and set expectations with the users. --- README.md | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/README.md b/README.md index 086d800..0bdfc4e 100644 --- a/README.md +++ b/README.md @@ -55,18 +55,19 @@ SERVING_ENDPOINT_NAME= ## Setup and Installation +> **Note**: The following steps provide high-level guidance for setting up the Databricks AI Agent Bot. These instructions are not exhaustive and assume familiarity with Azure services, Databricks, and Microsoft Teams development. Additional configuration and troubleshooting may be required based on your specific environment and requirements. + ### 1. Databricks Configuration #### Create a Serving Endpoint 1. In your Databricks workspace, navigate to **Serving** -2. Create a new serving endpoint or use an existing one that supports chat completions +2. Create a new serving endpoint or use an existing one that supports the OpenAI Responses API 3. Note the endpoint name for the `SERVING_ENDPOINT_NAME` environment variable -4. Ensure the endpoint supports the OpenAI chat completions format +4. Ensure the endpoint supports the OpenAI Responses API format -#### Configure OAuth Token Exchange -1. Set up OIDC token exchange in your Databricks workspace -2. Configure the workspace to accept Microsoft Entra ID tokens -3. Ensure the `/oidc/v1/token` endpoint is accessible +#### Configure OAuth Token Federation Policy +1. Follow this guide to configure federation policy: [Configure a federation policy - Azure Databricks | Microsoft Learn](https://learn.microsoft.com/en-us/azure/databricks/dev-tools/cli/authentication#configure-a-federation-policy) +2. Test token exchange using this guide: [Authenticate with an identity provider token - Azure Databricks | Microsoft Learn](https://learn.microsoft.com/en-us/azure/databricks/dev-tools/auth/oauth-federation-exchange#exchange-a-federated-jwt-for-a-databricks-oauth-token) ### 2. Azure Bot Service Configuration @@ -81,7 +82,8 @@ SERVING_ENDPOINT_NAME= 2. Create a new connection with: - **Name**: Use for `ConnectionName` environment variable - **Service Provider**: Microsoft Entra ID v2 - - **Scopes**: `openid profile User.Read` (or as required by your Databricks setup) + - **Scopes**: `access_as_user` (exposed API scope for V2 access token) +3. For scope creation reference: [Configure an application to expose a web API - Microsoft Learn](https://learn.microsoft.com/en-us/entra/identity-platform/quickstart-configure-app-expose-web-apis#add-a-scope) ### 3. Teams App Configuration From eac9cda7e7108beecafb6d1b4c753e9598aa38fe Mon Sep 17 00:00:00 2001 From: manffred-calvosanchez_data Date: Fri, 10 Oct 2025 00:07:22 -0600 Subject: [PATCH 05/21] Updating databricks client to manage different type of agents in Databricks (chat completions and responses). --- app.py | 2 +- appManifest/manifest.json | 7 +- bots/__init__.py | 2 +- bots/auth_bot.py | 4 +- bots/dialog_bot.py | 2 +- client/databricks_client.py | 182 ++++++++++++++++++++++++++++++------ config.py | 3 +- dialogs/__init__.py | 2 +- dialogs/logout_dialog.py | 2 +- dialogs/main_dialog.py | 41 +++++--- helpers/__init__.py | 2 +- helpers/dialog_helper.py | 2 +- requirements.txt | 1 + 13 files changed, 197 insertions(+), 55 deletions(-) diff --git a/app.py b/app.py index 2f3f9c5..27ef9f0 100644 --- a/app.py +++ b/app.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. import sys diff --git a/appManifest/manifest.json b/appManifest/manifest.json index 3fb74da..88d1b02 100644 --- a/appManifest/manifest.json +++ b/appManifest/manifest.json @@ -35,8 +35,13 @@ "supportsFiles": false } ], + "permissions": [ + "identity", + "messageTeamMembers" + ], "validDomains": [ "mychatbotteams.azurewebsites.net", - "token.botframework.com" + "token.botframework.com", + "login.microsoftonline.com" ] } \ No newline at end of file diff --git a/bots/__init__.py b/bots/__init__.py index d6506ff..2c055b0 100644 --- a/bots/__init__.py +++ b/bots/__init__.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from .dialog_bot import DialogBot diff --git a/bots/auth_bot.py b/bots/auth_bot.py index ab7a5b5..cd232b8 100644 --- a/bots/auth_bot.py +++ b/bots/auth_bot.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from typing import List @@ -31,7 +31,7 @@ async def on_members_added_activity( # To learn more about Adaptive Cards, see https://aka.ms/msbot-adaptivecards for more details. if member.id != turn_context.activity.recipient.id: await turn_context.send_activity( - "Welcome to Teams Authentication Bot Example. Type 'logout' to sign out." + "Welcome to Databricks Bot Example. Type 'logout' to sign out." ) async def on_token_response_event(self, turn_context: TurnContext): diff --git a/bots/dialog_bot.py b/bots/dialog_bot.py index 60afb2f..628dff9 100644 --- a/bots/dialog_bot.py +++ b/bots/dialog_bot.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from botbuilder.core import ConversationState, UserState, TurnContext diff --git a/client/databricks_client.py b/client/databricks_client.py index 30465af..8bf55d2 100644 --- a/client/databricks_client.py +++ b/client/databricks_client.py @@ -1,12 +1,16 @@ +# Copyright Ā© Databricks, Inc. All rights reserved. +# Licensed under the MIT License. + import logging +import uuid import httpx from databricks.sdk import WorkspaceClient +from databricks_ai_bridge.genie import Genie class DatabricksClient: - def __init__(self, databricks_host: str, serving_endpoint_name:str, request_timeout: float = 300): + def __init__(self, databricks_host: str, request_timeout: float = 300): self.databricks_host = databricks_host - self.serving_endpoint_name = serving_endpoint_name self.client = httpx.AsyncClient( timeout=httpx.Timeout(request_timeout), ) @@ -26,52 +30,170 @@ async def exchange_token(self, provider_oauth_token: str): return response.json()['access_token'] - def create_databricks_payload(self, text:str, history: list, stream: bool = False): - - messages = list(history) + def _throw_unexpected_endpoint_format(self): + raise Exception("This app can only run against ChatModel, ChatAgent, or ResponsesAgent endpoints") + + def _convert_to_responses_format(self, messages): + """Convert chat messages to ResponsesAgent API format.""" + input_messages = [] + for msg in messages: + if msg["role"] == "user": + input_messages.append({"role": "user", "content": msg["content"]}) + elif msg["role"] == "assistant": + # Handle assistant messages with tool calls + if msg.get("tool_calls"): + # Add function calls + for tool_call in msg["tool_calls"]: + input_messages.append({ + "type": "function_call", + "id": tool_call["id"], + "call_id": tool_call["id"], + "name": tool_call["function"]["name"], + "arguments": tool_call["function"]["arguments"] + }) + # Add assistant message if it has content + if msg.get("content"): + input_messages.append({ + "type": "message", + "id": msg.get("id", str(uuid.uuid4())), + "content": [{"type": "output_text", "text": msg["content"]}], + "role": "assistant" + }) + else: + # Regular assistant message + input_messages.append({ + "type": "message", + "id": msg.get("id", str(uuid.uuid4())), + "content": [{"type": "output_text", "text": msg["content"]}], + "role": "assistant" + }) + elif msg["role"] == "tool": + input_messages.append({ + "type": "function_call_output", + "call_id": msg.get("tool_call_id"), + "output": msg["content"] + }) + return input_messages + + def _parse_responses_output(self, response): + result_messages = [] - messages.append({"role": "user", "content": text}) + logging.info(f"Response from openai client: {response}") - payload = {"input": messages} + for item in response.output: + logging.info(f"Item type: {item.type}") + if item.type == "message": + content = "".join([e.text for e in item.content if e.type == "output_text"]) + if content: + result_messages.append({"role": "assistant", "content": content}) + elif item.type == "function_call": + tool_calls = [{"id": item.call_id, + "type": "function", + "function": {"name": item.name, + "arguments": item.arguments}}] - if stream: - payload["stream"] = stream + result_messages.append({"role": "assistant", "content": "", "tool_calls": tool_calls}) + elif item.type == "function_call_output": + result_messages.append({"role": "tool", "content": item.output, "tool_call_id": item.call_id}) - return payload + logging.info(f"Response parsed from openai client: {result_messages}") + return result_messages - def parse_model_output(self, response): - parsed_output = [] + def _parse_chat_response(self, response): logging.info(f"Response from openai client: {response}") - logging.info(f"Response output from openai client: {response.output}") - for item in response.output: - logging.info(f"Item type: {item.type}") - if item.type == "message": - parsed_output.append({"type": "message", "text": "".join([e.text for e in item.content])}) - if item.type == "function_call": - parsed_output.append({"type": "tool_call", "arguments": item.arguments, "name": item.name, "call_id": item.call_id}) - if item.type == "function_call_output": - parsed_output.append( - {"type": "tool_result", "call_id": item.call_id, "output": item.output}) + result_messages = [] + if hasattr(response, "messages") and response.messages: + result_messages.extend(response.messages) + elif hasattr(response, "choices") and response.choices: + choice_message = response.choices[0].message + message_content = choice_message.content + if isinstance(choice_message.content, list): + message_content = "".join( + [part.get("text", "") for part in choice_message.content if part.get("type") == "text"]) + message = {"role": "assistant", "content": message_content} + if choice_message.tool_calls: + message["tool_calls"] = choice_message.tool_calls + result_messages.append(message) + return result_messages + + def _get_endpoint_task_type(self, workspace_client, endpoint_name: str) -> str: + """Get the task type of a serving endpoint.""" + try: + ep = workspace_client.serving_endpoints.get(endpoint_name) + return ep.task if ep.task else "chat/completions" + except Exception: + return "chat/completions" + + def _query_responses_endpoint(self, + workspace_client, + messages: list, + serving_endpoint_name: str) -> list: + + input_messages = self._convert_to_responses_format(messages) + + openai_client = workspace_client.serving_endpoints.get_open_ai_client() + + response = openai_client.responses.create(model=serving_endpoint_name, input=input_messages) + + result_messages = self._parse_responses_output(response) + + if not result_messages: + self._throw_unexpected_endpoint_format() + + return result_messages - logging.info(f"Response parsed from openai client: {parsed_output}") - return parsed_output + def _query_chat_endpoint(self, workspace_client, messages: list, serving_endpoint_name: str) -> list: + """Calls a model serving endpoint with chat/completions format.""" - async def call_model_endpoint(self, text:str, provider_oauth_token: str, history:list, stream: bool = False): + openai_client = workspace_client.serving_endpoints.get_open_ai_client() + + res = openai_client.chat.completions.create(model=serving_endpoint_name, messages=messages) + + result_messages = self._parse_chat_response(res) + + if not result_messages: + self._throw_unexpected_endpoint_format() + + return result_messages + + async def call_model_endpoint(self, + serving_endpoint_name: str, + text:str, + provider_oauth_token: str, + history:list): oauth_db_token = await self.exchange_token(provider_oauth_token) workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) - openai_client = workspace_client.serving_endpoints.get_open_ai_client() + task_type = self._get_endpoint_task_type(workspace_client, serving_endpoint_name) + + logging.info(f"Serving endpoint task type: {task_type}") logging.info(f"Actual history in the chatbot: {history}") - payload = self.create_databricks_payload(text, history, stream=stream) + messages = history + [{"role": "user", "content": text}] + + if task_type == "agent/v1/responses": + result_messages = self._query_responses_endpoint(workspace_client, messages, serving_endpoint_name) + else: + result_messages = self._query_chat_endpoint(workspace_client, messages, serving_endpoint_name) + + return result_messages + + async def call_genie_space(self, question: str, + provider_oauth_token:str, + conversation_id: str, + genie_space_id: str): + + oauth_db_token = await self.exchange_token(provider_oauth_token) + + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) - response = openai_client.responses.create(model=self.serving_endpoint_name, **payload) + genie = Genie(genie_space_id, workspace_client) - parsed_output = self.parse_model_output(response) + genie_result = genie.ask_question(question, conversation_id) - return parsed_output \ No newline at end of file + return genie_result.result diff --git a/config.py b/config.py index 753007a..6fe7d43 100644 --- a/config.py +++ b/config.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. import os @@ -18,3 +18,4 @@ class DefaultConfig: CONNECTION_NAME = os.environ.get("ConnectionName", "") DATABRICKS_HOST = os.environ.get("DATABRICKS_HOST", "") SERVING_ENDPOINT_NAME = os.environ.get("SERVING_ENDPOINT_NAME", "") + GENIE_SPACE_ID = os.environ.get("GENIE_SPACE_ID", "") diff --git a/dialogs/__init__.py b/dialogs/__init__.py index ab5189c..a2bcd77 100644 --- a/dialogs/__init__.py +++ b/dialogs/__init__.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from .logout_dialog import LogoutDialog diff --git a/dialogs/logout_dialog.py b/dialogs/logout_dialog.py index 7fb418e..c695a80 100644 --- a/dialogs/logout_dialog.py +++ b/dialogs/logout_dialog.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from botbuilder.dialogs import DialogTurnResult, ComponentDialog, DialogContext diff --git a/dialogs/main_dialog.py b/dialogs/main_dialog.py index 22b6631..b12b4af 100644 --- a/dialogs/main_dialog.py +++ b/dialogs/main_dialog.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from botbuilder.core import UserState, ConversationState, CardFactory, MessageFactory @@ -55,12 +55,13 @@ def __init__(self, connection_name: str, OAuthPrompt.__name__, OAuthPromptSettings( connection_name=connection_name, - text="Please Sign In Before Interacting with the Bot.", + text="Sign into Databricks to chat with agents.", title="Sign In", timeout=300000 ) ) - self.databricks_client = DatabricksClient(databricks_host, serving_endpoint_name) + self.databricks_client = DatabricksClient(databricks_host) + self.serving_endpoint_name = serving_endpoint_name self.add_dialog(self.oauth_prompt) @@ -99,19 +100,25 @@ async def ensure_signin_step(self, step_context: WaterfallStepContext): async def send_response_activities(self, input_text, response, new_history, dc_context): new_history.append({"role": "user", "content": input_text}) + new_history.extend(response) tool_calls = dict() for item in response: - if item["type"] == "message": - await dc_context.send_activity(item["text"]) - new_history.append({"role": "assistant", "content": item["text"]}) - elif item["type"] == "tool_call": - tool_calls[item["call_id"]] = item - elif item["type"] == "tool_result": + if item["role"] == "assistant" and "tool_calls" not in item: + await dc_context.send_activity(item["content"]) + elif item["role"] == "assistant" and "tool_calls" in item: + if item["content"]: + await dc_context.send_activity(item["content"]) + for tool_call in item["tool_calls"]: + tool_calls[tool_call["id"]] = tool_call + elif item["role"] == "tool": assert item[ - "call_id"] in tool_calls, f"Every tool call must have a tool result. Call id: {item['call_id']}" - tool_calls[item["call_id"]].update(item) + "tool_call_id"] in tool_calls, f"Every tool call must have a tool result. Call id: {item['tool_call_id']}" + tool_call = tool_calls[item["tool_call_id"]] + tool_info = {"name": tool_call["function"]["name"], + "arguments": tool_call["function"]["arguments"], + "output": item["content"]} activity = MessageFactory.attachment( - CardFactory.adaptive_card(self.create_tool_call_card(tool_calls[item["call_id"]]))) + CardFactory.adaptive_card(self.create_tool_call_card(tool_info))) await dc_context.send_activity(activity) return new_history @@ -130,8 +137,14 @@ async def api_call_step(self, step_context: WaterfallStepContext): # Call Databricks agent API. input_text = step_context.context.activity.text.lower() actual_history = await self.history.get(step_context.context, default_value_or_factory=list) - response = await self.databricks_client.call_model_endpoint(input_text, str(token_response.token), actual_history) - new_history = await self.send_response_activities(input_text, response, actual_history, step_context.context) + response = await self.databricks_client.call_model_endpoint(self.serving_endpoint_name, + input_text, + str(token_response.token), + actual_history) + new_history = await self.send_response_activities(input_text, + response, + actual_history, + step_context.context) await self.history.set(step_context.context, new_history) return await step_context.end_dialog() except Exception as e: diff --git a/helpers/__init__.py b/helpers/__init__.py index a824eb8..bb931cb 100644 --- a/helpers/__init__.py +++ b/helpers/__init__.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. from . import dialog_helper diff --git a/helpers/dialog_helper.py b/helpers/dialog_helper.py index 7c211ab..32da720 100644 --- a/helpers/dialog_helper.py +++ b/helpers/dialog_helper.py @@ -1,4 +1,4 @@ -# Copyright (c) Microsoft Corporation. All rights reserved. +# Copyright Ā© Databricks, Inc. All rights reserved. # Licensed under the MIT License. import logging diff --git a/requirements.txt b/requirements.txt index 58a57ca..b13c670 100644 --- a/requirements.txt +++ b/requirements.txt @@ -2,3 +2,4 @@ botbuilder-integration-aiohttp>=4.17.0 botbuilder-dialogs>=4.16.2 databricks-sdk[openai]==0.58.0 httpx +databricks-ai-bridge \ No newline at end of file From fc56397716ed383c0d6882f2fb21b2cf3ba05286 Mon Sep 17 00:00:00 2001 From: manffred-calvosanchez_data Date: Thu, 16 Oct 2025 18:56:46 -0600 Subject: [PATCH 06/21] Updating appManifest for teams app and adjusting app to work correctly with teams authentication procedure. --- app.py | 7 +++++++ appManifest/color.png | Bin 3415 -> 10107 bytes appManifest/manifest.json | 19 +++++++++---------- appManifest/outline.png | Bin 407 -> 1298 bytes 4 files changed, 16 insertions(+), 10 deletions(-) diff --git a/app.py b/app.py index 27ef9f0..5ea81fc 100644 --- a/app.py +++ b/app.py @@ -94,6 +94,13 @@ async def messages(req: Request) -> Response: try: # Process activity with the bot adapter and bot logic response = await ADAPTER.process(req, BOT) + + if response and response.body == b'null': + response.body = None + + logging.info(f"Response output: {response}") + logging.info(f"Response body output: {response.body}") + return response except Exception as e: diff --git a/appManifest/color.png b/appManifest/color.png index b8cf81afbe2f5bafd8563920edfadb78b7b71be6..0a56553f4e4f82a260f627a2476eefdfdc5afcf8 100644 GIT binary patch literal 10107 zcmXAvcQjnz*T(ORmLLcQ(LOuL*(wV0i;Fz2=?D~ae&)LAq!Y3}li$_2fwZ6bnAeP3X6eXm!;y{&R)I$S9q zL3BAYIigEbHO@@uUM23+y($~ik~*;#iljSEd|fcYEYBEW7;+2rA7fE2G1F~h4;_B% zH?qr1Y^W=a25VQnoY+?&54%`X*$KAGoiRRj4KI~p+M%083AQ{rJ9;~Dmeq?1mnm+! zy?gb)ny!7wFjo&+CPx>Qv!xASdk3`7B=VV}KQ zJpQ_z1AmH2`|co-w;~gHeP>r^7v|=!rAO9xNs6vrn>WR9{`gh7jeKF`1^{F<{|*q4 zn$Cc~NaC)du0*m0RZfPilRK+Yic{gGssl0rCYYGDUJAjtmtia9+r0* zO@wg%$rK=@x{!T?hU~oZRYzCB2eB7a)LQmyJnXT`AGva?XKtJEHqglV2>pcFfpky< z7fb1u2CP4w{?4R`HZ`njPF$|cxdqI-1s-3}{91mUc(_oD<+pId@jyCHJ~ipwJY1*y zZMOdbv=n3xW}3K}0hvbH#PfY0ENn^uXU~`E2u|~Jd>{}FqO4dLoFiPZ!=!~TF0Ys8 z18W!J0Zs|*EWRLafNgg{)|VLbX*k&?*GjoDf)_Y1X{h2vGVkTT+&N~EYeFwVf6c-` zLm!;3kP__=aJE~{|h0osdNQcq_thBOFYHQNH zR~FfY-9Zih(Uk&Fr7-TE$PAi zq^0v|f$)mgjsm%Fab(!tpbVBy0ph9Q^NJF!-^+fdw_%wULdp$7U|p)SkH(dD9zOQG zK>XK3Wh0{5ZkNF%u&CI19}I3fc^t_9W?CsVq~|`J#WYmrpe<;=gr`+Nx`h@HL|qrK zj|A_62$&VifB_&{Tieul(|P5q0@azM@ZE`t;qQ&4M_v#Jo0_I5k=)?+rRgTs-77MS zXJAz?7GZs*ea_9((Mj@;nnNTw*rKI=zbcshgVogC zIuxyE)?;p`L+jemF^B2Feg0%%MC?5Me0*5ubRdZq(T>H{pA*zM~gf~u0HrstglaT z{}(ObZzs?js%|VnPBVK}lvfWGdEM=KB(le?S6~LVvBM;XFDCK=j5oM$z%6K&MM;_f z_UsRJjfHl;^Ny0f+~1n$uZX;X4y++x;^t_3rxcM~9t%ovhHgOu_ec)MzM*bGVVqS& z+F5u0bB05Y*qoe9paA(a;+REm5m# zjR>qHl`!czg3-36g0a1;Ew2H>4+JAdqQh{_aWGBqUHDD*eh|m9Yn|+o#QamH!NFkq~ObiW_tmO{k-_-#dnn1opef=p7tg6`bJ|n zQ(DFSHsn#hKEZ0vFnJl$?{`;>w!ueBksep)!LHn0FVwdiEK&3xzbRvqI>BG!me znv({S{K-BYJNc(QimD6Lmg`oe?0of>Or#-+UKaqCO;J=FvJX=P3z^C?%weOobjffO zQ|u%*5PDAH4781`&D-WKPH#R0dK``J#|l}sr$Qys5m6Zx&hStx`_=JlL`KJvo?nml zZ(ZyuX=1OkV#K4(J3!AF(m*P_ApP>S1@VtbKI7{D^!laFNzN)&a3&boo-%VMBxwHh z$9J_1aYs>+e6sn@47&|?@?p57o^r+YH?4Ph!qW6?u2dd{=*8hHYGBz45*AqG19 z`!QAm(iu~*F|K73hZr}zA;A7JMNoxZ|7W8yhKgQf#l9PM(iknv3Ene;U0b-D19Dm& zQ4`Ws{$zV?(LFcRdz@v*g*wa5m-)G;8r7@>el#QS-$Re~h2(hI0u*r!IN=$_p?T1C z_Hg%Woq-FP^T$7&ZXGbe%>NG6T&|P--3)k>MIeQ_OgL zRgU8)T}38e=BS}B{>XM;X`L_RB>rZ zvPIVl9T|16da+_YDG3hn8zyMTYz?y8^%93UTTBy71E(#l zavm9-04h~tzF;d{I8!$OL3vRl$glO)b=CI49~9JoO0nvE=;Q~aL9Q|m1)|$6p-WYB z!71h*20jk!u4Za(O)uCJg)CIpJlm@^)LC4K%Miny1HYtm(|P0 zy+2@rFd6tT{i!=yP}yyD$=Hny>HVw6Tn!Mj?6F#M zl}Xnd_|Wky4>m5vcfGRg=Ou^~-}wmjB?DSWk+%(;g6^9bn+|-KH@9yuURNET+Pb`h zk4ujC(O15Y#YX1cs2hE=<#45A*HdHrCp#ov_GS%|x`ye0cM3{J1_rmO?TY>G=vu}m ztY6kffm>z~G(0q&5J=}?4OjJCo!XlNC#*Whx6x(4dR?Eb^{*=fg*9*|5OTyyLzwQJfZS=JAc<_0IPATbbN2^R z(mq0I%`#h2`*9Pf?D+ZpyH_SSo*BmVD-e*KPDZp~L0&f)9Zh*5ELZF+9dU4dt zB%AE`>`xkI1tY-H*`jh1Ww0ZIol4^SZmhIFYWLhotVt_ji`d8Ix9Nm{^<0vH!jmv=4ojq?D7;Y0;j$hoK+s?dvvBKc2_JwtrM{!)$b)>bSzH&ZYdF8qNRzZf59o{5qxY2g%n z*9d{pdEt?VyJ>AA4$Ib@Du=k_P;{Bec}pOi7fA;VVZm=)V;v}Pd1L+(`lzDc*_XIqs>h2(m?dTE?7i@JN zBMhN|nj(n=tBX%Gao#XYi@K5vyw^5KYL5I~FfaCQeg7FgNt8r+9Sa>8I_HB2zNiXH zy|&Y-mU#(5JEIQM!ZC7-mQ|=+w_kY|HYxAV#dRJZ<(@LVDj^7np0;Qq6*lj92cwmp z?U;8mWy@x|!NsAIgg_UTQ)50&Prj3n8vctjlhnb2P?U%6WfBOFPp@D*1+&^`yTn~Z zJlD@P6WcUT0fY%BX+$-8vF+{LY^tK*l2{!ua3{8@_Y7HafSG-uH$3$@KhHDHRp#<3 z;R*N1Pnm>!)oWe3*{N(ed;sm?IisXUUFxw`lvh_>86- zr$kgwiz(tb!6{{Vg!uk-os^ba0kYg_YX~tE=Udf7PAo!srl{<&%UhxiZlC^vL3)Mf zgIj&A>x3smVvQ_ivJu5+NXrC<+v%hS75K4NjGx@ny~$2Her}?V4;;C>XY@#P-hnI` zGuJa2mE~P=8LY9q7Ek$;$83#6y3+MC@}_k4wR!F4_<2Z%_m6v(zAq(c0M!rdq(o%M zsw?0ru(N~y`yDl}8C$3Cu7Wd+mUMfMXHvQKL8~BLJ=(VT66u7)Np-mrg&wo_^o#=r zuU2IZUR92blklu|#DH~!@5pKDXz%mNet^_kGP;=ZP9|HDZfpEh+=p)FEt))=%(Apc zCFIqIjt#VxYWQfe0SN~ST~_fd%D*OC4xez5`8LP!JMd;XQvcD4zt^)*C@gQlXJAcN z&X%5EQB$M(aEQe6zXLWm+7ohN(2*YmvT3vNn+V7^-DJOe#jrR0`0^(ihH6rY4{b1h z!H&l?g?ILdSuJIkxR!4_>#L~ia*WRLpkyl68O)t6sSR0NW}V9u1SYWYODKb1K5RpH znt7sm{Z09<&7usZf}ozEpH?p{YugrmV@aN~wQzglj6U!9fgX>{?z&l*1qZxIj1<5= zkSZtax?prN82HdMJ?$8oS5z0&fMtesIv+biWtlmClJk=AOjP`t%^dJFfsL*ktQ!$l zl>+EFGJXywIIPca3RFQjWHGtYn61@sIC!$;M4xjD-Mq|DuuLSN`xftxu#9 zRB1>i2OAMTj~-#&qh!RFiigS*dWku9mlbMJv?mXQL1jV)p;nz8P2@Cj`&c0tMCI^b zHR&AgWXBeemRW^WSk!>F_Ev{11Nmiw^?%tS!nM3Z;uK9S0bDO7*#eRViZ=x}J#e1c zH`%#Xv)h;X4Q8nX#mr9qa0Aw&i{QLiP+bYp$;leOA1J$#4zz9DtDe%H+jv2e?IXV-vwQJ{DPB=hpQ^XVSc)^z0-tlrC_oRjfrw@Z~ zW@Y~MB|)sRE!NNPT73Q1U0mdO|Kt-;NyrcvbkH}lb}yyK(6jMKr(8)fb(~<6V-$)HbRqo@!XvHT`Gr_pbHUFDmtu?_>w1(^%ZLSuxk`Bg0`73N3%__kdkaanhJrwDmfy^ysxMHn zoM8ynAblFZeexY0tnW%rdD)04J<6t$@4=>Nc}H--Tx-PUR)}~ojaYiZmrzu`U#(H> z%@?lrmFNRDQB-TID&(p4oZIBW4`0-^1&KD^H9)M&nw74W@dAxJg8};9rWY=~*e{I! z#$0^mMQwRAQgVuc^t$?6_PsI@KNXrCPP#@F4T<}nZ|dsh7zRRRvsi!we^`L6uCj}# z5wvaPoskD9C=NnFb-(zI4=qFC@VNEUNZqeyt~A1*$IA=Q8rW(n14O}V(*0f!gUjKZ zq3<$d18j*u^0SX0mS|<|d~F0}_l~pkKGO?I+g9HX;;yoTB34|=#bf-`cSwR5c?J=fp#@$UtJ!>!Ntg+2Lf&q|w6_^I3b3;uT+xfosN~1wKwnp_nEak2toD#IdiRJ>SYycGP7GS6G5I|Q z-1e0tCU!y(+mKyYjE`Z#yo0sRLk}*Dp>3nuGrCyC+aYHi&VfImWXcLac7&TGJigSk zp?|77U}Ne|oI#CUf9Lno6?xW!BUUIUGv#G+UWRj1?~#g3iN*yX(CBckukYO&?8|lZ ztU5E7R7o>KeZdZ~SoSDxoD*w^lY(@zE}mHut2lfb&Eg;FA`_die*w5_XMVDuYnhX& zmA_)b{I{|b*4%@IMagvNmcPmlE1x?w&CgY`#f*u%%h{ z=uni1a*Oy3seqXC*V_U6uY{3jK*B-W_k8|?Xd^u`d z+e1s&VuApkQdH+X8ZQco>Hk4N7x&4KbqDO9fJT#Dd^|`#w4|fao~EqxYJGF^PFGah z^n_>htBBFCacfwy{8Vw8Yz0(ApVN=xqlGu#fzw_xK9V|vao(MC<~Xt`0}se=uHy}b zPMR=fBWU0}>otu~AF?!9Hbll)sgqvX5kBVh8x&nPJB zd`pP0ovJnhz7C&l*|3iYE6|LGPR%}gyvT&zr3$E^^J3TUw^fS0`rNX~I357P@Ee

r)x`c$%rXRn4=AwFnIEQs^xyYy!ydyY1yTG2>~dTcZKfH znvgF!M!5oQ!?(u`#w>>#q#vH~!$XTbYI+)UpG<_W!H76x9Q_p0JxcSQfwZvIUZCi^DiSelwD zxi8^`Zv<;_I8)Qa$5D~gkMKXXgxb7pJM3H`NMWZGr06X||8z9^rD3ik&t4%$D_iZT zR|}e2ktT#{6%bzDo9`0%mhu#6l;GsVNjK6YlYXlUq7mT?n>(sr$f{a-#?hR>=r7Mz z5)-IpkAU1hqTf!QObp{E?y^6Mk~uFGsJW;RPOM;h$S-5`FXia^1U8I0ub&F1pj%%E z@T0sYE^T6WC(^nTqrVH&bd=Tbo7aBaSp_a6cCM%`8pqDJU^&-EJNdO^a$tX+f@i2S z#w(a3V6m+sKap&`rqL{wobOj3H&l8L&StH8NpvKE-xf0VM4kA$;%-{=GTOir3LPRT zJc4HvyZFhBByur{3M@eZYt+1q?74wK#cZBA&|;00bR6Q}v^WVAY@KT<-{8z}aEm1L z;_GON9gBAxVkZmHxMsSJ_Ny>9Tj{2U9976dlsUyy*opt28WK5rH!F$vY9o((4k(a( zsYWF@Tf#U6`tgf#Jj_(fC_}~^wX&mD*&Ni>$E6BMf>CC-uj7nPV<}nlbFBJ0<^eX> z<_AIq(s%!`+e`B_CHGVj{8Ti02`aEZ^1er@%ca&JOtr9y5o*_uUc|Er%MW$Vyd4@< zvx0`*3;-y!FB|No-DvxO-^RJ3{Ce>pE^87`%l0+T32CXX_d!F;R3{uM>ui)pH#jcY ztZh69o4;zTH_fGGtRF~ar>w9Xe9|!_7%)COpxH<0CUMg&eWNLf_94c+D>t1=gfuRI zEM)!r!78z>_BdklPA~Hq-3nj@B%0{)tSn0-eueGYxzI*8x-6r2%`8#kzC7F7n~TG51<)FV?wwo>42-O{fF z0pn^I^q_1B&r7qwPW-F*j+mPkJ{7E7*^!nW%5NnJ5~R$ksw+agI_JbnTx6^y# z>b9H%1etTMX+(}U=DW+*KXKzJ85C)diQaF71xik6e|RZwu6;!UDVv79m7xq0j6V3_ z5P4cx-x+pCG#AWv@>MaN(dED?g3;txClx^80$j`r)(}*@VTj% zMVvzLhx*H-#|A~uwsN%FJhH>Czj$c*s)^VUe}CiO%=f$PVbxQvM|Q>0b*`^N185n9 zz6G_Bw_OEnmvjFFvKKH5aCo!H5Z;VZ%5TOZ(byA|weRaz;7b;T!jPwvFjmWgc~ikE+<{EFX9&WYo2z%X2JADYqQp zSvF2p3URg~tsw?0=RFF9hjx0zyh!M>CK!%~)BR7{?O8n1Zbk<>VaB=pauJ)@c?6JE*e+_Z)n!<~MjU#y+ zkGvBrt9Ocav*6pY%>nWgQejBHcL`O+L8TQ+m%P6C>m0Tm3@b_VbZ8f2>?g)+KTkHU zcF(3P&z}?|#Zza3dWF3Lz}z|n*y`G`l$C9QwvJQhJ#|gu<1cACm=MXnnDTTH*fJ)M zRM_0q%PC@#JNw9y>I1@j9;Ua+&5dz50u)<6{N#1m;MvK-l#JB8U(f%*G(!$w7pO)0 z33n(JA4jExPzv@Sfmp)0WIPLqPmbWikZQ1qH`I!{+#v4LA9y@50!>bbc;Vd(`m)LC zJi1{I{?J<=I3kGxw(aQjY?o?Fc?8e3GIeB_1s^<1ZY7nV7sWLS=DppLyretf%B3J5 z{`~*~$@)vUm`QgVh1eaLZyWa};&rWRh@5)7oqg_@X%g@6wvFlpN|G=U`N9R6FWn7E zk3g<7ioAqupsw6UBMo38(It`pyCiAJ}TU48XSBn&GEqS2!j zs<@eZPkk$^ZV=bQmB~;R;QFV&Ve5Sud}&aDrE~VrNCW?JRR^I8`p^0^i=L?$nPUg! zOTnD}(h4Ew6&as1Zj_WWOrle!yr1dlLLYiB-b&Wt?yQKS5J7->Nok zoc30ZO~X}DSGge|lsWyFyxkhdU9V1C0wi1Up|VsC?jcm7o0_g>8YrM9HvOgljH=Ds zx4bU|@lii$F~K6}9XLMzt5*Lu*z6%K>S3x67$c`-oLXnmCe7%)r?+Y;%{bdixhGVWP8@>;3Y(n5NI+1}`M2mlLVG zT6?b_h^C$0v)~9eEpRf2i`trPRP!o)7J#yzNFAAMMkBQMbM{I1a+YV4{^$vuLK5A*{|%ag>$%Yax6WL_qQyMMKN0`Dq!rNggZr#CCBjmo;Nxn z4HA5E%zjtJYD&_J;v(^2f%q9m%_&=TiK6a^bW&8cVW%us+j9;GC>(AZRP3zt-!NZ_ zU0SK)6p6H!Z7=iASEhMxe<^4Z)Q8RZQEIA@~E8a4~Bub@6U0?gsigV43 zAb#n+k#Fx-mYmpv7w!RDaUgz~uj-Yu%?#>A@LpEVysIE|=xZlS6q5-v*75J=<>2#Q zb?-kt_p5PC5F=r^Fd;%kam(I;{WcyPq(!qD;*7!SmF1D-YC-oH{q?y@0#}WA$2mB@ z1^?x6eMs-cEPA!9x*V*Os|o%mQMF?2m$MO{Ebnujxfdo^$O}Rl;&!@t%K(k;>X@XK zr?xWWWs6M|XI^(OQ7g21dmj@v4}mp!rq*O?eAXf>qI@jj>1Ovjr8vY=>gH!pK`=s? z5FF#p{K@jyiWlmyd6E5A zPT=#6v(ThyLIFjq#TCDS{+e>I3&gC!<{b%Dg(pz?g+KcU`FA_@P6y~MdK>kbh-pMdj$2-Qh5R=26X?KbO-#po^sG7q-n|^5n=#4Yx(@gGlrT7&wU=}1y{L#<# z;epRxOm59}Ud*LDHDYp&Zdm=B{1YV3*!fs^D0)m>41CX!G^mEf&t&2gL2J~T{>!ku z`at6wT&)c1Ze(uN;#dlWPTt^AT!I=1S3sx`jDj7a@n!&oVJD@V=_~`ESh^}U7C%UF zFtuCF)omLu+KA=o+2P^MtvhRMUWKIQOVAZ#63)gZn49LefVVY=^e3LPhC;v)(XFhS z>u+|-+Cl`%hYyU3AtP8FFRfftEVM)OEXSAe9RO+d`Q>0kSx=Wtz&wDf; zi&tF{f8e%Q%_2TGgg|;X^H@LE;Q^7c%H)X1i*GPQ8xLeIlIp+s_=7bIxw_i_pZvEKQ>3j=&?FMc>$zkOMe(5>iSeU?O2MfHIH>+DKZU{&xe5#XOYKicj PT7Zg@hGO|Mi;({Tl|y2w literal 3415 zcmb_f_cz=97yl$yB&9JzRh6h2tH#4qGlGguP@5VZ)TmuMREiEYsmAqpTZ7ZnE>F-ih-`S z)jiPabibc~4T5Do@MgZ}C5dq?7H{rvYr!LtVV;haHWm>H5pk+~G>pJtSPwz9!%QIL z?J6p?*$Q$^sbaC}3#mquX(;945bnpoc+%>4bmj2j*4KG@ZlhvIK1EKveQp-tp;sflS z4}SX;$jwoVae}M%3TBb@f-(BCG-m~}LW z311k8hKz8Ecm+M)P%mwS`Qda^pus{!e?Y+KDQD2B zWjuLo3{6=k`fmQI5d@(}*Q181Mj`he_jbr58C>@^+LzKri!pF}V7#<_PpQz&%C;U{ zmw+W{t0J1#nQ=&npU~H@5560!cFBrXbr9|2B0^~cU|iuMlNCdQc=W{4l5?D+6VaEh zTMw4Le|CpisEssdz5I_WB6-(_;8BOb0Ov8s8pGkEy3dRw%({?pOI-F=klY?eZ? zUVhJNclMhOiaUeo1=K6XJM&%_W3cuMl0&!|dZ*m;OnJ@X0hcbckvNZBg(+D^|Ij*W z^k!?ARMd55LmON%i4$H$oX@f6BX!4A;^vP8 z8cz4BuYM-<o;D&UDP5xiVZj*vOwL(Xgi^WuW~qbXAKq2Luow#G(c({?o;I6o^aPh zY8-5*rVevAtn+kvbMgF0e2aRCg<-9As)UjYZ6KflvEXw~s4oA9`rIcL$EwC#Nl4!Y z{Ra>{I}!nf;fS&)z+jL655PntETI$6U8Y}Ig2{rj%v@0jcn*%`A)a!{%}s7NBl@YZ zF=5*reV$RHd3{o<&n#+Q@`qDF353xaQpB`4xV}riJ9I9)n@3Z)XG}5(V{Q&3aR3@U zfvScEs@b=w&t&>>-{+3xqK!b>z!qBbNS|r5c*fsepeyv}`T2T3^Rl^VEuDJ791>m# z2v4z4^&I6;*?N?Y>{&QA68>t1^-&FL3ENmAhPS{0r|=(*lqbEP>9cOMLGp_HYhQZg z5|nV2{_Izd_;#CdtTqsobR}=S-qFTrJ-x;iS2#i#z#&uT!%~by2H7SHE59gi?MRJ@ z&uPeey)XN;6>?uj&+koIuhrru!~8?iOjP)pOk zZS*!=6WN?lHJ?`i{nB-e%fBUOPJ{yj=4Qw0yy+VSJ~h!ic41=jIWl86;2wQpJ$|c; zR^8lfv6@E+Ml{RZa7=y6$Fm2e{S_LC&C&1z_6HAE5R)AY98`77m2}Wv?2u>t#n znVG&}p_ND4RUXyAe0eXPm~gRFy97$f;5uNp5E%g15TTUE!!9}f9|!fPptQ}hXUJ-Lf~U%GJe zsq^FU`Ls)2UH98$x8x$=Tx0Fa`MacR@Y*8VNB4KDI$rXuP3tLT~d$yTUmB8m)7qg;fcbUj22v9YhPg)l!VIN8UIm#P<%(f!Xxw-=tty8Y31-^i)60)F`@KU!EX(mkf zQ)GeUGN)evp^?tyIxI4pQA!m=31izfrrvagzaMa~$#cu04I6IB;GGvc4WT-%YB+-dV^gTZZh%XO`b}DECWpOoZjqt9 zqktOLcvhMktKKW=LeH#wDjj)gZTsybRlro)>};szu4ZDya*m$j46iaD|7AtPR&)iG z*~&F{db|zcArblJB^#hfDfNHcBoXPrl|fJ_nY6|4PZvm8y%nhrBrMds%ST0DAoy9= zfGS2J3)T=H-9zf)Va%IxUrlHoa+k}BTWY5cQm5cg1m;kyx6jIVo} zncTNdzEOT^iXh`mZlRk{pWp?fwB`;UK8j^m!oH0&482 zLtYN=)+aYNZ4sk7|&V_eX z>Q)oVz#n+pJ})Bur(co;;PZGpQTW%-s;*VNl8sfFGp0FfZcJIui)lqu)fus9RW8x5>XRi#eKcG&_};xJr8+Kr5*T z`xf#w6!*t}>W)r?K}`cUBF1xChxm1CeQ~Iv!hpZ*aAfA2Oj+4dO7$ZY#HUkTBv7VZ z9{ummlF5yEz#3Q3qr@tUyEH39^e^h#n-ossc?E}3wwVM06<*ub6=g#PU8^A^X*rp* zHdbNBWv)qo)pwXWCP(eOSERnk<+Lwz$c=q_b{Oy9D-rhbvBhiC9BkT4BP$o|ked-g z13lVezZV!hdr*Cp&gcWv1m>P7>o8p1rPUe)cvFI#EF&G+lUbFSDxq3w?&ORaa)Y!@?0&a>GT8psQ{JX#@_+az{5K+M YJx2difYK9bhlEpZpl7Q49&", "developer": { "name": "Databricks, inc", "websiteUrl": "https://www.databricks.com/", @@ -10,8 +10,8 @@ "termsOfUseUrl": "https://www.databricks.com/" }, "name": { - "short": "Teams Auth OBO", - "full": "Teams Auth On Behalf Of The User" + "short": "Teams-Databricks Auth OBO", + "full": "Teams-Databricks Auth On Behalf Of The User" }, "description": { "short": "Demonstrates authorization in Teams against Databricks agent using OOB.", @@ -24,10 +24,10 @@ "accentColor": "#abcdef", "bots": [ { - "botId": "8802b634-6b26-4299-bab7-696591b0f2b7", + "botId": "", "scopes": [ - "team", - "personal" + "personal", + "groupChat" ], "isNotificationOnly": false, "supportsCalling": false, @@ -40,8 +40,7 @@ "messageTeamMembers" ], "validDomains": [ - "mychatbotteams.azurewebsites.net", - "token.botframework.com", - "login.microsoftonline.com" + "", + "token.botframework.com" ] } \ No newline at end of file diff --git a/appManifest/outline.png b/appManifest/outline.png index 2c3bf6fa65f152de0cb50056effd5aea7d287ec1..44c01dc9b61c06fbae7f4b0b0b60c08eedfdf84d 100644 GIT binary patch delta 1259 zcmVEX>4Tx04R}tkvmAkKpe)urb?w$Iv7F3 zAwzYtgAc^9R-p(LLaorMgL(7?O&XFE7e~Rh;NWAi>fqw6tAnc`2tGjE1t&!pDe-?v zp+$@b$Nl*3zK^@_0Ipe8G^-~8Xu54?Qb{3~Ull^H2qOv}gMXNi6w8Y2uiw>6FjqJXRTRG1jVerF~EK!bm}1S!O!T5u~t)C5R9p zqmBxyuo0(KCx694hW3*_{$bZIl1nC61qwM9P=f}^^@IJv?{2N))TEabNdmnuj`J}J zgm!^e!*RZk9jA2y1fPK`z2mPnf!R;e8yzin1Vp!ii|dXi?*W%Pz|fN+nUX67X>!FP z@P0<$lm`ZGf$*BwTYDd;4?vc>O5Xqnhrn2gve$jy-4WN_+rMYp{rv#HeR9PenPB6V@BEHYsnSfR3$G+Go zm(~>ATgA{5ivPl#7MJnO_!%#yKfAF%jQ9YPTcEfLABT<_dUJ+$JcFY+5wRQJ1jT6{ z9L0*rvED`kop?K}JP_I3HUhYmw~|We%MIu#l*!fjI>CVX!%*IxV8K_Ip3w&x z&>QY~DngN0$L`ahU>fJogLe-WX7qn?p`*WKKs`D3VsQmNoC=WY$UL|S7Vk$Rk$75MMNpTYBPc(|_$3E^6N8$v(e#*BU*UXIrs$16BL zm*L9n-xYW-Wd6X?0z?n$1wBcf&a24k|L|%d ztAz(vCUJhQkVm&9IMY)#kZdbJZYMS*@qQ?&>j#s(*ocb?xba!ge+^VT?MOC&H8FOl zXL>Z4y2pM>kaK3m4pB$WaYG$S-;9e71>BCwcos&JD)o-hF~}Yq?odyfpQ+hcetSkg zw-D#gSGxcJ delta 363 zcmV-x0hIod3YPpKIe5CzEu75>u=@pET1$f$3 zWVFBih6s7SN0jcbU8_vVs!rH}FBn2sM`RCUcLJ;s=UyoFdPFp2zEd zgg0?Pe-rn}Om*ijZqw>@iY>C(MuTPXb+(~rb;o(IT+JH z_`n2@`UWMFKfoA=^ Date: Fri, 17 Oct 2025 13:56:15 -0600 Subject: [PATCH 07/21] Updating appManifest scopes --- appManifest/manifest.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/appManifest/manifest.json b/appManifest/manifest.json index d0f964f..f5907c9 100644 --- a/appManifest/manifest.json +++ b/appManifest/manifest.json @@ -26,8 +26,8 @@ { "botId": "", "scopes": [ - "personal", - "groupChat" + "team", + "personal" ], "isNotificationOnly": false, "supportsCalling": false, From accc0f27922cc612eed5e732ffecbe1e43b5e5fd Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Thu, 23 Oct 2025 20:50:34 -0700 Subject: [PATCH 08/21] Move existing bot code to azure-bot Signed-off-by: Sid Murching --- .gitignore | 171 ++++++++++++++++++ README.md => azure-bot/README.md | 0 app.py => azure-bot/app.py | 0 .../appManifest}/color.png | Bin .../appManifest}/manifest.json | 0 .../appManifest}/outline.png | Bin {bots => azure-bot/bots}/__init__.py | 0 {bots => azure-bot/bots}/auth_bot.py | 0 {bots => azure-bot/bots}/dialog_bot.py | 0 {client => azure-bot/client}/__init__.py | 0 .../client}/databricks_client.py | 0 config.py => azure-bot/config.py | 0 {dialogs => azure-bot/dialogs}/__init__.py | 0 .../dialogs}/logout_dialog.py | 0 {dialogs => azure-bot/dialogs}/main_dialog.py | 0 {helpers => azure-bot/helpers}/__init__.py | 0 .../helpers}/dialog_helper.py | 0 .../requirements.txt | 0 bots/__pycache__/__init__.cpython-312.pyc | Bin 311 -> 0 bytes bots/__pycache__/auth_bot.cpython-312.pyc | Bin 2156 -> 0 bytes bots/__pycache__/dialog_bot.cpython-312.pyc | Bin 6308 -> 0 bytes client/__pycache__/__init__.cpython-312.pyc | Bin 186 -> 0 bytes .../databricks_client.cpython-312.pyc | Bin 3725 -> 0 bytes dialogs/__pycache__/__init__.cpython-312.pyc | Bin 326 -> 0 bytes .../__pycache__/logout_dialog.cpython-312.pyc | Bin 3194 -> 0 bytes .../__pycache__/main_dialog.cpython-312.pyc | Bin 5744 -> 0 bytes helpers/__pycache__/__init__.cpython-312.pyc | Bin 258 -> 0 bytes .../__pycache__/dialog_helper.cpython-312.pyc | Bin 2852 -> 0 bytes 28 files changed, 171 insertions(+) create mode 100644 .gitignore rename README.md => azure-bot/README.md (100%) rename app.py => azure-bot/app.py (100%) rename {appManifest => azure-bot/appManifest}/color.png (100%) rename {appManifest => azure-bot/appManifest}/manifest.json (100%) rename {appManifest => azure-bot/appManifest}/outline.png (100%) rename {bots => azure-bot/bots}/__init__.py (100%) rename {bots => azure-bot/bots}/auth_bot.py (100%) rename {bots => azure-bot/bots}/dialog_bot.py (100%) rename {client => azure-bot/client}/__init__.py (100%) rename {client => azure-bot/client}/databricks_client.py (100%) rename config.py => azure-bot/config.py (100%) rename {dialogs => azure-bot/dialogs}/__init__.py (100%) rename {dialogs => azure-bot/dialogs}/logout_dialog.py (100%) rename {dialogs => azure-bot/dialogs}/main_dialog.py (100%) rename {helpers => azure-bot/helpers}/__init__.py (100%) rename {helpers => azure-bot/helpers}/dialog_helper.py (100%) rename requirements.txt => azure-bot/requirements.txt (100%) delete mode 100644 bots/__pycache__/__init__.cpython-312.pyc delete mode 100644 bots/__pycache__/auth_bot.cpython-312.pyc delete mode 100644 bots/__pycache__/dialog_bot.cpython-312.pyc delete mode 100644 client/__pycache__/__init__.cpython-312.pyc delete mode 100644 client/__pycache__/databricks_client.cpython-312.pyc delete mode 100644 dialogs/__pycache__/__init__.cpython-312.pyc delete mode 100644 dialogs/__pycache__/logout_dialog.cpython-312.pyc delete mode 100644 dialogs/__pycache__/main_dialog.cpython-312.pyc delete mode 100644 helpers/__pycache__/__init__.cpython-312.pyc delete mode 100644 helpers/__pycache__/dialog_helper.cpython-312.pyc diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..06adadc --- /dev/null +++ b/.gitignore @@ -0,0 +1,171 @@ +.idea +pycache +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +pip-wheel-metadata/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +.python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# Visual Studio Code +.vscode/ + +# IntelliJ +.idea/ +ml-models.iws +ml-models.iml +ml-models.ipr + +.DS_Store +.databricks +.gradio + +# Node.js / Next.js / Web development +cookies.txt +node_modules +.pnp +.pnp.js +coverage +.next/ +out/ +*.pem +npm-debug.log* +yarn-debug.log* +yarn-error.log* +.pnpm-debug.log* +.env.local +.env.development.local +.env.test.local +.env.production.local +.turbo +.vercel +.env*.local + +# Playwright testing +/test-results/ +/playwright-report/ +/blob-report/ +/playwright/* +tsconfig.tsbuildinfo diff --git a/README.md b/azure-bot/README.md similarity index 100% rename from README.md rename to azure-bot/README.md diff --git a/app.py b/azure-bot/app.py similarity index 100% rename from app.py rename to azure-bot/app.py diff --git a/appManifest/color.png b/azure-bot/appManifest/color.png similarity index 100% rename from appManifest/color.png rename to azure-bot/appManifest/color.png diff --git a/appManifest/manifest.json b/azure-bot/appManifest/manifest.json similarity index 100% rename from appManifest/manifest.json rename to azure-bot/appManifest/manifest.json diff --git a/appManifest/outline.png b/azure-bot/appManifest/outline.png similarity index 100% rename from appManifest/outline.png rename to azure-bot/appManifest/outline.png diff --git a/bots/__init__.py b/azure-bot/bots/__init__.py similarity index 100% rename from bots/__init__.py rename to azure-bot/bots/__init__.py diff --git a/bots/auth_bot.py b/azure-bot/bots/auth_bot.py similarity index 100% rename from bots/auth_bot.py rename to azure-bot/bots/auth_bot.py diff --git a/bots/dialog_bot.py b/azure-bot/bots/dialog_bot.py similarity index 100% rename from bots/dialog_bot.py rename to azure-bot/bots/dialog_bot.py diff --git a/client/__init__.py b/azure-bot/client/__init__.py similarity index 100% rename from client/__init__.py rename to azure-bot/client/__init__.py diff --git a/client/databricks_client.py b/azure-bot/client/databricks_client.py similarity index 100% rename from client/databricks_client.py rename to azure-bot/client/databricks_client.py diff --git a/config.py b/azure-bot/config.py similarity index 100% rename from config.py rename to azure-bot/config.py diff --git a/dialogs/__init__.py b/azure-bot/dialogs/__init__.py similarity index 100% rename from dialogs/__init__.py rename to azure-bot/dialogs/__init__.py diff --git a/dialogs/logout_dialog.py b/azure-bot/dialogs/logout_dialog.py similarity index 100% rename from dialogs/logout_dialog.py rename to azure-bot/dialogs/logout_dialog.py diff --git a/dialogs/main_dialog.py b/azure-bot/dialogs/main_dialog.py similarity index 100% rename from dialogs/main_dialog.py rename to azure-bot/dialogs/main_dialog.py diff --git a/helpers/__init__.py b/azure-bot/helpers/__init__.py similarity index 100% rename from helpers/__init__.py rename to azure-bot/helpers/__init__.py diff --git a/helpers/dialog_helper.py b/azure-bot/helpers/dialog_helper.py similarity index 100% rename from helpers/dialog_helper.py rename to azure-bot/helpers/dialog_helper.py diff --git a/requirements.txt b/azure-bot/requirements.txt similarity index 100% rename from requirements.txt rename to azure-bot/requirements.txt diff --git a/bots/__pycache__/__init__.cpython-312.pyc b/bots/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index af519f34aa62ce9d1ab513097597c9f71fa643d2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 311 zcmX@j%ge<81Uo#pXB-C7k3k$5V1hC}s{k3(8B!Qh7;_kM8KW2(L2RZRrd;MIW+0n6 zg(aOSilvfOlkFuVP^l*4El!ur#GL$er~DEihuyKXBm>ASVg^bUu>c7_P1ajnDPSe> zNl=Mf9Em_>U@l0>E%x~M#GIV?_>~Nwfn0`P;rgM)sYS*5xrupcX+^0iddZ17W%c*Ou^=H#ip+ZYAZL~_-ytGmnD8pptCYf}8)_Z4T zH?C0HpeaP^gIXWOH>rY8Qs2z8FD^wA1`2{c*tc!BAtg_qduO+a8ZYd*-~E|$&pF>Y zXMZUavIxdEt)CWN*AV)HFFHeQgVrK|+sH*O_ECUMj5(HkDUeM$P)sFIO;zm4z7}N6 zjKGSo2Zm_~toqp?XXXHFZpP0C1+##Wgw7&YUqP-x3U}o6EwihZE#)5YMF}U$IgiC9 zIq5wUg-e7oJNBaRLTtw*G2UZ@rZ9K0N<+92ldCa&q)mCYAI$^mIv#%XeixySke}99iERcZa{T?2$p^+-IhOARj_PrS^?rtpta2R@&~vi%`Bh2 zM0_U-NHLCz7l|FPB99lu+jB&b6d}OHx31bj#V6(B#ae|Fk3vyI)%Ym?gn9E}5fwO1AO;j6k7kF z;Hie?kq)Q-dxh&|mq7f*X>RIh@TEscGx}EFZfdA!;8yKs?W=Pe!{d$N@r~h=jp36U z!xN3+iF@4>n*%Ry3><3=9NX-_+>)iizHKD+_B|BB01ZtYmOG4_c>VZfKmM^F#CjG_ z_G#-m9k8>PSS7L1@1msH6R|0E!PScKQft>WoFBgLda#{yW9`-#idG zptf@$$)$ZuG{KTexy~d@tD%*43{kqGl9O(S5@?!Mg+@^Hajhh!T#9gWzdPg&0yA-R zg@hI*tP+KcSY!!uVflqU)a`d+6qK-2AOvxLc0Ie%HQMMJ-OTsgn!Y)`nI8hQkw4tX zAKu6xZ{&~vaQM8hvW_*tr2=nIp&by*O0t#} zL~hmRIB!|+S8YE%p*#yI?@pAHCz=T5}K1-CRQv>Td3d>`5#rUzh`afaU% zSuFaZ5WN2}{txhE1vMp2ox#5kzt&VafRsG&d^01Uj(P{T3;}zxYTxFek*~{*L*q@E zFDB2m_}m=Q)a>0?wh>Hp5LAi>W=h?O7S}3XI4|r?x}gouRz2S(wCqHb?D33yk@W33 zPnl-I<)nAd0fQzS*uwYZz=HU@W99U=l`h-)(%yqlDg0ZcCVh)i^LVlB=%>PdUG=sQOhVt^L{tuQ diff --git a/bots/__pycache__/dialog_bot.cpython-312.pyc b/bots/__pycache__/dialog_bot.cpython-312.pyc deleted file mode 100644 index 982f48ae646b0b7cd4c297de07a72addc534c634..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6308 zcmd5=U2Gf25#Ht9QKWbhDUy;bSwAORKW1c+u`Sz)WXqBD6T6aKyRy-UK|)ZxlT4XE z%H7GbSUL@Ar?rcui3>DJP!uT8hd4-6_^krkhx$iGfWDBCnj;;}&SCV}QDftfb*^d;JF<{6=4+H@IBW8Qp; z^;`5hOpf+ji)0p%RbEEtvEMdqNB;yTO&cVpA_=uYLKR7H5(V#`eqfYxYcC(`EW~+d3#lDl3CW1z@3w zONH{w4I&Z_F2cM1OV|>>!ea=-hO;6WSo(7Xw#4huDw*GQ7a>6W2sX)E)DrbmhJhYQ zG+VrVb3JD(3>}4U62tb*{lBapOQEen{r7qn(SG#WF?hS-wQfhU77mHNb|jv-V3LW= zwP%)Wjgp5qB~8CZzsWljMskhOWER3{>GOV$0d~N%wWU&o_`LIJGX_h&MH9>I2jgQj z3D&$|dfBv?`!3)Gbe@_-3-|+S0fPgMCY&Rp7#F#4jElztWF`B!MDIjg6c=}LQXIVS z4DrYX*qz*h7$?pt#zTVcr;G7%gg0UGv z(p|;@^H3`wn9hK1l@Z7YRI4loL&8)rbXK*_iQ$+W5YC6xDoKd(fnY8^zekJ2S|Txo ziXpBC0-;Dyk^%w4)m{;a5ydS~Bx(x&kQ-4Ou%!>6c=S*72wZ}$vr1gda0yqv;`F4Q9>uvY?c8@g`IpnH&M}1@OR-}aXWf$Ts!wrx(@yWTd8mYyJ{Ri%CJLUKCo~37ox~NSl1iJt{h7>d9S^6EtG29y~ggj%hoDv zYnp9cW7{&$7R9+U?cAB_39UMLh2=kH`G03CNZ+reD2(_b9;(2vItRPa5+3ZJ-`)K?iP7mW3$fDG;sYIPfJw=Jce2({e;_qE=U^ zS2?3;O5G^!hdtkA@!CqlgwR_hr`?97+>m<&&ImPWsyFH5An|6CYa`#@} zrnvgkuKrcmLB%zcb`7n%MlKI$+IJ}J{pt38rF|^jKK9|U)pmc1ZM$!Gy}9u7AK!J< zFR@qIRI`80@qDJEPw5y)cML2iS3Aa)*6}ylzq=qu^eK(|(v5(ctBu2odpPADzU$t) zG=6nF)%Mhy`{_($r_$J)ZtP9%4X!p$DekG1dn#)|HT4e}RNJIC99nY^7d3ogwedy8 z{bI`fV#ZPTfS@Ho=ZwR>WV>q1IGQq!j`a%KRs98`tExXEMYbMs$1p+T+7bNdzI#d| z1B3(8FCVhV(Llg?vH;F~%e$7hry2*=*h8Cu^AA^@&nfJ4AG6PC;5-T7{0A`nPwGZH z@jHWuMk>+GEqDacHyauWy$6q&=$pMgK;LS^BQ$+$8%^j=4DD}qd4Rs{z$0e*b`9a* zt|k22b?t=i!y^{@cE83yMEENR1AoP&m9$oxwbG)MmDbVi)QW2mjkZvCTzIt3e8;iDCOr8%ohGt2WD0#}^KcCMlHb*`=pZoLX-{&!XVzcb3 zjJS-K~8MHgVzJE1+s;F-T~RdVK}iB_GCVfFtkIa zSow_;TdN_RO^hWqJE7e1J>O3?oLFN|<{G?x=-L6L`q*!3WZYYp#;%S5Gy|o$yVLG& z#oeEF_b(q?b$<^4t=?0jQ9HU zUm*B=Mhekw{KI2c?QmB(H6M=faHSfGi$c*%$oSRT61i6L=%90@dlq$7kkcmc+;F6;%G>u*uC&Da}c>D`ExO#0bR2AjU^#Mn=XWW*`dyO}aAZ diff --git a/client/__pycache__/databricks_client.cpython-312.pyc b/client/__pycache__/databricks_client.cpython-312.pyc deleted file mode 100644 index 8f861e531ec10c0ecf7c1ab20266d7670aae937d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3725 zcmai1U2N3Y6~6Y^GoJq>lb;!qkO3C54v-0d0;anQfoREgOI9Up+YKtNV>^k5i9Kt3 z0-1VRw2M}Qx-C>IHCX{#9{NBFmGYRjFG$-5q`r7Wq>0?Elxlfs-$JueTb_EZJ)Q|6 zRj)GV-s^Mjz2}~v@7(-38Vw<6@v%S7j06$-8c)1N)dw3#VX%NiBocWv;r7G?0dq1> z&iW>N1R~KnB>HY3k$OnFt0t&-POg8i;XYbjH(-Uhc0PzvNFv{idax(#o!cvRo`taXH2i$<+c1GFAd@f=2P3zcB+LkJW?)&T<^0~L*>B=9&9C0b z^gXQa^N8?9uy~;DNf1tY5=tXaOKLJ!2S!Yj${8pI3zX5d)S>2n( z{qCiU zBt@u4FP3mT`Z4qyzXkFxBCuCc4s*7&JCGUo$TDdXA}RVrN~G^nWgp0H-mShS*YH$x z6NnDqBfpn9A47)y7$WIcHq;mMOTiV0=TnwaOG>(p=PrH8SyGpRkSSglK zWl_k?9m+V@D!o?Jl&mZnQ`rKq@w0l?xpUTG9yI0)QZ{$hC}p;nn$~#5aHZS@(lud* zSEi*B!{_q{`GTxN&_wm(WKI$wIj|w^wwIxu_L@7NV@JW zW+^xLVsI~;D&UNya=NOpv=)a?S^|If6}7yEZ*DW1q!(00mvn4f9HVKi1v6rvqVBe( zzE@)DWdVGiJ_+RS=-=1TT5CtGb#JwGuhp{ezGB6WRebAywDa)l?$Om9@2+#t<)cqFtRB_x@Opc8Cf?}ZQl0lh9>c|+oQ|`tY1Jz15-@WLjxyX zFcNKPB7x)cZbOa$#5O6JDAT4-gP%!@)EltIq{k6UkJ(fa(wrr2N>lTa9S~H-fV{S8 zSt%3^Z8u0_W*(Ha{d}PSPzhM$_$LiKZ5Jk3%4TN4v;4HA+kpZ%x4#$EE&0JlyH1LYk{-wA_l{jNk8E8mT5vt}xlP z#E!*Lt8H&3@V9X5YIsD}XM0PGvy#HbIU27!kkkq_(Vml+b7 zfn4NO$s7*-ElrF-e4rf8q%BP*Oy<|bFNku`z^Ul8T)Q`F_9A5l@1T2|B5wQp1Afxm zp9CELJ*pfM10rM6crP({mn?@ty0dwXH_Bl+*wW09$(Z3sA5G<7x&aH{Zf+AX5lbSaUkzhVx0uNXa3iezMEP8E1V2EZ(}K;GtM zcb35jTMC?!cumsMELn=2bu<3pMU36iVJMQSCYPM0EUERtEdZ+|9!jWQ>c3FOU{@Do zMoL4*D@IyY(#-_UrhkLj77ZIf!!hZ=yxm&Yv+>%}?Y3#j;P8bwUamh2v;+9Spm%2- z-YX7fxG$SPUY(wX7n@DV%9N_%aMrL_?G}xfb%}#-oNn+&QFj&h0k*QSO>l(+c@t*E z+(&^pSubZTSH1UZT^&2>+S}7^KVH8RpBQw!!{PFD7|)~Fu$-s29mOEunCe(l7YRM; z)~*D+s`8w>mFK_GYfV$uA4A(nVNLWwfGD?$_3f$k9j^8rwvL>8+BaHDjaFjMx(Al1 z+R#XKXynP*O7|s;NvIC zja33`i9@&FtR)7jiGf<;P!<1OUZ!d%E>urkuo9Q8@BI)yRG4Q>{8r><#OgY_^n;~u zTOFrXnA2-r-Lg#wdG49}djoDRYcC9db){^NjqMt|a#4FLp>2EEUmsP7{ zY=wF6S+aW(ZZ-fni}zSe&-yX6;~nBZ>uJ<}k$9o+0lR(FF>;JpAV!9$#j`OOez}hr zIYNEeKOBO|pNkX^$6u*QRn0q*t)0P~BbeYKuizeAMZlv~S?q^PV2M7jvL{0{cK`z5jvR3)A1= zxvK&`9iCjcplV?tcC1;R2X*m^o5GHcFYEn+dBZiuTiQFY4p&Qk2at6iK@fjIv9D3& SuV~<3)G$Hx{u5#99Qa>GBBNCR diff --git a/dialogs/__pycache__/__init__.cpython-312.pyc b/dialogs/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 43485530f917cf702e30409e3ef53b8ed8a68951..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 326 zcmX@j%ge<81Uo#pXIur+k3k$5V1hC}s{k3(8B!Qh7;_kM8KW2(L2RZRrd;MIW+0n6 zg(aOSilvfOlkFuVP^l*4Egqlz^!(Bim(0YR{B$6n%QrDI55g;A1}Z6H0TOGcq!M^!v KM)o33pd0`t|67^> diff --git a/dialogs/__pycache__/logout_dialog.cpython-312.pyc b/dialogs/__pycache__/logout_dialog.cpython-312.pyc deleted file mode 100644 index 46615e6d14624d01c4c05e664e7e70bd5039402d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3194 zcmbVOUu+vm8K2od_QsCu#NNc$rXjvduhfv(SLu9BoIm}O1atyQd&vo~gqDqWoY-A^ z!|b|kf{&|c1wlHb-ow$}+dbw~PO8Kqod5|Q$`dbH0$Q`WlTLyMc(W9$Mt$O&T{}&- zt#t5N{$}Qz*>7fk-@o~DEEYjP((gZ6{X-a`f3QK9#7<}TYtXraG^Al2)o>BlxFT2M zi#%pBP8VuoQS|z}9;$_lVT?HRb)*TmkR}rD9`EZZO6#Fa_}^@h;oZLZimDqcOAT6o zgP0B7a^rJGZOy2Yy5&!~F~6NN>K3_fWw0B6(jD0aSNRjyX7^`-SS zVwU`!d>-lyc7Fh!pTPwPDq;;4ISmtD;|O0CHc=UC{H@-ipb12T!-ljF3712jJPdNl zlQWsWRsvZMla)EG=ax{6Y@+M97+pk}h}-*3W5sB658tDC_fsmG8!s94I)QT zNT~yecRto<|J?$x_wXHLd1A&auG?**UqlgV;(OSaVASNANZHr3jy}W}GlDCcjWt4P z53F(}MXA&)mZDJj6v3om61_~Bxjl*7qbMa^HBCh^1rVmw=^Kxg9D~@*)zo^qOo^5) zsrpsJRO_WxvXPrNN{t%eZss)AQkQ9^bj8fg?{_a4)*_)-D-u%+dU^cwRsHK-VO9mDfW(jcixu1vNicvDgXIMc2oG9bbJ?P;bs+plPX%t z4I66=RnrYoVu2?eLY7_XCj57Q4D3Dp4TQl5nyA{r5QO{m1lgP5(mw)Rf63k9e=3UT zExyT9uE{SRu*S~?um_yBiJRQLj*WufEuu5%oizMIAMw{Pjlovn^bs(L1qN!YS@Z~q z+&H+zGFhqAI}l|=+6QtfGd93hz|B~6(0&-0EF!^9=^He`790h(ar_XN0euW^^E|M< ze=%4;PYvC9``x!cxW1Em#p!=#GxYape=B)(^IM+|9DDaW_Q2TIckf@@893|2&u&X+ z+hLSAz1I$*zH|7o`2uLUIWU9o;z{BCp&9%!o)Uf|&fwqS8C=@;R9M8k(!;$8o&Otm zuE_!J+#%fgF7DkpcqZy20n{lwxes>&;DKPYD)k2GA}$<2oOz#*bT=Y=ppS!mVS(UL8=}!PiL~qZWth|n=(;V3QWQ9$|;0yV|t|d#rN=P%VAAb7778n*BQSB z2G=m#hq}+%g+)z(sw^}DWU=`0sLDbqKpN_XEer2zDliSh=j{G09GQ0osLK2(K#+Kt z7X|2l*yl@uPhr@ttidpV^6+dMz4){ajeA}&d0IxF{-SHG;I4~k9exme=R2t{!ct#` zzvzbiGO%&8@ZwQK#@ML2kH4 zOjBJUAY*Gp-MB`m8(~RbF)h_1F26!7${d(gK2~NG#f`Bo6jp&6P}ba7nHn`^jT)fU zT6ejM=2yZ+Gq!J9nq?v@O7&iDm;{K8K3 zLMu7sBuDM!sFNJClVeWuw4FTdB+uB%Gds!2&G}Y5dHecb;v=p6jFVrm^9xS?lAXWg zvMBahkg*p{}-bFU?~`(A|fo;u|W=Iz0JOFr(%Cv5qIBahqixFb*5@{}Xb z+VX5me!-DX+48BU)=b#)#BNwfB_1K6KhgGB1f@ot!ISpjNoVj?d+^om{H2}2%TC|r z?Y_&c;g_A^343^AXSm>`3fp~!_5kWjw?))DhugiNW3L@S{cqsCRtl2gYXHtq=Fvg~ zKj83!AUxn-V*Qu#LP&UUGR68cctH{#%!Z-=rx0dbVm#AJzs!siQIwjYHFV|=y^8X~ zhN^eGgEuYeJxD0aq;AMquEHyg@-Xm9D`f%eRS!4Tc`Jb{=!%lTevx1=iF3Y%+2h4~ zcd^82z6osa7HV^XnEPN9KoN7F@DEwj?iIyzc&pY%to>uzZ?`Y24?RKqh$t5B4>M_D zdv1{l3+*E@acXP$5rT#$yZcaZNyj(o%f(gnp`Nu)`xy1TM0 zl@5Z$Xcg2&T@;O7q-YbMKwQ{G`%x4M@=@DKe4s$V4IM1AbpQfw(e$GLQ8|e0{`Ad~ zOUg1`80Y}|=FRcuy*D%OnEjK@R)L@lAN*+U7fyu!N*Y=*6p*!5KrSK%F*t^Tq}Ii7 zTo=^E^+7!*ZGDW28-j+oF=$lVRLm4N2hDLy(4w{tv5I(Quu{d1F>Blww5hl$W{*39 z4i(2SXV9rbtTWoG2~et`x4qQTu&d@PIFDnbL!*eXoI^|n>$sxN_Xgcd*kAci(wOw? z6w5Jz<)_4u$SPZgxx{If7eZo$OQ>Bi?l=KM{7fho+aC$VxLKvD*fAxt^FSi9XGEA- zd1&u~ICq%m;`5^7-q4t0MKO|?75rGS?hT8P(-Cp$=+Zn3ZRh@w7@FoI;ZwqJEW##4 zGUp&S%Poj`ZqPQw*wbt*oEJV1)zblV?I3{f!zr*Rh#3^rF__geI{5VPQ7pw6Si?-k z3Yx);@tieiWK67SM#q@XQ9*M7w*YQgL1%EVVhZ^yl!})_kwigNN0>__SP+>@(1}o- zRUAxFUV4rbM5P)w?=&b!vk7LNizLMSyov?rrv-IaU9d=TSMz;g;|6UkfRm$HtI{F# zQ+yGL#X%8z@+FCWScy=n9o@KsHV0oR%*QB6B@s;(sSVK5E-p%9Xcg&`C~DT`mFlRY zgliaswTlJxMH-;jjzgOWby<4d2702kTEyxiR;6L(v=cqrXc9AUxQyYd@k&8MO>dCP zs?8GXH7u!1>Nm`=7yD7PS;I>;=u=y2X_zKC>}**ou|=E7m|#X3V};%_#yYFhBr9J3bghGCbC+DnC#Do=t7!hNvViF^977S1^pBO0^ zVU=PK7Uo%=Cwr%uX%HbI(lo({StX^gPc2nYg65Rf7NpA_+Pa z4zq&5@q7)LZz6!)CdJByz=04N@E|<7RLYhO_7+whgi;Y=7@ARC5l^nP;$joR0?*O{ zYyqq;fKyUzq4@|M2A`F0ZXxmlNr8@zFn*Il32bae6_*DekN7PF6!lOnG#!qG1c9an zBBD<%Z#;ahm-tJeHy%pN%_7&>9Bh%WyLfthtXPX10ZpgNEelYa@kleOUYTI{hDBU(ix_uc>z3d4{o`CGx zBYF0$E~P!=KyLk8L+3qbz3gn4obC6VZJG8R8SjBiU0bH1^}fZh#kFQIH`>Vx-uo`p z+V{KO)bOk6mXoBXDQmX=$nm!05+&FArCR@8b626~p0z&Xtjku_%2nM`RdSA|-8R;bU-O>3Yl+zE1=GUf(-u7O&T5N8;q#Z%!J z@GZJSZ4S6VaM+tmdA>~132f#J?gmF~n^%EAP;_EZa(6eCC+RUocIf|;wCG67N?LSU zC4rJfr@xtE3Js$3zlBeP?+|=#@L7MOU&MZD5}XlticO3h^;hsdSWU6dvLdZI5p_T@ z74C#$f`O1I@;ve7ie6wv#X8LrKT~iLs)K1HgH(cKDtf*H8hn?!s=64>v%)-=5Lkg+ z6)<#u8=27tV7YOFEk5pO_P`Y3BLH6!yUaGX$<5oP=IwIxkkmYM%__fiRC?)Xy7`#w zJ+@;1!qNDv6PLfedg2;$ck4vPRU^B)Bv+U03P`TN<)qv@EcFgwJC=4$$kvIJbt3EC zDtr4RZ{P1$kiQEb;oh5K$3V(FkaM8A<9I!5omkJh$JR6Ek@c)&;Gu8=SK?A;EkBTW)-%m zu#Lb+ES97kZ^5_r2e7?@#o}@n3-*Sh8*LqkNQo9xn0%W+u#aLYQ|g6Or9i*=)}7UZ zEiWc@z?+iT!QTmY2fE^7RY`j7Cv4c{z$N7|6sl;1Ig+7Zv5HI-7~a%u2}lQ3!0YPBlm z)7l56QK>r&9_1a1+>Aj;44z8cM<~L&lMwUInqD_e73xL2h|cIwqD2f^z7O9I)W~F< z967pNbCO%|g&@sNaErctQtpc+0?QqHXCe3VMFbcSeWBB#NGvoRV|^hJ@^9#kb0jwm zEZ2~SGGLLg`qK8zz(mL-{Vv4_k!T?%7GfDhHa-BG%l8nl9e`pJ;Gqc_bp92sRFGCa z&M9WCmnSbJcs6s3kg=;dHhG*VZZ#i{a||0RzE&Y#io9a^)|oInPhPo-iKqjw2E_oh z9Pc-mf+fF$NI}d!&%h4+79giCJVg|<7DW}z3M9)=Gv;D6ZHJ!a+6~e4kraI#MumSt zCZ`6{42gU_lDB8Yo@r{8n+Bw&0lDdUsp8!TFtfx)(bW5IY+0!q1`en~9$+JuL3`w4$v}f-H zD%08hLGt~i+&L(94z60|od>0z2X9ZOJCDdMM=sbiEnDT5L8)a>Zkdo;CT{P)`^vG@ zaYjBKm5xW#ubfK7xpd3CTt9!ooHHWeRnsNabjdY4rJ9|qW2?`nYer?~Xv#U7Y4ppD z{ZeCpx^Y1E45XX`IV*BEKS?olKOa#n_bl?hiXQ+M+XQ})wN5_By89mFDp2bycs)}; zzYZf?#sKvUJh_&3tq2JbmN7i~Yb{XG0$%eymoom(?Cts4bQ?Fyx9re)`}ZYy_wgP^`i$`my z8+9InKZ{4b)QvtD;2#_IdPdvPCk=SCf%>Fz5b#f{@MshDX>}XHcj8eW_317$=cW^n zwo*5@=m}oc48%8g;PGneWzL+b!cR z{jFzqkK6RO{dla$c)P0>@H;v@ZZO`_Qv^3*m~+QWR=#615Zpx(+(TCOSMq8`T0zrb zkP9(*LMv99{tnnv;X%C{c7`XhhzCc7)OR6IT%($m^2C@FTj7NoX5pPn`!U}My#m+} zdMTg4l1GC2WG8vKFbQD&9Lnjo8G0`3a|o(TOM6Zab#_~K&HyzWdQ)f4L@+bz82s8o zu%p;w7)(7kzJ{RA)zlewq_*$>20_jHH3t4FMaKz>W117E7a}o+f0Uj%{DGI`315 PG2U^Z;cG-{mCt_xj(P)) diff --git a/helpers/__pycache__/__init__.cpython-312.pyc b/helpers/__pycache__/__init__.cpython-312.pyc deleted file mode 100644 index 884e611498dac70b62305a0c9118ad186bace73e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 258 zcmX@j%ge<81Uo#pXUqW7k3k$5V1hC}ivbza8B!Qh7;_kM8KW2(8B&`Sl7&-6`tY#kd#*@P1#h`(9$Y&>~2a+RSHCbEE|w3DYcW(u9Y0Z1fW;DBWWpe zSDjr-CMy|1V-$8#z{n*|5CrM5g%iLz#XU7fV7EXIB`dWQHZEeIIW#xc3J}-LA#e6a zRAK`jU}xUE`QDrN9%sJcU%R`z5VY$jK3<4K5&AcqgpCh`t>1uf1?fn~WmI5eq>L+( zLZreKILzi;nXd>1p&}Ny}AgwrwM^#~Dkj)-%j&R^iK2GByZCMcpS`CI;slp$DG)cu$w2SO-neynax4N}m! zz?cmJd~d+FbsDtKBMke-krPx2p;AyA1`V5zJ5_X;z1LPFp>~a1-N83s+V!CCuwoB*7FLCl^dR2Z9qLNS zQ0k0}KSYR5KO!b>6_rvQn^1Zez4zT{_k}B+@!^(X{l1?AMrQ62vG5Pi@?p2+d-37Y zW1YA{>0x%du#D*7BjOOl958Yo554Kp4`QKy2G!6g`t5!A{sy0wMNTHf9YFN-nz&hV?ayBX+RIw~jEqN~wTwu1|$@La&T!KEidZD-VXg^_tx2i|&%XJ>L0t`0`8 zz&TwL&x^HST)`{o9QO)Z!5?xfSdsE|<%DTCa377jVb06jEFr6=E~~oEl>T)cst+() zG0ZB{sb)9l)WyC99faER)bi28N*?#w06&a!js2zQa@CogsmX#$+W@-)kH5 zreO{{KuPA*Ic-5T=Se*kXqO2!=4u}M)qiqoqUtQjOVp|?IkJa`uzUmAqVf)`3|{my z(TBeL3sesnPPDxp>**;<0ZOYT>=`CACR0W&^&ihzl-6>xZB>Y~z}8oY>BuY80udbe z$vF$#87n<*6bN@7pe}Eia~2iBkIO+V-Q5}`z}efBmtfon1tmql2y_oXDWfJ=n1Y0A zZp6@CX_kPu0OIndS|N-qq0EL;qcMh#Ly^dmV_2pu5SEP_cc`k7Syfwf<4e>qog$RQ z?Y4=j7eiYmMj5ftZq_k6al8CDiZ(nGy8}7o=O`t1wd~j~CX}6Po9!$I?S66iWt&iY zxT2bKa}=tqspVzMR!wbz)Q2Z6ty+P)*~7Z(sI%127VY85ZSlC}%n-V4Xk>W7dm;em zJrPBJI;iax--Sw$yX^g-b6q`X)D| zsoQ-|{O;NHzTCx=n^O96{!;$!XEvlK?qmc_iN$JJr~K zdL#Mswuq9MR^mW2aiEn@nhB+G_|<0O)o>Tj_3hp#e?PO)JJLvwv;~xW4Y&JH|MU1B z*!CX1*N&k*NB{R#3MKpQB0ea@iuBXnnq_h8&&i=n0XXfA74pjG7c={l|FdcP`#@z!2}$$bJC z-0Bx#`Bs(#Ij_XrSh2{?eX;0v6^j*1ua=oC6^p;9s^wrsV)xn5*gds$${v2o{>4#v zrV%{|-5<~v^oTMRrR)jzI(7x@5$GuI()B&PoB zp=d7jp>$LBkh+Ok%bBejWu4F=&7#ELn?>QKb}ada+xBB*^x)t5Jv$&a+?$H(-|k(r zs%>aMnR5nk%8C5+DL?ujF-tE7meKv#xL=rK^l6x~zex78(6u>?@m37s#OLVP=jeD7 o9sd`~!6CtWUyrs>PZRaLedzt%JGppjok0t%I@-T(jq From 57a252bd7c8843b077ab9cace19d18f0d25756e6 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Fri, 24 Oct 2025 09:35:48 -0700 Subject: [PATCH 09/21] Initial slackbot code Signed-off-by: Sid Murching --- slack-bot/.env.example | 24 + slack-bot/.gitignore | 46 + slack-bot/OAUTH_SETUP.md | 233 ++ slack-bot/README.md | 269 ++ slack-bot/RUN.md | 112 + slack-bot/app.py | 350 +++ slack-bot/client/__init__.py | 3 + slack-bot/client/databricks_client.py | 199 ++ slack-bot/config.py | 32 + slack-bot/hackathon.py | 275 ++ slack-bot/handlers/__init__.py | 4 + slack-bot/handlers/auth_handler.py | 166 ++ slack-bot/handlers/message_handler.py | 190 ++ slack-bot/pyproject.toml | 29 + slack-bot/references/databricks-oauth-flow.md | 182 ++ slack-bot/requirements.txt | 18 + slack-bot/storage/__init__.py | 3 + slack-bot/storage/state_manager.py | 81 + slack-bot/uv.lock | 2609 +++++++++++++++++ 19 files changed, 4825 insertions(+) create mode 100644 slack-bot/.env.example create mode 100644 slack-bot/.gitignore create mode 100644 slack-bot/OAUTH_SETUP.md create mode 100644 slack-bot/README.md create mode 100644 slack-bot/RUN.md create mode 100644 slack-bot/app.py create mode 100644 slack-bot/client/__init__.py create mode 100644 slack-bot/client/databricks_client.py create mode 100644 slack-bot/config.py create mode 100644 slack-bot/hackathon.py create mode 100644 slack-bot/handlers/__init__.py create mode 100644 slack-bot/handlers/auth_handler.py create mode 100644 slack-bot/handlers/message_handler.py create mode 100644 slack-bot/pyproject.toml create mode 100644 slack-bot/references/databricks-oauth-flow.md create mode 100644 slack-bot/requirements.txt create mode 100644 slack-bot/storage/__init__.py create mode 100644 slack-bot/storage/state_manager.py create mode 100644 slack-bot/uv.lock diff --git a/slack-bot/.env.example b/slack-bot/.env.example new file mode 100644 index 0000000..43428d1 --- /dev/null +++ b/slack-bot/.env.example @@ -0,0 +1,24 @@ +# Slack Configuration +SLACK_BOT_TOKEN=xoxb-your-bot-token-here +SLACK_SIGNING_SECRET=your-signing-secret-here +SLACK_CLIENT_ID=your-client-id-here +SLACK_CLIENT_SECRET=your-client-secret-here + +# Databricks Configuration +DATABRICKS_HOST=https://your-workspace.databricks.com +# For local development with databricks-cli, leave CLIENT_ID and CLIENT_SECRET empty +# For custom OAuth app, provide your credentials +DATABRICKS_CLIENT_ID= +DATABRICKS_CLIENT_SECRET= +SERVING_ENDPOINT_NAME=your-agent-endpoint-name + +# Optional: State Storage Configuration +STATE_STORAGE=memory # Options: memory, redis, dynamodb +REDIS_URL=redis://localhost:6379 + +# Server Configuration +PORT=3000 + +# OAuth Redirect URI (leave empty for local development, uses http://localhost:PORT/oauth/callback) +# For production, set to your public callback URL +OAUTH_REDIRECT_URI= diff --git a/slack-bot/.gitignore b/slack-bot/.gitignore new file mode 100644 index 0000000..d8576d4 --- /dev/null +++ b/slack-bot/.gitignore @@ -0,0 +1,46 @@ +# Environment variables +.env +.env.local +.env.*.local + +# Python +__pycache__/ +*.py[cod] +*$py.class +*.so +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +*.egg-info/ +.installed.cfg +*.egg + +# Virtual environments +venv/ +.venv/ +ENV/ +env/ + +# IDE +.vscode/ +.idea/ +*.swp +*.swo +*~ + +# Logs +*.log + +# OS +.DS_Store +Thumbs.db diff --git a/slack-bot/OAUTH_SETUP.md b/slack-bot/OAUTH_SETUP.md new file mode 100644 index 0000000..641fac9 --- /dev/null +++ b/slack-bot/OAUTH_SETUP.md @@ -0,0 +1,233 @@ +# OAuth Setup for Local Development + +This guide explains how to configure Databricks OAuth for local development with the Slack bot. + +## OAuth Flow with Localhost + +The bot uses **OAuth 2.0 with PKCE** (Proof Key for Code Exchange) for secure authentication. For local development, the callback URL points to your local machine. + +### Architecture + +``` +User clicks "Sign in" → Databricks OAuth → User authenticates → +Databricks redirects to http://localhost:3000/oauth/callback → +Bot exchanges code for token → User authenticated +``` + +## Setup Options + +### Option 1: Using `databricks-cli` (Recommended for Development) + +The simplest approach for local development is to use the built-in `databricks-cli` OAuth client: + +1. **Leave OAuth credentials empty in `.env`**: + ```bash + DATABRICKS_CLIENT_ID= + DATABRICKS_CLIENT_SECRET= + ``` + +2. **The bot will automatically use `databricks-cli`** as the client ID + +3. **No Databricks OAuth app configuration needed!** The `databricks-cli` client is pre-configured in all Databricks workspaces + +4. **Start your bot**: + ```bash + python app.py + ``` + +5. **The OAuth callback URL** is automatically set to: + ``` + http://localhost:3000/oauth/callback + ``` + +6. **Test the flow**: + - Open Slack and message your bot + - Click the "Sign in to Databricks" button + - Log in to Databricks + - You'll be redirected to localhost (the bot handles the callback) + - Authentication complete! + +### Option 2: Custom OAuth App (For Production) + +If you want to use your own OAuth app (required for production deployment): + +1. **Create OAuth app in Databricks**: + - Go to your Databricks workspace + - Navigate to: **Settings → Developer → OAuth Apps** + - Click "Create App" + +2. **Configure the app**: + - **App Name**: Slack Bot + - **Redirect URLs**: Add your callback URLs + - For local dev: `http://localhost:3000/oauth/callback` + - For production: `https://your-domain.com/oauth/callback` + - **Scopes**: Select `all-apis` and `offline_access` + +3. **Save credentials to `.env`**: + ```bash + DATABRICKS_CLIENT_ID=your-client-id-here + DATABRICKS_CLIENT_SECRET=your-client-secret-here + ``` + +4. **Update redirect URI for production** (in `.env`): + ```bash + OAUTH_REDIRECT_URI=https://your-domain.com/oauth/callback + ``` + +## Key Implementation Details + +### PKCE Flow (RFC 7636) + +The bot implements PKCE for enhanced security: + +1. **Generate code verifier** (random 32-byte string) +2. **Generate code challenge** (SHA256 hash of verifier) +3. **Authorization request** includes `code_challenge` and `code_challenge_method=S256` +4. **Token exchange** includes the original `code_verifier` + +This prevents authorization code interception attacks. + +### Code Flow + +**Authorization URL** (`handlers/auth_handler.py:51-60`): +```python +oauth_url = ( + f"{DATABRICKS_HOST}/oidc/v1/authorize" + f"?client_id={client_id}" + f"&redirect_uri={redirect_uri}" + f"&response_type=code" + f"&state={state}" + f"&code_challenge={code_challenge}" + f"&code_challenge_method=S256" + f"&scope=all-apis+offline_access" +) +``` + +**Token Exchange** (`handlers/auth_handler.py:88-135`): +```python +POST /oidc/v1/token +{ + "grant_type": "authorization_code", + "code": "", + "redirect_uri": "http://localhost:3000/oauth/callback", + "client_id": "databricks-cli", + "code_verifier": "" +} +``` + +**Callback Handler** (`app.py:175-296`): +- Receives authorization code from Databricks +- Validates state parameter (CSRF protection) +- Exchanges code for access token using PKCE +- Stores token in state manager +- Sends success message to Slack + +## Databricks OAuth App Configuration + +### If Using Custom OAuth App + +You need to add the redirect URI to your Databricks OAuth app: + +1. **Open Databricks** +2. **Settings → Developer → OAuth Apps → Your App → Edit** +3. **Add Redirect URLs**: + ``` + http://localhost:3000/oauth/callback + ``` + +4. **Save changes** + +### If Using databricks-cli + +**No configuration needed!** The `databricks-cli` client is pre-registered with a wildcard redirect URI that allows `localhost` callbacks. + +## Testing Locally + +1. **Start the bot**: + ```bash + cd slack-bot + python app.py + ``` + +2. **You should see**: + ``` + INFO - Starting bot in HTTP mode on port 3000... + INFO - OAuth callback available at: http://localhost:3000/oauth/callback + ``` + +3. **Test in Slack**: + - Invite bot to a channel: `/invite @DatabricksBot` + - Send a message: `Hello!` + - Bot responds with auth button (ephemeral, only visible to you) + - Click "Sign in to Databricks" + - Browser opens to Databricks login + - After login, redirected to `http://localhost:3000/oauth/callback` + - Success page shown + - Return to Slack - you're authenticated! + +## Troubleshooting + +### "Redirect URI mismatch" error + +**Cause**: The redirect URI in the OAuth request doesn't match what's configured in Databricks. + +**Solution**: +- If using `databricks-cli`: Ensure `DATABRICKS_CLIENT_ID` is empty in `.env` +- If using custom app: Add `http://localhost:3000/oauth/callback` to allowed redirect URIs + +### "Invalid client_id" error + +**Cause**: Client ID not recognized by Databricks. + +**Solution**: +- Verify `DATABRICKS_HOST` is correct (include `https://`) +- If using custom app, verify `DATABRICKS_CLIENT_ID` matches your OAuth app +- If using `databricks-cli`, leave `DATABRICKS_CLIENT_ID` empty + +### Callback URL not reachable + +**Cause**: Bot not running or port blocked. + +**Solution**: +- Ensure bot is running: `python app.py` +- Check port is available: `lsof -i :3000` +- Check firewall isn't blocking localhost connections + +### Token exchange fails + +**Cause**: PKCE validation failed or invalid authorization code. + +**Solution**: +- Check logs for detailed error +- Ensure `code_verifier` is stored correctly +- Don't reuse authorization codes (they're single-use) +- Authorization codes expire quickly (usually 10 minutes) + +## Security Considerations + +1. **State parameter**: Validates the OAuth callback came from our request (CSRF protection) +2. **PKCE**: Protects against authorization code interception +3. **Token storage**: Currently in-memory (replace with encrypted storage for production) +4. **HTTPS**: Use HTTPS for production deployments +5. **Token expiration**: Tokens are validated and expired automatically + +## Production Deployment + +For production, you'll need: + +1. **Public HTTPS endpoint** (e.g., `https://your-bot.com`) +2. **Custom OAuth app** with production redirect URI +3. **Persistent state storage** (Redis/DynamoDB) +4. **Encrypted token storage** +5. **Proper secret management** (AWS Secrets Manager, etc.) + +Update `.env`: +```bash +OAUTH_REDIRECT_URI=https://your-bot.com/oauth/callback +DATABRICKS_CLIENT_ID=your-client-id +DATABRICKS_CLIENT_SECRET=your-client-secret +STATE_STORAGE=redis +REDIS_URL=redis://your-redis-server:6379 +``` + +And add `https://your-bot.com/oauth/callback` to your Databricks OAuth app's allowed redirect URIs. diff --git a/slack-bot/README.md b/slack-bot/README.md new file mode 100644 index 0000000..6591851 --- /dev/null +++ b/slack-bot/README.md @@ -0,0 +1,269 @@ +# Databricks Slack Bot + +A Slack bot that integrates with Databricks agents, providing conversational AI capabilities directly in Slack channels and threads. + +## Features + +- **OAuth Authentication**: Secure authentication with Databricks on behalf of each user +- **Thread-based Conversations**: Maintains conversation context within Slack threads +- **Ephemeral Messages**: Private authentication prompts visible only to the user +- **Tool Call Visualization**: Rich formatting for agent tool calls and results +- **Conversation History**: Per-thread conversation history for context-aware responses +- **State Management**: In-memory state storage (extensible to Redis/DynamoDB for production) + +## Architecture + +``` +slack-bot/ +ā”œā”€ā”€ app.py # Main application entry point +ā”œā”€ā”€ config.py # Configuration management +ā”œā”€ā”€ handlers/ +│ ā”œā”€ā”€ message_handler.py # Message event handling +│ └── auth_handler.py # OAuth flow management +ā”œā”€ā”€ storage/ +│ └── state_manager.py # State and conversation history +└── client/ + └── databricks_client.py # Databricks API integration +``` + +## Setup + +### 1. Prerequisites + +- Python 3.8+ +- Slack workspace with admin access +- Databricks workspace with agent serving endpoint +- OAuth application configured in Databricks + +### 2. Create Slack App + +1. Go to [api.slack.com/apps](https://api.slack.com/apps) and create a new app +2. Choose "From scratch" and select your workspace +3. Configure **OAuth & Permissions**: + - Add these **Bot Token Scopes**: + - `app_mentions:read` - Read mentions + - `chat:write` - Send messages + - `channels:history` - Read channel messages + - `groups:history` - Read private channel messages + - `im:history` - Read DM messages + - `im:write` - Send DMs + - `users:read` - Read user info + - Install app to workspace and save the **Bot User OAuth Token** + +4. Configure **Event Subscriptions**: + - Enable events + - Subscribe to bot events: + - `app_mention` - When bot is mentioned + - `message.channels` - Messages in channels + - `message.groups` - Messages in private channels + - `message.im` - Direct messages + - `app_home_opened` - When home tab is opened + +5. For development, enable **Socket Mode**: + - Go to **Socket Mode** and enable it + - Create an app-level token with `connections:write` scope + - Save the **App-Level Token** (starts with `xapp-`) + +6. Configure **App Home**: + - Enable the Home tab + - Enable Messages tab + +### 3. Environment Variables + +Create a `.env` file in the `slack-bot/` directory: + +```bash +# Slack Configuration +SLACK_BOT_TOKEN=xoxb-your-bot-token +SLACK_SIGNING_SECRET=your-signing-secret +SLACK_CLIENT_ID=your-client-id +SLACK_CLIENT_SECRET=your-client-secret + +# Databricks Configuration +DATABRICKS_HOST=https://your-workspace.databricks.com +DATABRICKS_CLIENT_ID=your-databricks-oauth-client-id +DATABRICKS_CLIENT_SECRET=your-databricks-oauth-client-secret +SERVING_ENDPOINT_NAME=your-agent-endpoint-name + +# Optional: State Storage +STATE_STORAGE=memory # or redis, dynamodb +REDIS_URL=redis://localhost:6379 + +# Server Configuration (for HTTP mode) +PORT=3000 +``` + +### 4. Install Dependencies + +```bash +cd slack-bot +pip install -r requirements.txt +``` + +### 5. Configure Databricks OAuth + +1. In your Databricks workspace, create an OAuth application: + ``` + Settings → Developer → OAuth Applications → Create App + ``` + +2. Configure: + - **Redirect URL**: `https://your-app.com/oauth/callback` + - **Scopes**: `all-apis` + +3. Save the Client ID and Client Secret + +### 6. Run the Bot + +**Development (Socket Mode):** +```bash +python app.py +``` + +**Production (HTTP Mode):** +```bash +# Set up a public endpoint (e.g., using ngrok for testing) +ngrok http 3000 + +# Update your Slack app's Event Subscriptions URL to: +# https://your-ngrok-url.ngrok.io/slack/events + +# Run the bot +python app.py +``` + +## Usage + +### In Slack + +1. **Invite the bot** to a channel: `/invite @DatabricksBot` + +2. **Send a message** in the channel or thread + +3. **First-time authentication**: + - The bot will send you a private (ephemeral) message with a "Sign in to Databricks" button + - Click the button and complete OAuth flow + - Return to Slack and send your message again + +4. **Ask questions**: + ``` + What is the revenue for Q4? + Show me the top 5 customers + Analyze the sales trends + ``` + +5. **Commands**: + - `logout` - Sign out from Databricks + - `clear` - Clear conversation history for the current thread + +### Message Flow + +1. User sends message → Bot checks authentication +2. If not authenticated → Show ephemeral auth prompt +3. If authenticated → Call Databricks agent with conversation history +4. Agent response → Post in thread with formatting +5. Tool calls → Display as rich Slack blocks + +## Key Differences from Teams Bot + +| Feature | Teams Bot | Slack Bot | +|---------|-----------|-----------| +| **Framework** | Bot Framework SDK | Slack Bolt | +| **Authentication** | OAuthPrompt built-in | Custom OAuth handler | +| **State Storage** | Bot Framework state | Custom StateManager | +| **Rich UI** | Adaptive Cards | Slack Blocks | +| **Private Messages** | Direct messages | Ephemeral messages | +| **Threading** | Reply to message | Thread timestamp | +| **Event Handling** | ActivityHandler | Event decorators | + +## Production Considerations + +### State Storage + +Replace in-memory storage with persistent storage: + +**Redis:** +```python +# In storage/state_manager.py +import redis +r = redis.from_url(os.environ.get("REDIS_URL")) +``` + +**DynamoDB:** +```python +import boto3 +dynamodb = boto3.resource('dynamodb') +table = dynamodb.Table('slack-bot-state') +``` + +### Security + +1. **Token Encryption**: Encrypt OAuth tokens at rest +2. **Secret Management**: Use AWS Secrets Manager or similar +3. **Request Validation**: Verify Slack request signatures +4. **Rate Limiting**: Implement rate limiting for API calls +5. **Audit Logging**: Log all user interactions and API calls + +### Scalability + +1. **Horizontal Scaling**: Use HTTP mode with load balancer +2. **Background Jobs**: Use Celery/SQS for long-running tasks +3. **Caching**: Cache frequently accessed data +4. **Database**: Use PostgreSQL/MongoDB for persistence + +### Monitoring + +1. **Logging**: Structured logging with correlation IDs +2. **Metrics**: Track response times, error rates, usage +3. **Alerts**: Set up alerts for errors and anomalies +4. **Tracing**: Implement distributed tracing + +## Troubleshooting + +### Bot doesn't respond + +1. Check bot is in the channel: `/invite @DatabricksBot` +2. Verify event subscriptions are configured +3. Check bot token is valid +4. Review logs for errors + +### Authentication fails + +1. Verify Databricks OAuth credentials +2. Check redirect URI matches configuration +3. Ensure user has Databricks workspace access + +### Messages lost + +1. Implement persistent state storage (Redis/DynamoDB) +2. Add retry logic for failed API calls +3. Use message queues for reliability + +## Development + +### Project Structure + +- `app.py` - Main Slack app with event handlers +- `config.py` - Environment configuration +- `handlers/` - Event handling logic + - `message_handler.py` - Process messages and agent calls + - `auth_handler.py` - OAuth flow management +- `storage/` - State management + - `state_manager.py` - Conversation and user state +- `client/` - External integrations + - `databricks_client.py` - Databricks API client + +### Testing + +```bash +# Run with test credentials +DATABRICKS_HOST=https://test.databricks.com python app.py + +# Use Slack's test features +# Create a test workspace and app +``` + +## License + +Copyright (c) Microsoft Corporation. All rights reserved. +Licensed under the MIT License. diff --git a/slack-bot/RUN.md b/slack-bot/RUN.md new file mode 100644 index 0000000..84aee2e --- /dev/null +++ b/slack-bot/RUN.md @@ -0,0 +1,112 @@ +# Running the Slack Bot with uv + +## Quick Start with uv + +### 1. Install dependencies (one-time setup) + +```bash +uv pip install -r requirements.txt +``` + +### 2. Run the bot + +```bash +python app.py +``` + +### 3. Alternative: One-liner (uv handles everything) + +```bash +# uv will handle dependencies automatically +uv run --with slack-bolt --with slack-sdk --with Flask --with databricks-sdk --with httpx --with python-dotenv python app.py +``` + +## Setup Environment Variables + +Before running, create a `.env` or `.env.local` file: + +```bash +# Option 1: Use .env.local (recommended for local development) +cp .env.example .env.local +# Edit .env.local with your credentials + +# Option 2: Use .env +cp .env.example .env +# Edit .env with your credentials +``` + +The app will automatically load from `.env.local` first, then fall back to `.env`. + +Required variables: +- `SLACK_BOT_TOKEN` - Your Slack bot token +- `SLACK_SIGNING_SECRET` - Your Slack signing secret +- `DATABRICKS_HOST` - Your Databricks workspace URL +- `SERVING_ENDPOINT_NAME` - Your agent endpoint name + +## Development Workflow + +### Install with dev dependencies + +```bash +uv sync --extra dev +``` + +### Run tests (when added) + +```bash +uv run pytest +``` + +### Format code + +```bash +uv run black . +uv run ruff check . +``` + +## Why uv? + +- **Fast**: 10-100x faster than pip +- **Reliable**: Deterministic dependency resolution +- **Simple**: No need for venv management +- **Modern**: Built in Rust, replaces pip, pip-tools, and virtualenv + +## Troubleshooting + +### "uv: command not found" + +Install uv: +```bash +curl -LsSf https://astral.sh/uv/install.sh | sh +``` + +Or with pip: +```bash +pip install uv +``` + +### Dependencies not found + +Make sure you're in the slack-bot directory: +```bash +cd slack-bot +uv sync +``` + +### Port already in use + +Change the port in `.env`: +```bash +PORT=3001 +``` + +## Alternative: Traditional venv + +If you prefer traditional venv: + +```bash +python -m venv venv +source venv/bin/activate # On Windows: venv\Scripts\activate +pip install -r requirements.txt +python app.py +``` diff --git a/slack-bot/app.py b/slack-bot/app.py new file mode 100644 index 0000000..e84e437 --- /dev/null +++ b/slack-bot/app.py @@ -0,0 +1,350 @@ +#!/usr/bin/env python3 +""" +Slack bot for Databricks Agent integration. + +This bot provides a Slack interface to Databricks agents, handling: +- OAuth authentication with Databricks +- Message handling in channels and threads +- Conversation history management +- Tool call visualization +""" + +import logging +import sys +import os +from pathlib import Path +from dotenv import load_dotenv +from slack_bolt import App +from slack_bolt.adapter.socket_mode import SocketModeHandler +from flask import Flask, request, redirect + +# Load environment variables from .env.local or .env +env_file = Path(__file__).parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) + print(f"Loaded environment from: {env_file}") +else: + env_file = Path(__file__).parent / ".env" + if env_file.exists(): + load_dotenv(env_file) + print(f"Loaded environment from: {env_file}") + else: + print("No .env or .env.local file found, using system environment variables") + +from config import DefaultConfig +from storage import StateManager +from handlers import AuthHandler, handle_message_event +from client import DatabricksClient + +# Configure logging +logging.basicConfig( + level=logging.INFO, + format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', + handlers=[ + logging.StreamHandler(sys.stdout) + ] +) +logger = logging.getLogger(__name__) + +# Load configuration +CONFIG = DefaultConfig() + +# Validate required configuration +required_configs = [ + "SLACK_BOT_TOKEN", + "SLACK_SIGNING_SECRET", + "DATABRICKS_HOST", + "SERVING_ENDPOINT_NAME" +] +missing_configs = [c for c in required_configs if not getattr(CONFIG, c)] +if missing_configs: + logger.error(f"Missing required configuration: {', '.join(missing_configs)}") + sys.exit(1) + +# Initialize Slack app +app = App( + token=CONFIG.SLACK_BOT_TOKEN, + signing_secret=CONFIG.SLACK_SIGNING_SECRET +) + +# Initialize components +state_manager = StateManager() +auth_handler = AuthHandler(CONFIG, state_manager) +databricks_client = DatabricksClient( + CONFIG.DATABRICKS_HOST, + CONFIG.SERVING_ENDPOINT_NAME +) + +logger.info("Slack bot initialized successfully") + + +@app.event("message") +async def message_handler(message, say, client): + """ + Handle incoming message events. + This is the main entry point for user messages. + """ + await handle_message_event( + message=message, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.event("app_mention") +async def handle_app_mention(event, say, client): + """ + Handle @mentions of the bot. + Treat as regular message. + """ + await handle_message_event( + message=event, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.command("/databricks-logout") +async def handle_logout_command(ack, command, respond): + """Handle /databricks-logout slash command""" + await ack() + user_id = command["user_id"] + auth_handler.handle_logout(user_id) + await respond("You have been logged out from Databricks.") + + +@app.command("/databricks-clear") +async def handle_clear_command(ack, command, respond): + """Handle /databricks-clear slash command to clear conversation history""" + await ack() + channel_id = command["channel_id"] + # For commands, we don't have a specific thread, so clear all history + # In production, you might want to handle this differently + await respond("Conversation history management via slash command is not fully implemented. Use 'clear' in a message instead.") + + +@app.event("app_home_opened") +async def handle_app_home_opened(event, client): + """ + Handle app home opened event. + Show welcome message and authentication status. + """ + user_id = event["user"] + is_authenticated = auth_handler.is_authenticated(user_id) + + blocks = [ + { + "type": "header", + "text": { + "type": "plain_text", + "text": "Welcome to Databricks Agent Bot" + } + }, + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": "*What I can do:*\n• Answer questions using Databricks agents\n• Execute tool calls and show results\n• Maintain conversation context in threads" + } + }, + { + "type": "divider" + } + ] + + if is_authenticated: + blocks.append({ + "type": "section", + "text": { + "type": "mrkdwn", + "text": "āœ… *Status:* Authenticated\n\nYou can start chatting with me in any channel where I'm added!" + } + }) + else: + auth_message = auth_handler.create_auth_message(user_id, "app_home") + blocks.extend(auth_message["blocks"]) + + try: + client.views_publish( + user_id=user_id, + view={ + "type": "home", + "blocks": blocks + } + ) + except Exception as e: + logger.error(f"Error publishing home view: {e}") + + +# Custom route for OAuth callback +flask_app = Flask(__name__) + + +@flask_app.route("/oauth/callback", methods=["GET"]) +async def oauth_callback(): + """ + Handle OAuth callback from Databricks. + Exchanges authorization code for access token. + """ + try: + # Get authorization code and state from query parameters + code = request.args.get("code") + state = request.args.get("state") + error = request.args.get("error") + + if error: + logger.error(f"OAuth error: {error}") + return f""" + + +

āŒ Authentication Failed

+

Error: {error}

+

Please close this window and try again in Slack.

+ + + """, 400 + + if not code or not state: + logger.error("Missing code or state in OAuth callback") + return """ + + +

āŒ Invalid Request

+

Missing required parameters.

+

Please close this window and try again in Slack.

+ + + """, 400 + + # Parse state to get user_id + try: + user_id, channel_id = state.split(":") + except ValueError: + logger.error(f"Invalid state format: {state}") + return """ + + +

āŒ Invalid State

+

Please close this window and try again in Slack.

+ + + """, 400 + + # Validate state + if not auth_handler.validate_state(state, user_id): + logger.error(f"State validation failed for user {user_id}") + return """ + + +

āŒ Security Validation Failed

+

Please close this window and try again in Slack.

+ + + """, 400 + + # Exchange code for token + access_token, expires_in = await auth_handler.exchange_code_for_token(code, user_id) + + if not access_token: + logger.error(f"Failed to exchange code for token for user {user_id}") + return """ + + +

āŒ Token Exchange Failed

+

Unable to complete authentication.

+

Please close this window and try again in Slack.

+ + + """, 500 + + # Store token + state_manager.store_user_token(user_id, access_token, expires_in) + + logger.info(f"Successfully authenticated user {user_id}") + + # Send success message to Slack + try: + from slack_sdk import WebClient + slack_client = WebClient(token=CONFIG.SLACK_BOT_TOKEN) + slack_client.chat_postMessage( + channel=channel_id, + text=f"āœ… <@{user_id}> You're now authenticated! Send me a message to get started.", + ) + except Exception as e: + logger.error(f"Failed to send success message to Slack: {e}") + + # Return success page + return """ + + +

āœ… Authentication Successful!

+

You're now connected to Databricks.

+

You can close this window and return to Slack.

+ + + + """, 200 + + except Exception as e: + logger.error(f"Error in OAuth callback: {e}") + import traceback + traceback.print_exc() + return """ + + +

āŒ Error

+

An unexpected error occurred.

+

Please close this window and try again in Slack.

+ + + """, 500 + + +@app.error +async def custom_error_handler(error, body, logger): + """Global error handler""" + logger.error(f"Error: {error}") + logger.error(f"Request body: {body}") + + +def main(): + """Main entry point""" + try: + # HTTP mode with Flask for OAuth callback support + logger.info(f"Starting bot in HTTP mode on port {CONFIG.PORT}...") + logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") + + from slack_bolt.adapter.flask import SlackRequestHandler + handler = SlackRequestHandler(app) + + # Register Slack events endpoint + @flask_app.route("/slack/events", methods=["POST"]) + def slack_events(): + return handler.handle(request) + + # Start Flask server + flask_app.run(port=CONFIG.PORT, debug=False) + + except KeyboardInterrupt: + logger.info("Bot stopped by user") + except Exception as e: + logger.error(f"Error starting bot: {e}") + import traceback + traceback.print_exc() + sys.exit(1) + + +if __name__ == "__main__": + main() diff --git a/slack-bot/client/__init__.py b/slack-bot/client/__init__.py new file mode 100644 index 0000000..88eb439 --- /dev/null +++ b/slack-bot/client/__init__.py @@ -0,0 +1,3 @@ +from .databricks_client import DatabricksClient + +__all__ = ["DatabricksClient"] diff --git a/slack-bot/client/databricks_client.py b/slack-bot/client/databricks_client.py new file mode 100644 index 0000000..8bf55d2 --- /dev/null +++ b/slack-bot/client/databricks_client.py @@ -0,0 +1,199 @@ +# Copyright Ā© Databricks, Inc. All rights reserved. +# Licensed under the MIT License. + +import logging +import uuid + +import httpx +from databricks.sdk import WorkspaceClient +from databricks_ai_bridge.genie import Genie + +class DatabricksClient: + def __init__(self, databricks_host: str, request_timeout: float = 300): + self.databricks_host = databricks_host + self.client = httpx.AsyncClient( + timeout=httpx.Timeout(request_timeout), + ) + + async def exchange_token(self, provider_oauth_token: str): + + url = f"{self.databricks_host}/oidc/v1/token" + + data = { + "subject_token": provider_oauth_token, # replace with your JWT token variable + "subject_token_type": "urn:ietf:params:oauth:token-type:jwt", + "grant_type": "urn:ietf:params:oauth:grant-type:token-exchange", + "scope": "all-apis" + } + + response = await self.client.post(url, data=data) + + return response.json()['access_token'] + + def _throw_unexpected_endpoint_format(self): + raise Exception("This app can only run against ChatModel, ChatAgent, or ResponsesAgent endpoints") + + def _convert_to_responses_format(self, messages): + """Convert chat messages to ResponsesAgent API format.""" + input_messages = [] + for msg in messages: + if msg["role"] == "user": + input_messages.append({"role": "user", "content": msg["content"]}) + elif msg["role"] == "assistant": + # Handle assistant messages with tool calls + if msg.get("tool_calls"): + # Add function calls + for tool_call in msg["tool_calls"]: + input_messages.append({ + "type": "function_call", + "id": tool_call["id"], + "call_id": tool_call["id"], + "name": tool_call["function"]["name"], + "arguments": tool_call["function"]["arguments"] + }) + # Add assistant message if it has content + if msg.get("content"): + input_messages.append({ + "type": "message", + "id": msg.get("id", str(uuid.uuid4())), + "content": [{"type": "output_text", "text": msg["content"]}], + "role": "assistant" + }) + else: + # Regular assistant message + input_messages.append({ + "type": "message", + "id": msg.get("id", str(uuid.uuid4())), + "content": [{"type": "output_text", "text": msg["content"]}], + "role": "assistant" + }) + elif msg["role"] == "tool": + input_messages.append({ + "type": "function_call_output", + "call_id": msg.get("tool_call_id"), + "output": msg["content"] + }) + return input_messages + + def _parse_responses_output(self, response): + result_messages = [] + + logging.info(f"Response from openai client: {response}") + + for item in response.output: + logging.info(f"Item type: {item.type}") + if item.type == "message": + content = "".join([e.text for e in item.content if e.type == "output_text"]) + if content: + result_messages.append({"role": "assistant", "content": content}) + elif item.type == "function_call": + tool_calls = [{"id": item.call_id, + "type": "function", + "function": {"name": item.name, + "arguments": item.arguments}}] + + result_messages.append({"role": "assistant", "content": "", "tool_calls": tool_calls}) + elif item.type == "function_call_output": + result_messages.append({"role": "tool", "content": item.output, "tool_call_id": item.call_id}) + + logging.info(f"Response parsed from openai client: {result_messages}") + return result_messages + + def _parse_chat_response(self, response): + + logging.info(f"Response from openai client: {response}") + + result_messages = [] + if hasattr(response, "messages") and response.messages: + result_messages.extend(response.messages) + elif hasattr(response, "choices") and response.choices: + choice_message = response.choices[0].message + message_content = choice_message.content + if isinstance(choice_message.content, list): + message_content = "".join( + [part.get("text", "") for part in choice_message.content if part.get("type") == "text"]) + message = {"role": "assistant", "content": message_content} + if choice_message.tool_calls: + message["tool_calls"] = choice_message.tool_calls + result_messages.append(message) + return result_messages + + def _get_endpoint_task_type(self, workspace_client, endpoint_name: str) -> str: + """Get the task type of a serving endpoint.""" + try: + ep = workspace_client.serving_endpoints.get(endpoint_name) + return ep.task if ep.task else "chat/completions" + except Exception: + return "chat/completions" + + def _query_responses_endpoint(self, + workspace_client, + messages: list, + serving_endpoint_name: str) -> list: + + input_messages = self._convert_to_responses_format(messages) + + openai_client = workspace_client.serving_endpoints.get_open_ai_client() + + response = openai_client.responses.create(model=serving_endpoint_name, input=input_messages) + + result_messages = self._parse_responses_output(response) + + if not result_messages: + self._throw_unexpected_endpoint_format() + + return result_messages + + def _query_chat_endpoint(self, workspace_client, messages: list, serving_endpoint_name: str) -> list: + """Calls a model serving endpoint with chat/completions format.""" + + openai_client = workspace_client.serving_endpoints.get_open_ai_client() + + res = openai_client.chat.completions.create(model=serving_endpoint_name, messages=messages) + + result_messages = self._parse_chat_response(res) + + if not result_messages: + self._throw_unexpected_endpoint_format() + + return result_messages + + async def call_model_endpoint(self, + serving_endpoint_name: str, + text:str, + provider_oauth_token: str, + history:list): + + oauth_db_token = await self.exchange_token(provider_oauth_token) + + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + + task_type = self._get_endpoint_task_type(workspace_client, serving_endpoint_name) + + logging.info(f"Serving endpoint task type: {task_type}") + + logging.info(f"Actual history in the chatbot: {history}") + + messages = history + [{"role": "user", "content": text}] + + if task_type == "agent/v1/responses": + result_messages = self._query_responses_endpoint(workspace_client, messages, serving_endpoint_name) + else: + result_messages = self._query_chat_endpoint(workspace_client, messages, serving_endpoint_name) + + return result_messages + + async def call_genie_space(self, question: str, + provider_oauth_token:str, + conversation_id: str, + genie_space_id: str): + + oauth_db_token = await self.exchange_token(provider_oauth_token) + + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + + genie = Genie(genie_space_id, workspace_client) + + genie_result = genie.ask_question(question, conversation_id) + + return genie_result.result diff --git a/slack-bot/config.py b/slack-bot/config.py new file mode 100644 index 0000000..6db270c --- /dev/null +++ b/slack-bot/config.py @@ -0,0 +1,32 @@ +#!/usr/bin/env python3 +"""Slack Bot Configuration""" + +import os + + +class DefaultConfig: + """Slack Bot Configuration""" + + # Server settings + PORT = int(os.environ.get("PORT", "3000")) + + # Slack app credentials + SLACK_BOT_TOKEN = os.environ.get("SLACK_BOT_TOKEN", "") + SLACK_SIGNING_SECRET = os.environ.get("SLACK_SIGNING_SECRET", "") + SLACK_CLIENT_ID = os.environ.get("SLACK_CLIENT_ID", "") + SLACK_CLIENT_SECRET = os.environ.get("SLACK_CLIENT_SECRET", "") + + # OAuth settings for Databricks + DATABRICKS_HOST = os.environ.get("DATABRICKS_HOST", "") + DATABRICKS_CLIENT_ID = os.environ.get("DATABRICKS_CLIENT_ID", "") + DATABRICKS_CLIENT_SECRET = os.environ.get("DATABRICKS_CLIENT_SECRET", "") + + # Databricks serving endpoint + SERVING_ENDPOINT_NAME = os.environ.get("SERVING_ENDPOINT_NAME", "") + + # State storage (for production, use Redis or DynamoDB) + STATE_STORAGE = os.environ.get("STATE_STORAGE", "memory") # memory, redis, dynamodb + REDIS_URL = os.environ.get("REDIS_URL", "redis://localhost:6379") + + # OAuth callback URI (defaults to localhost for development) + OAUTH_REDIRECT_URI = os.environ.get("OAUTH_REDIRECT_URI", "") diff --git a/slack-bot/hackathon.py b/slack-bot/hackathon.py new file mode 100644 index 0000000..853936d --- /dev/null +++ b/slack-bot/hackathon.py @@ -0,0 +1,275 @@ +from typing import Callable, Dict, List, Literal, Optional, Union +import os +from dataclasses import dataclass + + +import autogen +from autogen import ConversableAgent, AssistantAgent, register_function +from typing_extensions import Annotated + + +from slack_bolt import App +from slack_bolt.adapter.socket_mode import SocketModeHandler + + +from agent import GenieAgent + + + + + +# llm={ +# "config_list": [ +# { +# "model": "databricks-meta-llama-3-1-70b-instruct", +# "api_key": os.environ.get("DATABRICKS_TOKEN"), +# "base_url": "https://e2-dogfood.staging.cloud.databricks.com/serving-endpoints" +# } +# ] +# } + + +llm={ + "config_list": [ + { + "model": "gpt-4o", + "api_key": os.environ.get("OPENAI_API_KEY"), + "temperature": 0.0 + } + ] +} + + +def ask_database(query: Annotated[str, "Natural language query to the database."]): + # ask genie + return None + + + + +class SlackUserProxyAgent(autogen.UserProxyAgent): + def __init__( + self, + name: str, + is_termination_msg: Optional[Callable[[Dict], bool]] = None, + max_consecutive_auto_reply: Optional[int] = None, + human_input_mode: Literal["ALWAYS", "TERMINATE", "NEVER"] = "ALWAYS", + function_map: Optional[Dict[str, Callable]] = None, + code_execution_config: Union[Dict, Literal[False]] = {}, + default_auto_reply: Optional[Union[str, Dict, None]] = "", + llm_config: Optional[Union[Dict, Literal[False]]] = False, + system_message: Optional[Union[str, List]] = "", + description: Optional[str] = None, + **kwargs, + ): + super().__init__( + name=name, + is_termination_msg=is_termination_msg, + max_consecutive_auto_reply=max_consecutive_auto_reply, + human_input_mode=human_input_mode, + function_map=function_map, + code_execution_config=code_execution_config, + default_auto_reply=default_auto_reply, + llm_config=llm_config, + system_message=system_message, + description=description, + **kwargs, + ) + + + def get_human_input(self, prmopt) -> str: + return "" + + + + +app = App(token=slack_bot_token) + + + + +@app.event("app_mention") +def handle_app_mention_events(body, logger): + print(f"app_mention: {body}") + + + + +@app.event("message") +def handle_message_events(message, say): + print("") + print(f">>> message={message}") + # print(f"{message['user']}: {message['text']}") + # say(f"Hey there <@{message['user']}>!") + + + # get bot user id + slack_bot_user_id = "xx" # app.client.test_auth()['user_id'] + + + # get histry of the channel + messages = get_history(message['channel']) + print("") + print(f">>> history={messages}") + newMessage = messages[-1] + messages = messages[:-1] + + + # create user proxyies + users = map(lambda x: x['name'], messages) + unique_users = set(users) + unique_users.discard(slack_bot_user_id) + + + user_proxy_agents = {} + for slack_user_id in unique_users: + user_proxy_agents[slack_user_id] = SlackUserProxyAgent( + name=slack_user_id, + system_message="", + code_execution_config=False, + human_input_mode="TERMINATE", # https://microsoft.github.io/autogen/docs/reference/agentchat/user_proxy_agent + #default_auto_reply="exit", + is_termination_msg=lambda x: True, + ) + + + # create agents + # joke = ConversableAgent( + # "joke", + # system_message="You can come up with very nice jokes.", + # llm_config=llm, + # code_execution_config=False, # Turn off code execution, by default it is off. + # function_map=None, # No registered functions, by default it is None. + # human_input_mode="NEVER", + # ) + + + # judge = ConversableAgent( + # "judge", + # system_message="Rate the joke based on creativity, humor, and originality.", + # llm_config=llm, + # code_execution_config=False, # Turn off code execution, by default it is off. + # function_map=None, # No registered functions, by default it is None. + # human_input_mode="NEVER", + # ) + + + # stale_data_checker = ConversableAgent( + # "check", + # system_message="Check table is older than 1 year, then let user know the name of the table if it is stale or not by tagging a user <@{U0776G4GQR5}>", + # llm_config=llm, + # code_execution_config=False, # Turn off code execution, by default it is off. + # function_map=None, # No registered functions, by default it is None. + # human_input_mode="NEVER", + # ) + + + table = GenieAgent( + space_id='01eeeaf13f6e1b0e91da8fc41228330b', + name="Agent_Table", + description="I can query information schema to find detailed info about tables.") + + data_polciy = ConversableAgent( + "Agent_Data_Policy", + description="Answer questions about data policy.", + system_message= + "You are an assistant that understands data policies but nothing else. " + "Please only provide relevant policy entries when asked. " + "Below are the data policies defined:\n " + "* A data table that was last modified half a year ago is considered stale." + "* A data table that contains more than 20 columns is considered complex.", + llm_config=llm, + code_execution_config=False, # Turn off code execution, by default it is off. + function_map=None, # No registered functions, by default it is None. + human_input_mode="NEVER", + ) + + + moon = UserProxyAgent( + name="User_Moonsoo", + description="A data engineer at Databricks.", + code_execution_config=False) + + + # Create planner agent. + planner = AssistantAgent( + name="Agent_Planner", + llm_config=llm, + description="Creates a plan for a complex task that might need more than one agents.", + system_message="You are a helpful AI assistant. You suggest a feasible plan " + "for finishing a complex task by decomposing it into sub-tasks. " + "Keep sub-task descriptions very concise.", + code_execution_config=False, + ) + + + moon = UserProxyAgent( + name="User_Moonsoo", + description="A data engineer at Databricks.", + code_execution_config=False) + + + groupchat = autogen.GroupChat(agents=list(user_proxy_agents.values()) + [table, data_polciy, moon], messages=[], max_round=10) + manager = autogen.GroupChatManager(groupchat, + system_message="You are moderating a channel with a few agents and users. " + "For a request that is relevant to more than one agent/user, " + f"please ask the {planner.name} agent to create a plan first.", + llm_config=llm) + + + + user_proxy_agents[newMessage['name']].initiate_chat( + manager, message=newMessage['content'] + ) + + + print("") + print(f">>> groupchat.messages={groupchat.messages}") + agent_messages = groupchat.messages[1:] + for msg in agent_messages: + icon = ":robot_face:" + if msg['name'] == 'Table': + icon = ":magic_wand:" + + + say( + text = f"{msg['content']}", + icon_emoji = icon, + username = msg['name'] # https://github.com/slackapi/bolt-python/blob/main/slack_bolt/context/say/say.py#L61, https://api.slack.com/methods/chat.postMessage + ) + + + + + + +def get_history(channel_id): + result = app.client.conversations_history(channel=channel_id) + # print(f">>> history_result = {result}") + history = result['messages'] + if history == None: + history = [] + + + messages = [] + for msg in reversed(history): + # print(f"msg={msg['text']}, user={msg['user']}") + if 'user' not in msg: + print(f"Skipping message without user: {msg}") + continue + + + messages.append({ + "role": "user", + "name": msg['user'], + "content": msg['text'] + }) + + + return messages + + +if __name__ == "__main__": + SocketModeHandler(app, slack_app_token).start() + + diff --git a/slack-bot/handlers/__init__.py b/slack-bot/handlers/__init__.py new file mode 100644 index 0000000..9e7afca --- /dev/null +++ b/slack-bot/handlers/__init__.py @@ -0,0 +1,4 @@ +from .message_handler import handle_message_event +from .auth_handler import AuthHandler + +__all__ = ["handle_message_event", "AuthHandler"] diff --git a/slack-bot/handlers/auth_handler.py b/slack-bot/handlers/auth_handler.py new file mode 100644 index 0000000..4ce6f23 --- /dev/null +++ b/slack-bot/handlers/auth_handler.py @@ -0,0 +1,166 @@ +"""OAuth authentication handler for Databricks""" + +import logging +import httpx +import hashlib +import base64 +import secrets +from typing import Dict, Optional, Tuple +from urllib.parse import urlencode + + +class AuthHandler: + """Handles OAuth flow for Databricks authentication with PKCE""" + + def __init__(self, config, state_manager): + self.config = config + self.state_manager = state_manager + + def _generate_pkce_pair(self) -> Tuple[str, str]: + """ + Generate PKCE code verifier and code challenge. + Following RFC 7636 for OAuth PKCE. + """ + # Generate code verifier (43-128 characters) + code_verifier = base64.urlsafe_b64encode(secrets.token_bytes(32)).decode('utf-8').rstrip('=') + + # Generate code challenge using S256 method + code_challenge = base64.urlsafe_b64encode( + hashlib.sha256(code_verifier.encode('utf-8')).digest() + ).decode('utf-8').rstrip('=') + + return code_verifier, code_challenge + + def get_auth_url(self, user_id: str, channel_id: str) -> str: + """ + Generate OAuth URL for Databricks authentication with PKCE. + Based on: https://docs.databricks.com/en/dev-tools/auth/oauth-m2m.html + """ + # Generate PKCE pair + code_verifier, code_challenge = self._generate_pkce_pair() + + # Store state and code verifier for callback validation + state = f"{user_id}:{channel_id}" + self.state_manager.set_user_data(user_id, "oauth_state", state) + self.state_manager.set_user_data(user_id, "code_verifier", code_verifier) + + # Build OAuth URL with PKCE + # Use databricks-cli as client_id if no custom client_id is configured + client_id = self.config.DATABRICKS_CLIENT_ID or "databricks-cli" + + oauth_url = ( + f"{self.config.DATABRICKS_HOST}/oidc/v1/authorize" + f"?client_id={client_id}" + f"&redirect_uri={self._get_redirect_uri()}" + f"&response_type=code" + f"&state={state}" + f"&code_challenge={code_challenge}" + f"&code_challenge_method=S256" + f"&scope=all-apis+offline_access" + ) + + return oauth_url + + def _get_redirect_uri(self) -> str: + """Get OAuth redirect URI""" + # For local development, use localhost + # For production, set OAUTH_REDIRECT_URI environment variable + redirect_uri = getattr(self.config, 'OAUTH_REDIRECT_URI', None) + if redirect_uri: + return redirect_uri + return f"http://localhost:{self.config.PORT}/oauth/callback" + + def create_auth_message(self, user_id: str, channel_id: str) -> Dict: + """Create Slack message with authentication button""" + auth_url = self.get_auth_url(user_id, channel_id) + + return { + "text": "Please authenticate with Databricks to continue", + "blocks": [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": "šŸ” *Authentication Required*\n\nPlease sign in to Databricks to use this bot." + } + }, + { + "type": "actions", + "elements": [ + { + "type": "button", + "text": { + "type": "plain_text", + "text": "Sign in to Databricks" + }, + "url": auth_url, + "action_id": "databricks_auth" + } + ] + } + ] + } + + def is_authenticated(self, user_id: str) -> bool: + """Check if user is authenticated""" + token = self.state_manager.get_user_token(user_id) + return token is not None + + def handle_logout(self, user_id: str): + """Handle user logout""" + self.state_manager.delete_user_token(user_id) + logging.info(f"User {user_id} logged out") + + async def exchange_code_for_token(self, code: str, user_id: str) -> Tuple[Optional[str], Optional[int]]: + """ + Exchange authorization code for access token using PKCE. + Implements OAuth 2.0 authorization code grant flow with PKCE. + """ + token_url = f"{self.config.DATABRICKS_HOST}/oidc/v1/token" + + # Retrieve code verifier from state + code_verifier = self.state_manager.get_user_data(user_id, "code_verifier") + if not code_verifier: + logging.error(f"Code verifier not found for user {user_id}") + return None, None + + # Use databricks-cli as client_id if no custom client_id is configured + client_id = self.config.DATABRICKS_CLIENT_ID or "databricks-cli" + + data = { + "grant_type": "authorization_code", + "code": code, + "redirect_uri": self._get_redirect_uri(), + "client_id": client_id, + "code_verifier": code_verifier, + } + + # Only include client_secret if using a custom OAuth app + if self.config.DATABRICKS_CLIENT_SECRET: + data["client_secret"] = self.config.DATABRICKS_CLIENT_SECRET + + try: + async with httpx.AsyncClient() as client: + response = await client.post(token_url, data=data) + response.raise_for_status() + token_data = response.json() + + access_token = token_data.get("access_token") + expires_in = token_data.get("expires_in", 3600) + + # Clean up code verifier after use + self.state_manager.set_user_data(user_id, "code_verifier", None) + + logging.info("Successfully exchanged code for token") + return access_token, expires_in + + except Exception as e: + logging.error(f"Error exchanging code for token: {e}") + if hasattr(e, 'response'): + logging.error(f"Response: {e.response.text if hasattr(e.response, 'text') else e.response}") + return None, None + + def validate_state(self, state: str, user_id: str) -> bool: + """Validate OAuth state parameter to prevent CSRF attacks""" + stored_state = self.state_manager.get_user_data(user_id, "oauth_state") + return stored_state == state diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py new file mode 100644 index 0000000..ea82f97 --- /dev/null +++ b/slack-bot/handlers/message_handler.py @@ -0,0 +1,190 @@ +"""Message event handler for Slack bot""" + +import logging +from typing import Dict, List +from slack_sdk import WebClient + + +async def handle_message_event( + message: Dict, + say, + client: WebClient, + state_manager, + auth_handler, + databricks_client, + logger +): + """ + Handle incoming Slack message events. + Responds in thread with ephemeral messages when appropriate. + """ + try: + # Extract message details + user_id = message.get("user") + text = message.get("text", "").strip() + channel_id = message.get("channel") + thread_ts = message.get("thread_ts") or message.get("ts") + + # Ignore bot messages and empty messages + if not user_id or not text or message.get("bot_id"): + return + + logger.info(f"Processing message from user {user_id}: {text}") + + # Handle special commands + if text.lower() == "logout": + await handle_logout(user_id, channel_id, thread_ts, say, auth_handler) + return + + if text.lower() == "clear": + await handle_clear_history(thread_ts, channel_id, user_id, say, state_manager) + return + + # Check authentication + if not auth_handler.is_authenticated(user_id): + logger.info(f"User {user_id} not authenticated") + auth_message = auth_handler.create_auth_message(user_id, channel_id) + + # Send ephemeral message (only visible to user) + client.chat_postEphemeral( + channel=channel_id, + user=user_id, + text=auth_message["text"], + blocks=auth_message.get("blocks") + ) + return + + # Get user's OAuth token + user_token = state_manager.get_user_token(user_id) + + # Get conversation history for this thread + history = state_manager.get_conversation_history(thread_ts) + + # Call Databricks agent + logger.info(f"Calling Databricks agent for user {user_id}") + response = await databricks_client.call_model_endpoint(text, user_token, history) + + # Process and send responses + await send_agent_responses( + response=response, + text=text, + channel_id=channel_id, + thread_ts=thread_ts, + user_id=user_id, + say=say, + client=client, + state_manager=state_manager + ) + + except Exception as e: + logger.error(f"Error handling message: {e}") + import traceback + traceback.print_exc() + + # Send error message as ephemeral + try: + client.chat_postEphemeral( + channel=channel_id, + user=user_id, + text="Sorry, an error occurred while processing your message. Please try again." + ) + except Exception: + pass + + +async def send_agent_responses( + response: List[Dict], + text: str, + channel_id: str, + thread_ts: str, + user_id: str, + say, + client: WebClient, + state_manager +): + """Send agent responses to Slack""" + # Add user message to history + state_manager.add_message_to_history(thread_ts, "user", text) + + tool_calls = {} + + for item in response: + if item["type"] == "message": + # Send message in thread (visible to everyone) + say( + text=item["text"], + thread_ts=thread_ts + ) + state_manager.add_message_to_history(thread_ts, "assistant", item["text"]) + + elif item["type"] == "tool_call": + # Store tool call for pairing with result + tool_calls[item["call_id"]] = item + + elif item["type"] == "tool_result": + # Pair with tool call and send as block + call_id = item["call_id"] + if call_id in tool_calls: + tool_call = tool_calls[call_id] + tool_call.update(item) + + # Create tool call card + tool_card = create_tool_call_block(tool_call) + + # Send in thread + say( + blocks=tool_card, + text=f"Tool: {tool_call['name']}", + thread_ts=thread_ts + ) + + +def create_tool_call_block(tool_call_info: Dict) -> List[Dict]: + """Create Slack blocks for tool call display""" + return [ + { + "type": "header", + "text": { + "type": "plain_text", + "text": "šŸ”§ Agent Tool Call" + } + }, + { + "type": "section", + "fields": [ + { + "type": "mrkdwn", + "text": f"*Tool:*\n{tool_call_info['name']}" + }, + { + "type": "mrkdwn", + "text": f"*Parameters:*\n```{tool_call_info['arguments']}```" + } + ] + }, + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": f"*Result:*\n```{tool_call_info['output']}```" + } + } + ] + + +async def handle_logout(user_id: str, channel_id: str, thread_ts: str, say, auth_handler): + """Handle logout command""" + auth_handler.handle_logout(user_id) + say( + text="You have been logged out successfully.", + thread_ts=thread_ts + ) + + +async def handle_clear_history(thread_ts: str, channel_id: str, user_id: str, say, state_manager): + """Handle clear history command""" + state_manager.clear_conversation_history(thread_ts) + say( + text="Conversation history cleared.", + thread_ts=thread_ts + ) diff --git a/slack-bot/pyproject.toml b/slack-bot/pyproject.toml new file mode 100644 index 0000000..3305ae6 --- /dev/null +++ b/slack-bot/pyproject.toml @@ -0,0 +1,29 @@ +[project] +name = "databricks-slack-bot" +version = "0.1.0" +description = "Slack bot for Databricks Agent integration" +requires-python = ">=3.8" +dependencies = [ + "slack-bolt>=1.18.0", + "slack-sdk>=3.23.0", + "Flask>=3.0.0", + "aiohttp>=3.9.0", + "databricks-sdk>=0.18.0", + "httpx>=0.25.0", + "python-dotenv>=1.0.0", +] + +[project.optional-dependencies] +dev = [ + "pytest>=7.0.0", + "black>=23.0.0", + "ruff>=0.1.0", +] + +[tool.black] +line-length = 100 +target-version = ['py38', 'py39', 'py310', 'py311'] + +[tool.ruff] +line-length = 100 +target-version = "py38" diff --git a/slack-bot/references/databricks-oauth-flow.md b/slack-bot/references/databricks-oauth-flow.md new file mode 100644 index 0000000..ba650d2 --- /dev/null +++ b/slack-bot/references/databricks-oauth-flow.md @@ -0,0 +1,182 @@ +# Databricks OAuth Authorization Flow + +This document contains the official Databricks documentation for implementing OAuth 2.0 authorization with PKCE. + +## Generate a workspace-level authorization code + +In your browser, navigate to the URL with the following replacements: + +- ``: Your Databricks workspace instance name, for example `dbc-a1b2345c-d6e7.cloud.databricks.com` +- ``: A local redirect (for example, `http://localhost:8020`) +- ``: Any plain-text value for response validation +- ``: The challenge string from Step 1 + +### Authorization URL Format + +``` +https:///oidc/v1/authorize +?client_id=databricks-cli +&redirect_uri= +&response_type=code +&state= +&code_challenge= +&code_challenge_method=S256 +&scope=all-apis+offline_access +``` + +### Steps + +1. **Navigate to the authorization URL** in your browser with the appropriate parameters + +2. **Log in when prompted** to access your Databricks account + +3. **Copy the authorization code** from your browser's address bar. It's the value after `code=` and before `&` in the redirect URL: + ``` + http://localhost:8020/?code=dcod...7fe6&state= + ``` + +4. **Verify that the state value** matches what you originally provided. If it doesn't, discard the code. + +5. **Continue to Generate a workspace-level access token** + +## Key Parameters Explained + +### client_id +- For development: Use `databricks-cli` (pre-configured in all workspaces) +- For production: Use your custom OAuth app's client ID + +### redirect_uri +- Must match exactly what's configured in your OAuth app +- For local development: `http://localhost:/oauth/callback` +- For production: `https://your-domain.com/oauth/callback` + +### response_type +- Always `code` for authorization code grant flow + +### state +- A random string to prevent CSRF attacks +- Must be verified in the callback +- Format: Any plain-text value (we use `user_id:channel_id`) + +### code_challenge +- SHA256 hash of the `code_verifier` +- Used in PKCE to prevent authorization code interception attacks +- Generated using: `base64urlsafe(sha256(code_verifier))` + +### code_challenge_method +- Always `S256` (SHA256 hashing) +- More secure than `plain` method + +### scope +- `all-apis`: Access to all Databricks APIs +- `offline_access`: Allows refresh tokens for long-lived access +- Multiple scopes separated by `+` or space + +## PKCE (Proof Key for Code Exchange) + +PKCE is an extension to the OAuth 2.0 authorization code flow that prevents authorization code interception attacks. + +### Code Verifier +- Random string, 43-128 characters +- Generated using: `base64urlsafe(random_bytes(32))` +- Must be stored securely until token exchange + +### Code Challenge +- Derived from code verifier +- Sent in authorization request +- Calculated as: `base64urlsafe(sha256(code_verifier))` + +### Why PKCE? +- Protects against authorization code interception +- Especially important for mobile apps and SPAs +- No need to expose client_secret in public clients + +## Token Exchange (Next Step) + +After receiving the authorization code, exchange it for an access token: + +```http +POST https:///oidc/v1/token +Content-Type: application/x-www-form-urlencoded + +grant_type=authorization_code +&code= +&redirect_uri= +&client_id=databricks-cli +&code_verifier= +``` + +If using a custom OAuth app with client secret: +```http +POST https:///oidc/v1/token +Content-Type: application/x-www-form-urlencoded + +grant_type=authorization_code +&code= +&redirect_uri= +&client_id= +&client_secret= +&code_verifier= +``` + +### Response + +```json +{ + "access_token": "dapi...", + "token_type": "Bearer", + "expires_in": 3600, + "refresh_token": "...", + "scope": "all-apis offline_access" +} +``` + +## Security Best Practices + +1. **Always validate the state parameter** in the callback +2. **Store code_verifier securely** until token exchange +3. **Use HTTPS in production** for all OAuth endpoints +4. **Never log or expose** access tokens or refresh tokens +5. **Implement token refresh logic** before expiration +6. **Use short-lived access tokens** with refresh tokens +7. **Store tokens encrypted at rest** +8. **Implement proper session management** + +## Common Errors + +### "Redirect URI mismatch" +- The redirect_uri in request doesn't match OAuth app configuration +- Solution: Update OAuth app settings or fix redirect_uri parameter + +### "Invalid client_id" +- Client ID not recognized +- Solution: Check client_id spelling or create OAuth app + +### "Invalid code_challenge" +- Code challenge format incorrect +- Solution: Ensure proper base64url encoding and SHA256 hashing + +### "Code has expired" +- Authorization codes typically expire in 10 minutes +- Solution: Complete token exchange quickly after receiving code + +### "Invalid code_verifier" +- code_verifier doesn't match code_challenge +- Solution: Ensure same code_verifier used for authorization and token exchange + +## References + +- [RFC 7636 - Proof Key for Code Exchange](https://tools.ietf.org/html/rfc7636) +- [OAuth 2.0 Authorization Framework](https://tools.ietf.org/html/rfc6749) +- [Databricks OAuth Documentation](https://docs.databricks.com/en/dev-tools/auth/oauth-m2m.html) + +## Implementation in This Bot + +The Slack bot implements this flow in: + +- **Authorization URL generation**: `slack-bot/handlers/auth_handler.py` (lines 34-62) +- **PKCE pair generation**: `slack-bot/handlers/auth_handler.py` (lines 19-32) +- **Token exchange**: `slack-bot/handlers/auth_handler.py` (lines 88-135) +- **OAuth callback handler**: `slack-bot/app.py` (lines 175-296) + +See `OAUTH_SETUP.md` for detailed setup instructions. diff --git a/slack-bot/requirements.txt b/slack-bot/requirements.txt new file mode 100644 index 0000000..d8e8050 --- /dev/null +++ b/slack-bot/requirements.txt @@ -0,0 +1,18 @@ +# Slack SDK +slack-bolt>=1.18.0 +slack-sdk>=3.23.0 + +# Web framework +Flask>=3.0.0 + +# Async support +aiohttp>=3.9.0 + +# Databricks SDK +databricks-sdk>=0.18.0 + +# HTTP client +httpx>=0.25.0 + +# Environment variables (optional) +python-dotenv>=1.0.0 diff --git a/slack-bot/storage/__init__.py b/slack-bot/storage/__init__.py new file mode 100644 index 0000000..dd564be --- /dev/null +++ b/slack-bot/storage/__init__.py @@ -0,0 +1,3 @@ +from .state_manager import StateManager + +__all__ = ["StateManager"] diff --git a/slack-bot/storage/state_manager.py b/slack-bot/storage/state_manager.py new file mode 100644 index 0000000..9c7404b --- /dev/null +++ b/slack-bot/storage/state_manager.py @@ -0,0 +1,81 @@ +"""State management for Slack bot conversations and user data""" + +import logging +from typing import Dict, List, Optional +from datetime import datetime, timedelta + + +class StateManager: + """ + Manages conversation state and user authentication data. + In production, replace with Redis, DynamoDB, or similar persistent storage. + """ + + def __init__(self): + # In-memory storage (replace with Redis/DB for production) + self._user_tokens: Dict[str, Dict] = {} # user_id -> {token, expires_at} + self._conversation_history: Dict[str, List[Dict]] = {} # thread_ts -> messages + self._user_data: Dict[str, Dict] = {} # user_id -> user data + + # User token management + def store_user_token(self, user_id: str, token: str, expires_in: int = 3600): + """Store OAuth token for a user""" + expires_at = datetime.utcnow() + timedelta(seconds=expires_in) + self._user_tokens[user_id] = { + "token": token, + "expires_at": expires_at + } + logging.info(f"Stored token for user {user_id}") + + def get_user_token(self, user_id: str) -> Optional[str]: + """Retrieve OAuth token for a user""" + if user_id not in self._user_tokens: + return None + + token_data = self._user_tokens[user_id] + if datetime.utcnow() > token_data["expires_at"]: + logging.info(f"Token expired for user {user_id}") + del self._user_tokens[user_id] + return None + + return token_data["token"] + + def delete_user_token(self, user_id: str): + """Delete user's OAuth token (logout)""" + if user_id in self._user_tokens: + del self._user_tokens[user_id] + logging.info(f"Deleted token for user {user_id}") + + # Conversation history management + def get_conversation_history(self, thread_ts: str) -> List[Dict]: + """Get conversation history for a thread""" + return self._conversation_history.get(thread_ts, []) + + def add_message_to_history(self, thread_ts: str, role: str, content: str): + """Add a message to conversation history""" + if thread_ts not in self._conversation_history: + self._conversation_history[thread_ts] = [] + + self._conversation_history[thread_ts].append({ + "role": role, + "content": content + }) + + def clear_conversation_history(self, thread_ts: str): + """Clear conversation history for a thread""" + if thread_ts in self._conversation_history: + del self._conversation_history[thread_ts] + logging.info(f"Cleared history for thread {thread_ts}") + + # User data management + def set_user_data(self, user_id: str, key: str, value): + """Store user-specific data""" + if user_id not in self._user_data: + self._user_data[user_id] = {} + self._user_data[user_id][key] = value + + def get_user_data(self, user_id: str, key: str, default=None): + """Retrieve user-specific data""" + if user_id not in self._user_data: + return default + return self._user_data[user_id].get(key, default) diff --git a/slack-bot/uv.lock b/slack-bot/uv.lock new file mode 100644 index 0000000..d3f6335 --- /dev/null +++ b/slack-bot/uv.lock @@ -0,0 +1,2609 @@ +version = 1 +revision = 2 +requires-python = ">=3.8" +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", + "python_full_version < '3.9'", +] + +[[package]] +name = "aiohappyeyeballs" +version = "2.4.4" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/7f/55/e4373e888fdacb15563ef6fa9fa8c8252476ea071e96fb46defac9f18bf2/aiohappyeyeballs-2.4.4.tar.gz", hash = "sha256:5fdd7d87889c63183afc18ce9271f9b0a7d32c2303e394468dd45d514a757745", size = 21977, upload-time = "2024-11-30T18:44:00.701Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b9/74/fbb6559de3607b3300b9be3cc64e97548d55678e44623db17820dbd20002/aiohappyeyeballs-2.4.4-py3-none-any.whl", hash = "sha256:a980909d50efcd44795c4afeca523296716d50cd756ddca6af8c65b996e27de8", size = 14756, upload-time = "2024-11-30T18:43:39.849Z" }, +] + +[[package]] +name = "aiohappyeyeballs" +version = "2.6.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/26/30/f84a107a9c4331c14b2b586036f40965c128aa4fee4dda5d3d51cb14ad54/aiohappyeyeballs-2.6.1.tar.gz", hash = "sha256:c3f9d0113123803ccadfdf3f0faa505bc78e6a72d1cc4806cbd719826e943558", size = 22760, upload-time = "2025-03-12T01:42:48.764Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/0f/15/5bf3b99495fb160b63f95972b81750f18f7f4e02ad051373b669d17d44f2/aiohappyeyeballs-2.6.1-py3-none-any.whl", hash = "sha256:f349ba8f4b75cb25c99c5c2d84e997e485204d2902a9597802b0371f09331fb8", size = 15265, upload-time = "2025-03-12T01:42:47.083Z" }, +] + +[[package]] +name = "aiohttp" +version = "3.10.11" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "aiohappyeyeballs", version = "2.4.4", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "aiosignal", version = "1.3.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "async-timeout", marker = "python_full_version < '3.9'" }, + { name = "attrs", version = "25.3.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "frozenlist", version = "1.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "multidict", version = "6.1.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "yarl", version = "1.15.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/25/a8/8e2ba36c6e3278d62e0c88aa42bb92ddbef092ac363b390dab4421da5cf5/aiohttp-3.10.11.tar.gz", hash = "sha256:9dc2b8f3dcab2e39e0fa309c8da50c3b55e6f34ab25f1a71d3288f24924d33a7", size = 7551886, upload-time = "2024-11-13T16:40:33.335Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/11/c7/575f9e82d7ef13cb1b45b9db8a5b8fadb35107fb12e33809356ae0155223/aiohttp-3.10.11-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:5077b1a5f40ffa3ba1f40d537d3bec4383988ee51fbba6b74aa8fb1bc466599e", size = 588218, upload-time = "2024-11-13T16:36:38.461Z" }, + { url = "https://files.pythonhosted.org/packages/12/7b/a800dadbd9a47b7f921bfddcd531371371f39b9cd05786c3638bfe2e1175/aiohttp-3.10.11-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8d6a14a4d93b5b3c2891fca94fa9d41b2322a68194422bef0dd5ec1e57d7d298", size = 400815, upload-time = "2024-11-13T16:36:40.547Z" }, + { url = "https://files.pythonhosted.org/packages/cb/28/7dbd53ab10b0ded397feed914880f39ce075bd39393b8dfc322909754a0a/aiohttp-3.10.11-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ffbfde2443696345e23a3c597049b1dd43049bb65337837574205e7368472177", size = 392099, upload-time = "2024-11-13T16:36:43.918Z" }, + { url = "https://files.pythonhosted.org/packages/6a/2e/c6390f49e67911711c2229740e261c501685fe7201f7f918d6ff2fd1cfb0/aiohttp-3.10.11-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:20b3d9e416774d41813bc02fdc0663379c01817b0874b932b81c7f777f67b217", size = 1224854, upload-time = "2024-11-13T16:36:46.473Z" }, + { url = "https://files.pythonhosted.org/packages/69/68/c96afae129201bff4edbece52b3e1abf3a8af57529a42700669458b00b9f/aiohttp-3.10.11-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2b943011b45ee6bf74b22245c6faab736363678e910504dd7531a58c76c9015a", size = 1259641, upload-time = "2024-11-13T16:36:48.28Z" }, + { url = "https://files.pythonhosted.org/packages/63/89/bedd01456442747946114a8c2f30ff1b23d3b2ea0c03709f854c4f354a5a/aiohttp-3.10.11-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:48bc1d924490f0d0b3658fe5c4b081a4d56ebb58af80a6729d4bd13ea569797a", size = 1295412, upload-time = "2024-11-13T16:36:50.286Z" }, + { url = "https://files.pythonhosted.org/packages/9b/4d/942198e2939efe7bfa484781590f082135e9931b8bcafb4bba62cf2d8f2f/aiohttp-3.10.11-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e12eb3f4b1f72aaaf6acd27d045753b18101524f72ae071ae1c91c1cd44ef115", size = 1218311, upload-time = "2024-11-13T16:36:53.721Z" }, + { url = "https://files.pythonhosted.org/packages/a3/5b/8127022912f1fa72dfc39cf37c36f83e0b56afc3b93594b1cf377b6e4ffc/aiohttp-3.10.11-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f14ebc419a568c2eff3c1ed35f634435c24ead2fe19c07426af41e7adb68713a", size = 1189448, upload-time = "2024-11-13T16:36:55.844Z" }, + { url = "https://files.pythonhosted.org/packages/af/12/752878033c8feab3362c0890a4d24e9895921729a53491f6f6fad64d3287/aiohttp-3.10.11-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:72b191cdf35a518bfc7ca87d770d30941decc5aaf897ec8b484eb5cc8c7706f3", size = 1186484, upload-time = "2024-11-13T16:36:58.472Z" }, + { url = "https://files.pythonhosted.org/packages/61/24/1d91c304fca47d5e5002ca23abab9b2196ac79d5c531258e048195b435b2/aiohttp-3.10.11-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:5ab2328a61fdc86424ee540d0aeb8b73bbcad7351fb7cf7a6546fc0bcffa0038", size = 1183864, upload-time = "2024-11-13T16:37:00.737Z" }, + { url = "https://files.pythonhosted.org/packages/c1/70/022d28b898314dac4cb5dd52ead2a372563c8590b1eaab9c5ed017eefb1e/aiohttp-3.10.11-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:aa93063d4af05c49276cf14e419550a3f45258b6b9d1f16403e777f1addf4519", size = 1241460, upload-time = "2024-11-13T16:37:03.175Z" }, + { url = "https://files.pythonhosted.org/packages/c3/15/2b43853330f82acf180602de0f68be62a2838d25d03d2ed40fecbe82479e/aiohttp-3.10.11-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:30283f9d0ce420363c24c5c2421e71a738a2155f10adbb1a11a4d4d6d2715cfc", size = 1258521, upload-time = "2024-11-13T16:37:06.013Z" }, + { url = "https://files.pythonhosted.org/packages/28/38/9ef2076cb06dcc155e7f02275f5da403a3e7c9327b6b075e999f0eb73613/aiohttp-3.10.11-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:e5358addc8044ee49143c546d2182c15b4ac3a60be01c3209374ace05af5733d", size = 1207329, upload-time = "2024-11-13T16:37:08.091Z" }, + { url = "https://files.pythonhosted.org/packages/c2/5f/c5329d67a2c83d8ae17a84e11dec14da5773520913bfc191caaf4cd57e50/aiohttp-3.10.11-cp310-cp310-win32.whl", hash = "sha256:e1ffa713d3ea7cdcd4aea9cddccab41edf6882fa9552940344c44e59652e1120", size = 363835, upload-time = "2024-11-13T16:37:10.017Z" }, + { url = "https://files.pythonhosted.org/packages/0f/c6/ca5d70eea2fdbe283dbc1e7d30649a1a5371b2a2a9150db192446f645789/aiohttp-3.10.11-cp310-cp310-win_amd64.whl", hash = "sha256:778cbd01f18ff78b5dd23c77eb82987ee4ba23408cbed233009fd570dda7e674", size = 382169, upload-time = "2024-11-13T16:37:12.603Z" }, + { url = "https://files.pythonhosted.org/packages/73/96/221ec59bc38395a6c205cbe8bf72c114ce92694b58abc8c3c6b7250efa7f/aiohttp-3.10.11-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:80ff08556c7f59a7972b1e8919f62e9c069c33566a6d28586771711e0eea4f07", size = 587742, upload-time = "2024-11-13T16:37:14.469Z" }, + { url = "https://files.pythonhosted.org/packages/24/17/4e606c969b19de5c31a09b946bd4c37e30c5288ca91d4790aa915518846e/aiohttp-3.10.11-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:2c8f96e9ee19f04c4914e4e7a42a60861066d3e1abf05c726f38d9d0a466e695", size = 400357, upload-time = "2024-11-13T16:37:16.482Z" }, + { url = "https://files.pythonhosted.org/packages/a2/e5/433f59b87ba69736e446824710dd7f26fcd05b24c6647cb1e76554ea5d02/aiohttp-3.10.11-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:fb8601394d537da9221947b5d6e62b064c9a43e88a1ecd7414d21a1a6fba9c24", size = 392099, upload-time = "2024-11-13T16:37:20.013Z" }, + { url = "https://files.pythonhosted.org/packages/d2/a3/3be340f5063970bb9e47f065ee8151edab639d9c2dce0d9605a325ab035d/aiohttp-3.10.11-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2ea224cf7bc2d8856d6971cea73b1d50c9c51d36971faf1abc169a0d5f85a382", size = 1300367, upload-time = "2024-11-13T16:37:22.645Z" }, + { url = "https://files.pythonhosted.org/packages/ba/7d/a3043918466cbee9429792ebe795f92f70eeb40aee4ccbca14c38ee8fa4d/aiohttp-3.10.11-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:db9503f79e12d5d80b3efd4d01312853565c05367493379df76d2674af881caa", size = 1339448, upload-time = "2024-11-13T16:37:24.834Z" }, + { url = "https://files.pythonhosted.org/packages/2c/60/192b378bd9d1ae67716b71ae63c3e97c48b134aad7675915a10853a0b7de/aiohttp-3.10.11-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0f449a50cc33f0384f633894d8d3cd020e3ccef81879c6e6245c3c375c448625", size = 1374875, upload-time = "2024-11-13T16:37:26.799Z" }, + { url = "https://files.pythonhosted.org/packages/e0/d7/cd58bd17f5277d9cc32ecdbb0481ca02c52fc066412de413aa01268dc9b4/aiohttp-3.10.11-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:82052be3e6d9e0c123499127782a01a2b224b8af8c62ab46b3f6197035ad94e9", size = 1285626, upload-time = "2024-11-13T16:37:29.02Z" }, + { url = "https://files.pythonhosted.org/packages/bb/b2/da4953643b7dcdcd29cc99f98209f3653bf02023d95ce8a8fd57ffba0f15/aiohttp-3.10.11-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:20063c7acf1eec550c8eb098deb5ed9e1bb0521613b03bb93644b810986027ac", size = 1246120, upload-time = "2024-11-13T16:37:31.268Z" }, + { url = "https://files.pythonhosted.org/packages/6c/22/1217b3c773055f0cb172e3b7108274a74c0fe9900c716362727303931cbb/aiohttp-3.10.11-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:489cced07a4c11488f47aab1f00d0c572506883f877af100a38f1fedaa884c3a", size = 1265177, upload-time = "2024-11-13T16:37:33.348Z" }, + { url = "https://files.pythonhosted.org/packages/63/5e/3827ad7e61544ed1e73e4fdea7bb87ea35ac59a362d7eb301feb5e859780/aiohttp-3.10.11-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:ea9b3bab329aeaa603ed3bf605f1e2a6f36496ad7e0e1aa42025f368ee2dc07b", size = 1257238, upload-time = "2024-11-13T16:37:35.753Z" }, + { url = "https://files.pythonhosted.org/packages/53/31/951f78751d403da6086b662760e6e8b08201b0dcf5357969f48261b4d0e1/aiohttp-3.10.11-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:ca117819d8ad113413016cb29774b3f6d99ad23c220069789fc050267b786c16", size = 1315944, upload-time = "2024-11-13T16:37:38.317Z" }, + { url = "https://files.pythonhosted.org/packages/0d/79/06ef7a2a69880649261818b135b245de5a4e89fed5a6987c8645428563fc/aiohttp-3.10.11-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:2dfb612dcbe70fb7cdcf3499e8d483079b89749c857a8f6e80263b021745c730", size = 1332065, upload-time = "2024-11-13T16:37:40.725Z" }, + { url = "https://files.pythonhosted.org/packages/10/39/a273857c2d0bbf2152a4201fbf776931c2dac74aa399c6683ed4c286d1d1/aiohttp-3.10.11-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f9b615d3da0d60e7d53c62e22b4fd1c70f4ae5993a44687b011ea3a2e49051b8", size = 1291882, upload-time = "2024-11-13T16:37:43.209Z" }, + { url = "https://files.pythonhosted.org/packages/49/39/7aa387f88403febc96e0494101763afaa14d342109329a01b413b2bac075/aiohttp-3.10.11-cp311-cp311-win32.whl", hash = "sha256:29103f9099b6068bbdf44d6a3d090e0a0b2be6d3c9f16a070dd9d0d910ec08f9", size = 363409, upload-time = "2024-11-13T16:37:45.143Z" }, + { url = "https://files.pythonhosted.org/packages/6f/e9/8eb3dc095ce48499d867ad461d02f1491686b79ad92e4fad4df582f6be7b/aiohttp-3.10.11-cp311-cp311-win_amd64.whl", hash = "sha256:236b28ceb79532da85d59aa9b9bf873b364e27a0acb2ceaba475dc61cffb6f3f", size = 382644, upload-time = "2024-11-13T16:37:47.685Z" }, + { url = "https://files.pythonhosted.org/packages/01/16/077057ef3bd684dbf9a8273a5299e182a8d07b4b252503712ff8b5364fd1/aiohttp-3.10.11-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:7480519f70e32bfb101d71fb9a1f330fbd291655a4c1c922232a48c458c52710", size = 584830, upload-time = "2024-11-13T16:37:49.608Z" }, + { url = "https://files.pythonhosted.org/packages/2c/cf/348b93deb9597c61a51b6682e81f7c7d79290249e886022ef0705d858d90/aiohttp-3.10.11-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:f65267266c9aeb2287a6622ee2bb39490292552f9fbf851baabc04c9f84e048d", size = 397090, upload-time = "2024-11-13T16:37:51.539Z" }, + { url = "https://files.pythonhosted.org/packages/70/bf/903df5cd739dfaf5b827b3d8c9d68ff4fcea16a0ca1aeb948c9da30f56c8/aiohttp-3.10.11-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:7400a93d629a0608dc1d6c55f1e3d6e07f7375745aaa8bd7f085571e4d1cee97", size = 392361, upload-time = "2024-11-13T16:37:53.586Z" }, + { url = "https://files.pythonhosted.org/packages/fb/97/e4792675448a2ac5bd56f377a095233b805dd1315235c940c8ba5624e3cb/aiohttp-3.10.11-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f34b97e4b11b8d4eb2c3a4f975be626cc8af99ff479da7de49ac2c6d02d35725", size = 1309839, upload-time = "2024-11-13T16:37:55.68Z" }, + { url = "https://files.pythonhosted.org/packages/96/d0/ba19b1260da6fbbda4d5b1550d8a53ba3518868f2c143d672aedfdbc6172/aiohttp-3.10.11-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1e7b825da878464a252ccff2958838f9caa82f32a8dbc334eb9b34a026e2c636", size = 1348116, upload-time = "2024-11-13T16:37:58.232Z" }, + { url = "https://files.pythonhosted.org/packages/b3/b9/15100ee7113a2638bfdc91aecc54641609a92a7ce4fe533ebeaa8d43ff93/aiohttp-3.10.11-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f9f92a344c50b9667827da308473005f34767b6a2a60d9acff56ae94f895f385", size = 1391402, upload-time = "2024-11-13T16:38:00.522Z" }, + { url = "https://files.pythonhosted.org/packages/c5/36/831522618ac0dcd0b28f327afd18df7fb6bbf3eaf302f912a40e87714846/aiohttp-3.10.11-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc6f1ab987a27b83c5268a17218463c2ec08dbb754195113867a27b166cd6087", size = 1304239, upload-time = "2024-11-13T16:38:04.195Z" }, + { url = "https://files.pythonhosted.org/packages/60/9f/b7230d0c48b076500ae57adb717aa0656432acd3d8febb1183dedfaa4e75/aiohttp-3.10.11-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1dc0f4ca54842173d03322793ebcf2c8cc2d34ae91cc762478e295d8e361e03f", size = 1256565, upload-time = "2024-11-13T16:38:07.218Z" }, + { url = "https://files.pythonhosted.org/packages/63/c2/35c7b4699f4830b3b0a5c3d5619df16dca8052ae8b488e66065902d559f6/aiohttp-3.10.11-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:7ce6a51469bfaacff146e59e7fb61c9c23006495d11cc24c514a455032bcfa03", size = 1269285, upload-time = "2024-11-13T16:38:09.396Z" }, + { url = "https://files.pythonhosted.org/packages/51/48/bc20ea753909bdeb09f9065260aefa7453e3a57f6a51f56f5216adc1a5e7/aiohttp-3.10.11-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:aad3cd91d484d065ede16f3cf15408254e2469e3f613b241a1db552c5eb7ab7d", size = 1276716, upload-time = "2024-11-13T16:38:12.039Z" }, + { url = "https://files.pythonhosted.org/packages/0c/7b/a8708616b3810f55ead66f8e189afa9474795760473aea734bbea536cd64/aiohttp-3.10.11-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:f4df4b8ca97f658c880fb4b90b1d1ec528315d4030af1ec763247ebfd33d8b9a", size = 1315023, upload-time = "2024-11-13T16:38:15.155Z" }, + { url = "https://files.pythonhosted.org/packages/2a/d6/dfe9134a921e05b01661a127a37b7d157db93428905450e32f9898eef27d/aiohttp-3.10.11-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:2e4e18a0a2d03531edbc06c366954e40a3f8d2a88d2b936bbe78a0c75a3aab3e", size = 1342735, upload-time = "2024-11-13T16:38:17.539Z" }, + { url = "https://files.pythonhosted.org/packages/ca/1a/3bd7f18e3909eabd57e5d17ecdbf5ea4c5828d91341e3676a07de7c76312/aiohttp-3.10.11-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:6ce66780fa1a20e45bc753cda2a149daa6dbf1561fc1289fa0c308391c7bc0a4", size = 1302618, upload-time = "2024-11-13T16:38:19.865Z" }, + { url = "https://files.pythonhosted.org/packages/cf/51/d063133781cda48cfdd1e11fc8ef45ab3912b446feba41556385b3ae5087/aiohttp-3.10.11-cp312-cp312-win32.whl", hash = "sha256:a919c8957695ea4c0e7a3e8d16494e3477b86f33067478f43106921c2fef15bb", size = 360497, upload-time = "2024-11-13T16:38:21.996Z" }, + { url = "https://files.pythonhosted.org/packages/55/4e/f29def9ed39826fe8f85955f2e42fe5cc0cbe3ebb53c97087f225368702e/aiohttp-3.10.11-cp312-cp312-win_amd64.whl", hash = "sha256:b5e29706e6389a2283a91611c91bf24f218962717c8f3b4e528ef529d112ee27", size = 380577, upload-time = "2024-11-13T16:38:24.247Z" }, + { url = "https://files.pythonhosted.org/packages/1f/63/654c185dfe3cf5d4a0d35b6ee49ee6ca91922c694eaa90732e1ba4b40ef1/aiohttp-3.10.11-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:703938e22434d7d14ec22f9f310559331f455018389222eed132808cd8f44127", size = 577381, upload-time = "2024-11-13T16:38:26.708Z" }, + { url = "https://files.pythonhosted.org/packages/4e/c4/ee9c350acb202ba2eb0c44b0f84376b05477e870444192a9f70e06844c28/aiohttp-3.10.11-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:9bc50b63648840854e00084c2b43035a62e033cb9b06d8c22b409d56eb098413", size = 393289, upload-time = "2024-11-13T16:38:29.207Z" }, + { url = "https://files.pythonhosted.org/packages/3d/7c/30d161a7e3b208cef1b922eacf2bbb8578b7e5a62266a6a2245a1dd044dc/aiohttp-3.10.11-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:5f0463bf8b0754bc744e1feb61590706823795041e63edf30118a6f0bf577461", size = 388859, upload-time = "2024-11-13T16:38:31.567Z" }, + { url = "https://files.pythonhosted.org/packages/79/10/8d050e04be447d3d39e5a4a910fa289d930120cebe1b893096bd3ee29063/aiohttp-3.10.11-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f6c6dec398ac5a87cb3a407b068e1106b20ef001c344e34154616183fe684288", size = 1280983, upload-time = "2024-11-13T16:38:33.738Z" }, + { url = "https://files.pythonhosted.org/packages/31/b3/977eca40afe643dcfa6b8d8bb9a93f4cba1d8ed1ead22c92056b08855c7a/aiohttp-3.10.11-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bcaf2d79104d53d4dcf934f7ce76d3d155302d07dae24dff6c9fffd217568067", size = 1317132, upload-time = "2024-11-13T16:38:35.999Z" }, + { url = "https://files.pythonhosted.org/packages/1a/43/b5ee8e697ed0f96a2b3d80b3058fa7590cda508e9cd256274246ba1cf37a/aiohttp-3.10.11-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:25fd5470922091b5a9aeeb7e75be609e16b4fba81cdeaf12981393fb240dd10e", size = 1362630, upload-time = "2024-11-13T16:38:39.016Z" }, + { url = "https://files.pythonhosted.org/packages/28/20/3ae8e993b2990fa722987222dea74d6bac9331e2f530d086f309b4aa8847/aiohttp-3.10.11-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bbde2ca67230923a42161b1f408c3992ae6e0be782dca0c44cb3206bf330dee1", size = 1276865, upload-time = "2024-11-13T16:38:41.423Z" }, + { url = "https://files.pythonhosted.org/packages/02/08/1afb0ab7dcff63333b683e998e751aa2547d1ff897b577d2244b00e6fe38/aiohttp-3.10.11-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:249c8ff8d26a8b41a0f12f9df804e7c685ca35a207e2410adbd3e924217b9006", size = 1230448, upload-time = "2024-11-13T16:38:43.962Z" }, + { url = "https://files.pythonhosted.org/packages/c6/fd/ccd0ff842c62128d164ec09e3dd810208a84d79cd402358a3038ae91f3e9/aiohttp-3.10.11-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:878ca6a931ee8c486a8f7b432b65431d095c522cbeb34892bee5be97b3481d0f", size = 1244626, upload-time = "2024-11-13T16:38:47.089Z" }, + { url = "https://files.pythonhosted.org/packages/9f/75/30e9537ab41ed7cb062338d8df7c4afb0a715b3551cd69fc4ea61cfa5a95/aiohttp-3.10.11-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:8663f7777ce775f0413324be0d96d9730959b2ca73d9b7e2c2c90539139cbdd6", size = 1243608, upload-time = "2024-11-13T16:38:49.47Z" }, + { url = "https://files.pythonhosted.org/packages/c2/e0/3e7a62d99b9080793affddc12a82b11c9bc1312916ad849700d2bddf9786/aiohttp-3.10.11-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:6cd3f10b01f0c31481fba8d302b61603a2acb37b9d30e1d14e0f5a58b7b18a31", size = 1286158, upload-time = "2024-11-13T16:38:51.947Z" }, + { url = "https://files.pythonhosted.org/packages/71/b8/df67886802e71e976996ed9324eb7dc379e53a7d972314e9c7fe3f6ac6bc/aiohttp-3.10.11-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:4e8d8aad9402d3aa02fdc5ca2fe68bcb9fdfe1f77b40b10410a94c7f408b664d", size = 1313636, upload-time = "2024-11-13T16:38:54.424Z" }, + { url = "https://files.pythonhosted.org/packages/3c/3b/aea9c3e70ff4e030f46902df28b4cdf486695f4d78fd9c6698827e2bafab/aiohttp-3.10.11-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:38e3c4f80196b4f6c3a85d134a534a56f52da9cb8d8e7af1b79a32eefee73a00", size = 1273772, upload-time = "2024-11-13T16:38:56.846Z" }, + { url = "https://files.pythonhosted.org/packages/e9/9e/4b4c5705270d1c4ee146516ad288af720798d957ba46504aaf99b86e85d9/aiohttp-3.10.11-cp313-cp313-win32.whl", hash = "sha256:fc31820cfc3b2863c6e95e14fcf815dc7afe52480b4dc03393c4873bb5599f71", size = 358679, upload-time = "2024-11-13T16:38:59.787Z" }, + { url = "https://files.pythonhosted.org/packages/28/1d/18ef37549901db94717d4389eb7be807acbfbdeab48a73ff2993fc909118/aiohttp-3.10.11-cp313-cp313-win_amd64.whl", hash = "sha256:4996ff1345704ffdd6d75fb06ed175938c133425af616142e7187f28dc75f14e", size = 378073, upload-time = "2024-11-13T16:39:02.065Z" }, + { url = "https://files.pythonhosted.org/packages/dd/f2/59165bee7bba0b0634525834c622f152a30715a1d8280f6291a0cb86b1e6/aiohttp-3.10.11-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:74baf1a7d948b3d640badeac333af581a367ab916b37e44cf90a0334157cdfd2", size = 592135, upload-time = "2024-11-13T16:39:04.774Z" }, + { url = "https://files.pythonhosted.org/packages/2e/0e/b3555c504745af66efbf89d16811148ff12932b86fad529d115538fe2739/aiohttp-3.10.11-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:473aebc3b871646e1940c05268d451f2543a1d209f47035b594b9d4e91ce8339", size = 402913, upload-time = "2024-11-13T16:39:08.065Z" }, + { url = "https://files.pythonhosted.org/packages/31/bb/2890a3c77126758ef58536ca9f7476a12ba2021e0cd074108fb99b8c8747/aiohttp-3.10.11-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:c2f746a6968c54ab2186574e15c3f14f3e7f67aef12b761e043b33b89c5b5f95", size = 394013, upload-time = "2024-11-13T16:39:10.638Z" }, + { url = "https://files.pythonhosted.org/packages/74/82/0ab5199b473558846d72901a714b6afeb6f6a6a6a4c3c629e2c107418afd/aiohttp-3.10.11-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d110cabad8360ffa0dec8f6ec60e43286e9d251e77db4763a87dcfe55b4adb92", size = 1255578, upload-time = "2024-11-13T16:39:13.14Z" }, + { url = "https://files.pythonhosted.org/packages/f8/b2/f232477dd3c0e95693a903c4815bfb8d831f6a1a67e27ad14d30a774eeda/aiohttp-3.10.11-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e0099c7d5d7afff4202a0c670e5b723f7718810000b4abcbc96b064129e64bc7", size = 1298780, upload-time = "2024-11-13T16:39:15.721Z" }, + { url = "https://files.pythonhosted.org/packages/34/8c/11972235a6b53d5b69098f2ee6629ff8f99cd9592dcaa620c7868deb5673/aiohttp-3.10.11-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0316e624b754dbbf8c872b62fe6dcb395ef20c70e59890dfa0de9eafccd2849d", size = 1336093, upload-time = "2024-11-13T16:39:19.11Z" }, + { url = "https://files.pythonhosted.org/packages/03/be/7ad9a6cd2312221cf7b6837d8e2d8e4660fbd4f9f15bccf79ef857f41f4d/aiohttp-3.10.11-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5a5f7ab8baf13314e6b2485965cbacb94afff1e93466ac4d06a47a81c50f9cca", size = 1250296, upload-time = "2024-11-13T16:39:22.363Z" }, + { url = "https://files.pythonhosted.org/packages/bb/8d/a3885a582d9fc481bccb155d082f83a7a846942e36e4a4bba061e3d6b95e/aiohttp-3.10.11-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c891011e76041e6508cbfc469dd1a8ea09bc24e87e4c204e05f150c4c455a5fa", size = 1215020, upload-time = "2024-11-13T16:39:25.205Z" }, + { url = "https://files.pythonhosted.org/packages/bb/e7/09a1736b7264316dc3738492d9b559f2a54b985660f21d76095c9890a62e/aiohttp-3.10.11-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:9208299251370ee815473270c52cd3f7069ee9ed348d941d574d1457d2c73e8b", size = 1210591, upload-time = "2024-11-13T16:39:28.311Z" }, + { url = "https://files.pythonhosted.org/packages/58/b1/ee684631f6af98065d49ac8416db7a8e74ea33e1378bc75952ab0522342f/aiohttp-3.10.11-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:459f0f32c8356e8125f45eeff0ecf2b1cb6db1551304972702f34cd9e6c44658", size = 1211255, upload-time = "2024-11-13T16:39:30.799Z" }, + { url = "https://files.pythonhosted.org/packages/8f/55/e21e312fd6c581f244dd2ed077ccb784aade07c19416a6316b1453f02c4e/aiohttp-3.10.11-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:14cdc8c1810bbd4b4b9f142eeee23cda528ae4e57ea0923551a9af4820980e39", size = 1278114, upload-time = "2024-11-13T16:39:34.141Z" }, + { url = "https://files.pythonhosted.org/packages/d8/7f/ff6df0e90df6759693f52720ebedbfa10982d97aa1fd02c6ca917a6399ea/aiohttp-3.10.11-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:971aa438a29701d4b34e4943e91b5e984c3ae6ccbf80dd9efaffb01bd0b243a9", size = 1292714, upload-time = "2024-11-13T16:39:37.216Z" }, + { url = "https://files.pythonhosted.org/packages/3a/45/63f35367dfffae41e7abd0603f92708b5b3655fda55c08388ac2c7fb127b/aiohttp-3.10.11-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:9a309c5de392dfe0f32ee57fa43ed8fc6ddf9985425e84bd51ed66bb16bce3a7", size = 1233734, upload-time = "2024-11-13T16:39:40.599Z" }, + { url = "https://files.pythonhosted.org/packages/ec/ee/74b0696c0e84e06c43beab9302f353d97dc9f0cccd7ccf3ee648411b849b/aiohttp-3.10.11-cp38-cp38-win32.whl", hash = "sha256:9ec1628180241d906a0840b38f162a3215114b14541f1a8711c368a8739a9be4", size = 365350, upload-time = "2024-11-13T16:39:43.852Z" }, + { url = "https://files.pythonhosted.org/packages/21/0c/74c895688db09a2852056abf32d128991ec2fb41e5f57a1fe0928e15151c/aiohttp-3.10.11-cp38-cp38-win_amd64.whl", hash = "sha256:9c6e0ffd52c929f985c7258f83185d17c76d4275ad22e90aa29f38e211aacbec", size = 384542, upload-time = "2024-11-13T16:39:47.093Z" }, + { url = "https://files.pythonhosted.org/packages/cc/df/aa0d1548db818395a372b5f90e62072677ce786d6b19680c49dd4da3825f/aiohttp-3.10.11-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:cdc493a2e5d8dc79b2df5bec9558425bcd39aff59fc949810cbd0832e294b106", size = 589833, upload-time = "2024-11-13T16:39:49.72Z" }, + { url = "https://files.pythonhosted.org/packages/75/7c/d11145784b3fa29c0421a3883a4b91ee8c19acb40332b1d2e39f47be4e5b/aiohttp-3.10.11-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:b3e70f24e7d0405be2348da9d5a7836936bf3a9b4fd210f8c37e8d48bc32eca6", size = 401685, upload-time = "2024-11-13T16:39:52.263Z" }, + { url = "https://files.pythonhosted.org/packages/e2/67/1b5f93babeb060cb683d23104b243be1d6299fe6cd807dcb56cf67d2e62c/aiohttp-3.10.11-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:968b8fb2a5eee2770eda9c7b5581587ef9b96fbdf8dcabc6b446d35ccc69df01", size = 392957, upload-time = "2024-11-13T16:39:54.668Z" }, + { url = "https://files.pythonhosted.org/packages/e1/4d/441df53aafd8dd97b8cfe9e467c641fa19cb5113e7601a7f77f2124518e0/aiohttp-3.10.11-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:deef4362af9493d1382ef86732ee2e4cbc0d7c005947bd54ad1a9a16dd59298e", size = 1229754, upload-time = "2024-11-13T16:39:57.166Z" }, + { url = "https://files.pythonhosted.org/packages/4d/cc/f1397a2501b95cb94580de7051395e85af95a1e27aed1f8af73459ddfa22/aiohttp-3.10.11-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:686b03196976e327412a1b094f4120778c7c4b9cff9bce8d2fdfeca386b89829", size = 1266246, upload-time = "2024-11-13T16:40:00.723Z" }, + { url = "https://files.pythonhosted.org/packages/c2/b5/7d33dae7630b4e9f90d634c6a90cb0923797e011b71cd9b10fe685aec3f6/aiohttp-3.10.11-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3bf6d027d9d1d34e1c2e1645f18a6498c98d634f8e373395221121f1c258ace8", size = 1301720, upload-time = "2024-11-13T16:40:04.111Z" }, + { url = "https://files.pythonhosted.org/packages/51/36/f917bcc63bc489aa3f534fa81efbf895fa5286745dcd8bbd0eb9dbc923a1/aiohttp-3.10.11-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:099fd126bf960f96d34a760e747a629c27fb3634da5d05c7ef4d35ef4ea519fc", size = 1221527, upload-time = "2024-11-13T16:40:06.851Z" }, + { url = "https://files.pythonhosted.org/packages/32/c2/1a303a072b4763d99d4b0664a3a8b952869e3fbb660d4239826bd0c56cc1/aiohttp-3.10.11-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c73c4d3dae0b4644bc21e3de546530531d6cdc88659cdeb6579cd627d3c206aa", size = 1192309, upload-time = "2024-11-13T16:40:09.65Z" }, + { url = "https://files.pythonhosted.org/packages/62/ef/d62f705dc665382b78ef171e5ba2616c395220ac7c1f452f0d2dcad3f9f5/aiohttp-3.10.11-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:0c5580f3c51eea91559db3facd45d72e7ec970b04528b4709b1f9c2555bd6d0b", size = 1189481, upload-time = "2024-11-13T16:40:12.77Z" }, + { url = "https://files.pythonhosted.org/packages/40/22/3e3eb4f97e5c4f52ccd198512b583c0c9135aa4e989c7ade97023c4cd282/aiohttp-3.10.11-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:fdf6429f0caabfd8a30c4e2eaecb547b3c340e4730ebfe25139779b9815ba138", size = 1187877, upload-time = "2024-11-13T16:40:15.985Z" }, + { url = "https://files.pythonhosted.org/packages/b5/73/77475777fbe2b3efaceb49db2859f1a22c96fd5869d736e80375db05bbf4/aiohttp-3.10.11-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:d97187de3c276263db3564bb9d9fad9e15b51ea10a371ffa5947a5ba93ad6777", size = 1246006, upload-time = "2024-11-13T16:40:19.17Z" }, + { url = "https://files.pythonhosted.org/packages/ef/f7/5b060d19065473da91838b63d8fd4d20ef8426a7d905cc8f9cd11eabd780/aiohttp-3.10.11-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:0acafb350cfb2eba70eb5d271f55e08bd4502ec35e964e18ad3e7d34d71f7261", size = 1260403, upload-time = "2024-11-13T16:40:21.761Z" }, + { url = "https://files.pythonhosted.org/packages/6c/ea/e9ad224815cd83c8dfda686d2bafa2cab5b93d7232e09470a8d2a158acde/aiohttp-3.10.11-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:c13ed0c779911c7998a58e7848954bd4d63df3e3575f591e321b19a2aec8df9f", size = 1208643, upload-time = "2024-11-13T16:40:24.803Z" }, + { url = "https://files.pythonhosted.org/packages/ba/c1/e1c6bba72f379adbd52958601a8642546ed0807964afba3b1b5b8cfb1bc0/aiohttp-3.10.11-cp39-cp39-win32.whl", hash = "sha256:22b7c540c55909140f63ab4f54ec2c20d2635c0289cdd8006da46f3327f971b9", size = 364419, upload-time = "2024-11-13T16:40:27.817Z" }, + { url = "https://files.pythonhosted.org/packages/30/24/50862e06e86cd263c60661e00b9d2c8d7fdece4fe95454ed5aa21ecf8036/aiohttp-3.10.11-cp39-cp39-win_amd64.whl", hash = "sha256:7b26b1551e481012575dab8e3727b16fe7dd27eb2711d2e63ced7368756268fb", size = 382857, upload-time = "2024-11-13T16:40:30.427Z" }, +] + +[[package]] +name = "aiohttp" +version = "3.13.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "aiohappyeyeballs", version = "2.6.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "aiosignal", version = "1.4.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "async-timeout", marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, + { name = "attrs", version = "25.4.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "frozenlist", version = "1.8.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "multidict", version = "6.7.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "propcache", version = "0.4.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "yarl", version = "1.22.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ba/fa/3ae643cd525cf6844d3dc810481e5748107368eb49563c15a5fb9f680750/aiohttp-3.13.1.tar.gz", hash = "sha256:4b7ee9c355015813a6aa085170b96ec22315dabc3d866fd77d147927000e9464", size = 7835344, upload-time = "2025-10-17T14:03:29.337Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e6/34/5097441cc3047eccc2e0bfed3760ed068489b8392545d3aec0d8fbfab2b5/aiohttp-3.13.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:2349a6b642020bf20116a8a5c83bae8ba071acf1461c7cbe45fc7fafd552e7e2", size = 735069, upload-time = "2025-10-17T13:58:56.602Z" }, + { url = "https://files.pythonhosted.org/packages/8c/2b/726466b4b4b16271a3db2a8a914d754d6cb9cee7bebde1f3ac6043e4e030/aiohttp-3.13.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:2a8434ca31c093a90edb94d7d70e98706ce4d912d7f7a39f56e1af26287f4bb7", size = 492575, upload-time = "2025-10-17T13:58:58.696Z" }, + { url = "https://files.pythonhosted.org/packages/82/1f/364e64292c95bb6c9e2823b0afa1ad3f06524c573d45df82294be572489d/aiohttp-3.13.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:0bd610a7e87431741021a9a6ab775e769ea8c01bf01766d481282bfb17df597f", size = 487862, upload-time = "2025-10-17T13:59:00.315Z" }, + { url = "https://files.pythonhosted.org/packages/23/b0/c5a774b3125ac854987b8ca45a6d995829987d01ece4525d3fc369a9ca88/aiohttp-3.13.1-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:777ec887264b629395b528af59b8523bf3164d4c6738cd8989485ff3eda002e2", size = 1666761, upload-time = "2025-10-17T13:59:02.224Z" }, + { url = "https://files.pythonhosted.org/packages/29/be/32c6c1d3a6c69e594b855bbf4014bea4c42008b0daac8c6e5c9f03207b89/aiohttp-3.13.1-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:ac1892f56e2c445aca5ba28f3bf8e16b26dfc05f3c969867b7ef553b74cb4ebe", size = 1634627, upload-time = "2025-10-17T13:59:03.829Z" }, + { url = "https://files.pythonhosted.org/packages/73/8d/fde3a8f4801b14e0b9490f5bc86c5106cb7d96bd60ff2aaee53749c72fe1/aiohttp-3.13.1-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:499a047d1c5e490c31d16c033e2e47d1358f0e15175c7a1329afc6dfeb04bc09", size = 1726564, upload-time = "2025-10-17T13:59:05.997Z" }, + { url = "https://files.pythonhosted.org/packages/52/b2/8290556f1f6b17b1af976a9abb17f9b54dc7218e11bbf6abbebaa7cc70fb/aiohttp-3.13.1-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:610be925f89501938c770f1e28ca9dd62e9b308592c81bd5d223ce92434c0089", size = 1814413, upload-time = "2025-10-17T13:59:08.975Z" }, + { url = "https://files.pythonhosted.org/packages/ef/6b/4b657e9fa72479df38117609d4ec8e4b07e8110b872df3872f9c6a96e26b/aiohttp-3.13.1-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:90eb902c06c6ac85d6b80fa9f2bd681f25b1ebf73433d428b3d182a507242711", size = 1667964, upload-time = "2025-10-17T13:59:10.606Z" }, + { url = "https://files.pythonhosted.org/packages/ee/ed/563de175d01fa26459a60a7c82dbf69d20e356d459476a7526329091b4c3/aiohttp-3.13.1-cp310-cp310-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ab8ac3224b2beb46266c094b3869d68d5f96f35dba98e03dea0acbd055eefa03", size = 1553917, upload-time = "2025-10-17T13:59:12.312Z" }, + { url = "https://files.pythonhosted.org/packages/39/26/48a4b5681eada16eb5b39cae277765aed1644b03610c43eadb8b331ccfea/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:79ac65b6e2731558aad1e4c1a655d2aa2a77845b62acecf5898b0d4fe8c76618", size = 1637730, upload-time = "2025-10-17T13:59:14.395Z" }, + { url = "https://files.pythonhosted.org/packages/c1/43/57b137af37344e03c7f6b28ddf38a4af820b53c1fa9ce13f668fe468d2e2/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:4dadbd858ed8c04d1aa7a2a91ad65f8e1fbd253ae762ef5be8111e763d576c3c", size = 1644088, upload-time = "2025-10-17T13:59:16.749Z" }, + { url = "https://files.pythonhosted.org/packages/0d/c4/e49bafa4babef09929b10968a6b6efe3707fbaa5c5bb7c8db7f810232269/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:e0b2ccd331bc77149e88e919aa95c228a011e03e1168fd938e6aeb1a317d7a8a", size = 1696215, upload-time = "2025-10-17T13:59:18.711Z" }, + { url = "https://files.pythonhosted.org/packages/15/e4/8414be434b3e50f9089ffa7c4d5130ba6ff0d1c6fa9f55cd760b088abbe0/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_riscv64.whl", hash = "sha256:fba3c85fb24fe204e73f3c92f09f4f5cfa55fa7e54b34d59d91b7c5a258d0f6a", size = 1540617, upload-time = "2025-10-17T13:59:20.46Z" }, + { url = "https://files.pythonhosted.org/packages/bd/8b/31cb6725f819b74a9c0b0055c500187294e73aea40708b6a5aa7b328ea4c/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:8d5011e4e741d2635cda18f2997a56e8e1d1b94591dc8732f2ef1d3e1bfc5f45", size = 1713509, upload-time = "2025-10-17T13:59:22.61Z" }, + { url = "https://files.pythonhosted.org/packages/24/ac/49a79c2711423cfa091e265c46e58617de31258c64502b890f25421cb742/aiohttp-3.13.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:c5fe2728a89c82574bd3132d59237c3b5fb83e2e00a320e928d05d74d1ae895f", size = 1654702, upload-time = "2025-10-17T13:59:24.396Z" }, + { url = "https://files.pythonhosted.org/packages/30/52/1cf23cffeda1f079f20cd9c72174a76e8b0c6595def6803892e37ee35c8a/aiohttp-3.13.1-cp310-cp310-win32.whl", hash = "sha256:add14a5e68cbcfc526c89c1ed8ea963f5ff8b9b4b854985b07820c6fbfdb3c3c", size = 430898, upload-time = "2025-10-17T13:59:26.227Z" }, + { url = "https://files.pythonhosted.org/packages/0e/13/214a01f2936f4645b1fbd5cba9001331ca5af5c04bbdbe747eed330a8516/aiohttp-3.13.1-cp310-cp310-win_amd64.whl", hash = "sha256:a4cc9d9cfdf75a69ae921c407e02d0c1799ab333b0bc6f7928c175f47c080d6a", size = 453684, upload-time = "2025-10-17T13:59:28.129Z" }, + { url = "https://files.pythonhosted.org/packages/be/2c/739d03730ffce57d2093e2e611e1541ac9a4b3bb88288c33275058b9ffc2/aiohttp-3.13.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9eefa0a891e85dca56e2d00760945a6325bd76341ec386d3ad4ff72eb97b7e64", size = 742004, upload-time = "2025-10-17T13:59:29.73Z" }, + { url = "https://files.pythonhosted.org/packages/fc/f8/7f5b7f7184d7c80e421dbaecbd13e0b2a0bb8663fd0406864f9a167a438c/aiohttp-3.13.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6c20eb646371a5a57a97de67e52aac6c47badb1564e719b3601bbb557a2e8fd0", size = 495601, upload-time = "2025-10-17T13:59:31.312Z" }, + { url = "https://files.pythonhosted.org/packages/3e/af/fb78d028b9642dd33ff127d9a6a151586f33daff631b05250fecd0ab23f8/aiohttp-3.13.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:bfc28038cd86fb1deed5cc75c8fda45c6b0f5c51dfd76f8c63d3d22dc1ab3d1b", size = 491790, upload-time = "2025-10-17T13:59:33.304Z" }, + { url = "https://files.pythonhosted.org/packages/1e/ae/e40e422ee995e4f91f7f087b86304e3dd622d3a5b9ca902a1e94ebf9a117/aiohttp-3.13.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8b22eeffca2e522451990c31a36fe0e71079e6112159f39a4391f1c1e259a795", size = 1746350, upload-time = "2025-10-17T13:59:35.158Z" }, + { url = "https://files.pythonhosted.org/packages/28/a5/fe6022bb869bf2d2633b155ed8348d76358c22d5ff9692a15016b2d1019f/aiohttp-3.13.1-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:65782b2977c05ebd78787e3c834abe499313bf69d6b8be4ff9c340901ee7541f", size = 1703046, upload-time = "2025-10-17T13:59:37.077Z" }, + { url = "https://files.pythonhosted.org/packages/5a/a5/c4ef3617d7cdc49f2d5af077f19794946f0f2d94b93c631ace79047361a2/aiohttp-3.13.1-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:dacba54f9be3702eb866b0b9966754b475e1e39996e29e442c3cd7f1117b43a9", size = 1806161, upload-time = "2025-10-17T13:59:38.837Z" }, + { url = "https://files.pythonhosted.org/packages/ad/45/b87d2430aee7e7d00b24e3dff2c5bd69f21017f6edb19cfd91e514664fc8/aiohttp-3.13.1-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:aa878da718e8235302c365e376b768035add36b55177706d784a122cb822a6a4", size = 1894546, upload-time = "2025-10-17T13:59:40.741Z" }, + { url = "https://files.pythonhosted.org/packages/e8/a2/79eb466786a7f11a0292c353a8a9b95e88268c48c389239d7531d66dbb48/aiohttp-3.13.1-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0e4b4e607fbd4964d65945a7b9d1e7f98b0d5545736ea613f77d5a2a37ff1e46", size = 1745683, upload-time = "2025-10-17T13:59:42.59Z" }, + { url = "https://files.pythonhosted.org/packages/93/1a/153b0ad694f377e94eacc85338efe03ed4776a396c8bb47bd9227135792a/aiohttp-3.13.1-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:0c3db2d0e5477ad561bf7ba978c3ae5f8f78afda70daa05020179f759578754f", size = 1605418, upload-time = "2025-10-17T13:59:45.229Z" }, + { url = "https://files.pythonhosted.org/packages/3f/4e/18605b1bfeb4b00d3396d833647cdb213118e2a96862e5aebee62ad065b4/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:9739d34506fdf59bf2c092560d502aa728b8cdb33f34ba15fb5e2852c35dd829", size = 1722379, upload-time = "2025-10-17T13:59:46.969Z" }, + { url = "https://files.pythonhosted.org/packages/72/13/0a38ad385d547fb283e0e1fe1ff1dff8899bd4ed0aaceeb13ec14abbf136/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:b902e30a268a85d50197b4997edc6e78842c14c0703450f632c2d82f17577845", size = 1716693, upload-time = "2025-10-17T13:59:49.217Z" }, + { url = "https://files.pythonhosted.org/packages/55/65/7029d7573ab9009adde380052c6130d02c8db52195fda112db35e914fe7b/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:1bbfc04c8de7def6504cce0a97f9885a5c805fd2395a0634bc10f9d6ecb42524", size = 1784174, upload-time = "2025-10-17T13:59:51.439Z" }, + { url = "https://files.pythonhosted.org/packages/2d/36/fd46e39cb85418e45b0e4a8bfc39651ee0b8f08ea006adf217a221cdb269/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:6941853405a38a5eeb7d9776db77698df373ff7fa8c765cb81ea14a344fccbeb", size = 1593716, upload-time = "2025-10-17T13:59:53.367Z" }, + { url = "https://files.pythonhosted.org/packages/85/b8/188e0cb1be37b4408373171070fda17c3bf9c67c0d3d4fd5ee5b1fa108e1/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:7764adcd2dc8bd21c8228a53dda2005428498dc4d165f41b6086f0ac1c65b1c9", size = 1799254, upload-time = "2025-10-17T13:59:55.352Z" }, + { url = "https://files.pythonhosted.org/packages/67/ff/fdf768764eb427b0cc9ebb2cebddf990f94d98b430679f8383c35aa114be/aiohttp-3.13.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:c09e08d38586fa59e5a2f9626505a0326fadb8e9c45550f029feeb92097a0afc", size = 1738122, upload-time = "2025-10-17T13:59:57.263Z" }, + { url = "https://files.pythonhosted.org/packages/94/84/fce7a4d575943394d7c0e632273838eb6f39de8edf25386017bf5f0de23b/aiohttp-3.13.1-cp311-cp311-win32.whl", hash = "sha256:ce1371675e74f6cf271d0b5530defb44cce713fd0ab733713562b3a2b870815c", size = 430491, upload-time = "2025-10-17T13:59:59.466Z" }, + { url = "https://files.pythonhosted.org/packages/ac/d2/d21b8ab6315a5d588c550ab285b4f02ae363edf012920e597904c5a56608/aiohttp-3.13.1-cp311-cp311-win_amd64.whl", hash = "sha256:77a2f5cc28cf4704cc157be135c6a6cfb38c9dea478004f1c0fd7449cf445c28", size = 454808, upload-time = "2025-10-17T14:00:01.247Z" }, + { url = "https://files.pythonhosted.org/packages/1a/72/d463a10bf29871f6e3f63bcf3c91362dc4d72ed5917a8271f96672c415ad/aiohttp-3.13.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0760bd9a28efe188d77b7c3fe666e6ef74320d0f5b105f2e931c7a7e884c8230", size = 736218, upload-time = "2025-10-17T14:00:03.51Z" }, + { url = "https://files.pythonhosted.org/packages/26/13/f7bccedbe52ea5a6eef1e4ebb686a8d7765319dfd0a5939f4238cb6e79e6/aiohttp-3.13.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:7129a424b441c3fe018a414401bf1b9e1d49492445f5676a3aecf4f74f67fcdb", size = 491251, upload-time = "2025-10-17T14:00:05.756Z" }, + { url = "https://files.pythonhosted.org/packages/0c/7c/7ea51b5aed6cc69c873f62548da8345032aa3416336f2d26869d4d37b4a2/aiohttp-3.13.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:e1cb04ae64a594f6ddf5cbb024aba6b4773895ab6ecbc579d60414f8115e9e26", size = 490394, upload-time = "2025-10-17T14:00:07.504Z" }, + { url = "https://files.pythonhosted.org/packages/31/05/1172cc4af4557f6522efdee6eb2b9f900e1e320a97e25dffd3c5a6af651b/aiohttp-3.13.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:782d656a641e755decd6bd98d61d2a8ea062fd45fd3ff8d4173605dd0d2b56a1", size = 1737455, upload-time = "2025-10-17T14:00:09.403Z" }, + { url = "https://files.pythonhosted.org/packages/24/3d/ce6e4eca42f797d6b1cd3053cf3b0a22032eef3e4d1e71b9e93c92a3f201/aiohttp-3.13.1-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:f92ad8169767429a6d2237331726c03ccc5f245222f9373aa045510976af2b35", size = 1699176, upload-time = "2025-10-17T14:00:11.314Z" }, + { url = "https://files.pythonhosted.org/packages/25/04/7127ba55653e04da51477372566b16ae786ef854e06222a1c96b4ba6c8ef/aiohttp-3.13.1-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:0e778f634ca50ec005eefa2253856921c429581422d887be050f2c1c92e5ce12", size = 1767216, upload-time = "2025-10-17T14:00:13.668Z" }, + { url = "https://files.pythonhosted.org/packages/b8/3b/43bca1e75847e600f40df829a6b2f0f4e1d4c70fb6c4818fdc09a462afd5/aiohttp-3.13.1-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:9bc36b41cf4aab5d3b34d22934a696ab83516603d1bc1f3e4ff9930fe7d245e5", size = 1865870, upload-time = "2025-10-17T14:00:15.852Z" }, + { url = "https://files.pythonhosted.org/packages/9e/69/b204e5d43384197a614c88c1717c324319f5b4e7d0a1b5118da583028d40/aiohttp-3.13.1-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3fd4570ea696aee27204dd524f287127ed0966d14d309dc8cc440f474e3e7dbd", size = 1751021, upload-time = "2025-10-17T14:00:18.297Z" }, + { url = "https://files.pythonhosted.org/packages/1c/af/845dc6b6fdf378791d720364bf5150f80d22c990f7e3a42331d93b337cc7/aiohttp-3.13.1-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:7bda795f08b8a620836ebfb0926f7973972a4bf8c74fdf9145e489f88c416811", size = 1561448, upload-time = "2025-10-17T14:00:20.152Z" }, + { url = "https://files.pythonhosted.org/packages/7a/91/d2ab08cd77ed76a49e4106b1cfb60bce2768242dd0c4f9ec0cb01e2cbf94/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:055a51d90e351aae53dcf324d0eafb2abe5b576d3ea1ec03827d920cf81a1c15", size = 1698196, upload-time = "2025-10-17T14:00:22.131Z" }, + { url = "https://files.pythonhosted.org/packages/5e/d1/082f0620dc428ecb8f21c08a191a4694915cd50f14791c74a24d9161cc50/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:d4131df864cbcc09bb16d3612a682af0db52f10736e71312574d90f16406a867", size = 1719252, upload-time = "2025-10-17T14:00:24.453Z" }, + { url = "https://files.pythonhosted.org/packages/fc/78/2af2f44491be7b08e43945b72d2b4fd76f0a14ba850ba9e41d28a7ce716a/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:163d3226e043f79bf47c87f8dfc89c496cc7bc9128cb7055ce026e435d551720", size = 1736529, upload-time = "2025-10-17T14:00:26.567Z" }, + { url = "https://files.pythonhosted.org/packages/b0/34/3e919ecdc93edaea8d140138049a0d9126141072e519535e2efa38eb7a02/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:a2370986a3b75c1a5f3d6f6d763fc6be4b430226577b0ed16a7c13a75bf43d8f", size = 1553723, upload-time = "2025-10-17T14:00:28.592Z" }, + { url = "https://files.pythonhosted.org/packages/21/4b/d8003aeda2f67f359b37e70a5a4b53fee336d8e89511ac307ff62aeefcdb/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:d7c14de0c7c9f1e6e785ce6cbe0ed817282c2af0012e674f45b4e58c6d4ea030", size = 1763394, upload-time = "2025-10-17T14:00:31.051Z" }, + { url = "https://files.pythonhosted.org/packages/4c/7b/1dbe6a39e33af9baaafc3fc016a280663684af47ba9f0e5d44249c1f72ec/aiohttp-3.13.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:bb611489cf0db10b99beeb7280bd39e0ef72bc3eb6d8c0f0a16d8a56075d1eb7", size = 1718104, upload-time = "2025-10-17T14:00:33.407Z" }, + { url = "https://files.pythonhosted.org/packages/5c/88/bd1b38687257cce67681b9b0fa0b16437be03383fa1be4d1a45b168bef25/aiohttp-3.13.1-cp312-cp312-win32.whl", hash = "sha256:f90fe0ee75590f7428f7c8b5479389d985d83c949ea10f662ab928a5ed5cf5e6", size = 425303, upload-time = "2025-10-17T14:00:35.829Z" }, + { url = "https://files.pythonhosted.org/packages/0e/e3/4481f50dd6f27e9e58c19a60cff44029641640237e35d32b04aaee8cf95f/aiohttp-3.13.1-cp312-cp312-win_amd64.whl", hash = "sha256:3461919a9dca272c183055f2aab8e6af0adc810a1b386cce28da11eb00c859d9", size = 452071, upload-time = "2025-10-17T14:00:37.764Z" }, + { url = "https://files.pythonhosted.org/packages/16/6d/d267b132342e1080f4c1bb7e1b4e96b168b3cbce931ec45780bff693ff95/aiohttp-3.13.1-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:55785a7f8f13df0c9ca30b5243d9909bd59f48b274262a8fe78cee0828306e5d", size = 730727, upload-time = "2025-10-17T14:00:39.681Z" }, + { url = "https://files.pythonhosted.org/packages/92/c8/1cf495bac85cf71b80fad5f6d7693e84894f11b9fe876b64b0a1e7cbf32f/aiohttp-3.13.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:4bef5b83296cebb8167707b4f8d06c1805db0af632f7a72d7c5288a84667e7c3", size = 488678, upload-time = "2025-10-17T14:00:41.541Z" }, + { url = "https://files.pythonhosted.org/packages/a8/19/23c6b81cca587ec96943d977a58d11d05a82837022e65cd5502d665a7d11/aiohttp-3.13.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:27af0619c33f9ca52f06069ec05de1a357033449ab101836f431768ecfa63ff5", size = 487637, upload-time = "2025-10-17T14:00:43.527Z" }, + { url = "https://files.pythonhosted.org/packages/48/58/8f9464afb88b3eed145ad7c665293739b3a6f91589694a2bb7e5778cbc72/aiohttp-3.13.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:a47fe43229a8efd3764ef7728a5c1158f31cdf2a12151fe99fde81c9ac87019c", size = 1718975, upload-time = "2025-10-17T14:00:45.496Z" }, + { url = "https://files.pythonhosted.org/packages/e1/8b/c3da064ca392b2702f53949fd7c403afa38d9ee10bf52c6ad59a42537103/aiohttp-3.13.1-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:6e68e126de5b46e8b2bee73cab086b5d791e7dc192056916077aa1e2e2b04437", size = 1686905, upload-time = "2025-10-17T14:00:47.707Z" }, + { url = "https://files.pythonhosted.org/packages/0a/a4/9c8a3843ecf526daee6010af1a66eb62579be1531d2d5af48ea6f405ad3c/aiohttp-3.13.1-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:e65ef49dd22514329c55970d39079618a8abf856bae7147913bb774a3ab3c02f", size = 1754907, upload-time = "2025-10-17T14:00:49.702Z" }, + { url = "https://files.pythonhosted.org/packages/a4/80/1f470ed93e06436e3fc2659a9fc329c192fa893fb7ed4e884d399dbfb2a8/aiohttp-3.13.1-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:0e425a7e0511648b3376839dcc9190098671a47f21a36e815b97762eb7d556b0", size = 1857129, upload-time = "2025-10-17T14:00:51.822Z" }, + { url = "https://files.pythonhosted.org/packages/cc/e6/33d305e6cce0a8daeb79c7d8d6547d6e5f27f4e35fa4883fc9c9eb638596/aiohttp-3.13.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:010dc9b7110f055006acd3648d5d5955bb6473b37c3663ec42a1b4cba7413e6b", size = 1738189, upload-time = "2025-10-17T14:00:53.976Z" }, + { url = "https://files.pythonhosted.org/packages/ac/42/8df03367e5a64327fe0c39291080697795430c438fc1139c7cc1831aa1df/aiohttp-3.13.1-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:1b5c722d0ca5f57d61066b5dfa96cdb87111e2519156b35c1f8dd17c703bee7a", size = 1553608, upload-time = "2025-10-17T14:00:56.144Z" }, + { url = "https://files.pythonhosted.org/packages/96/17/6d5c73cd862f1cf29fddcbb54aac147037ff70a043a2829d03a379e95742/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:93029f0e9b77b714904a281b5aa578cdc8aa8ba018d78c04e51e1c3d8471b8ec", size = 1681809, upload-time = "2025-10-17T14:00:58.603Z" }, + { url = "https://files.pythonhosted.org/packages/be/31/8926c8ab18533f6076ce28d2c329a203b58c6861681906e2d73b9c397588/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:d1824c7d08d8ddfc8cb10c847f696942e5aadbd16fd974dfde8bd2c3c08a9fa1", size = 1711161, upload-time = "2025-10-17T14:01:01.744Z" }, + { url = "https://files.pythonhosted.org/packages/f2/36/2f83e1ca730b1e0a8cf1c8ab9559834c5eec9f5da86e77ac71f0d16b521d/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:8f47d0ff5b3eb9c1278a2f56ea48fda667da8ebf28bd2cb378b7c453936ce003", size = 1731999, upload-time = "2025-10-17T14:01:04.626Z" }, + { url = "https://files.pythonhosted.org/packages/b9/ec/1f818cc368dfd4d5ab4e9efc8f2f6f283bfc31e1c06d3e848bcc862d4591/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:8a396b1da9b51ded79806ac3b57a598f84e0769eaa1ba300655d8b5e17b70c7b", size = 1548684, upload-time = "2025-10-17T14:01:06.828Z" }, + { url = "https://files.pythonhosted.org/packages/d3/ad/33d36efd16e4fefee91b09a22a3a0e1b830f65471c3567ac5a8041fac812/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:d9c52a65f54796e066b5d674e33b53178014752d28bca555c479c2c25ffcec5b", size = 1756676, upload-time = "2025-10-17T14:01:09.517Z" }, + { url = "https://files.pythonhosted.org/packages/3c/c4/4a526d84e77d464437713ca909364988ed2e0cd0cdad2c06cb065ece9e08/aiohttp-3.13.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:a89da72d18d6c95a653470b78d8ee5aa3c4b37212004c103403d0776cbea6ff0", size = 1715577, upload-time = "2025-10-17T14:01:11.958Z" }, + { url = "https://files.pythonhosted.org/packages/a2/21/e39638b7d9c7f1362c4113a91870f89287e60a7ea2d037e258b81e8b37d5/aiohttp-3.13.1-cp313-cp313-win32.whl", hash = "sha256:02e0258b7585ddf5d01c79c716ddd674386bfbf3041fbbfe7bdf9c7c32eb4a9b", size = 424468, upload-time = "2025-10-17T14:01:14.344Z" }, + { url = "https://files.pythonhosted.org/packages/cc/00/f3a92c592a845ebb2f47d102a67f35f0925cb854c5e7386f1a3a1fdff2ab/aiohttp-3.13.1-cp313-cp313-win_amd64.whl", hash = "sha256:ef56ffe60e8d97baac123272bde1ab889ee07d3419606fae823c80c2b86c403e", size = 450806, upload-time = "2025-10-17T14:01:16.437Z" }, + { url = "https://files.pythonhosted.org/packages/97/be/0f6c41d2fd0aab0af133c509cabaf5b1d78eab882cb0ceb872e87ceeabf7/aiohttp-3.13.1-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:77f83b3dc5870a2ea79a0fcfdcc3fc398187ec1675ff61ec2ceccad27ecbd303", size = 733828, upload-time = "2025-10-17T14:01:18.58Z" }, + { url = "https://files.pythonhosted.org/packages/75/14/24e2ac5efa76ae30e05813e0f50737005fd52da8ddffee474d4a5e7f38a6/aiohttp-3.13.1-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:9cafd2609ebb755e47323306c7666283fbba6cf82b5f19982ea627db907df23a", size = 489320, upload-time = "2025-10-17T14:01:20.644Z" }, + { url = "https://files.pythonhosted.org/packages/da/5a/4cbe599358d05ea7db4869aff44707b57d13f01724d48123dc68b3288d5a/aiohttp-3.13.1-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:9c489309a2ca548d5f11131cfb4092f61d67954f930bba7e413bcdbbb82d7fae", size = 489899, upload-time = "2025-10-17T14:01:22.638Z" }, + { url = "https://files.pythonhosted.org/packages/67/96/3aec9d9cfc723273d4386328a1e2562cf23629d2f57d137047c49adb2afb/aiohttp-3.13.1-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:79ac15fe5fdbf3c186aa74b656cd436d9a1e492ba036db8901c75717055a5b1c", size = 1716556, upload-time = "2025-10-17T14:01:25.406Z" }, + { url = "https://files.pythonhosted.org/packages/b9/99/39a3d250595b5c8172843831221fa5662884f63f8005b00b4034f2a7a836/aiohttp-3.13.1-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:095414be94fce3bc080684b4cd50fb70d439bc4662b2a1984f45f3bf9ede08aa", size = 1665814, upload-time = "2025-10-17T14:01:27.683Z" }, + { url = "https://files.pythonhosted.org/packages/3b/96/8319e7060a85db14a9c178bc7b3cf17fad458db32ba6d2910de3ca71452d/aiohttp-3.13.1-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c68172e1a2dca65fa1272c85ca72e802d78b67812b22827df01017a15c5089fa", size = 1755767, upload-time = "2025-10-17T14:01:29.914Z" }, + { url = "https://files.pythonhosted.org/packages/1c/c6/0a2b3d886b40aa740fa2294cd34ed46d2e8108696748492be722e23082a7/aiohttp-3.13.1-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:3751f9212bcd119944d4ea9de6a3f0fee288c177b8ca55442a2cdff0c8201eb3", size = 1836591, upload-time = "2025-10-17T14:01:32.28Z" }, + { url = "https://files.pythonhosted.org/packages/fb/34/8ab5904b3331c91a58507234a1e2f662f837e193741609ee5832eb436251/aiohttp-3.13.1-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8619dca57d98a8353abdc7a1eeb415548952b39d6676def70d9ce76d41a046a9", size = 1714915, upload-time = "2025-10-17T14:01:35.138Z" }, + { url = "https://files.pythonhosted.org/packages/b5/d3/d36077ca5f447649112189074ac6c192a666bf68165b693e48c23b0d008c/aiohttp-3.13.1-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:97795a0cb0a5f8a843759620e9cbd8889f8079551f5dcf1ccd99ed2f056d9632", size = 1546579, upload-time = "2025-10-17T14:01:38.237Z" }, + { url = "https://files.pythonhosted.org/packages/a8/14/dbc426a1bb1305c4fc78ce69323498c9e7c699983366ef676aa5d3f949fa/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:1060e058da8f9f28a7026cdfca9fc886e45e551a658f6a5c631188f72a3736d2", size = 1680633, upload-time = "2025-10-17T14:01:40.902Z" }, + { url = "https://files.pythonhosted.org/packages/29/83/1e68e519aff9f3ef6d4acb6cdda7b5f592ef5c67c8f095dc0d8e06ce1c3e/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:f48a2c26333659101ef214907d29a76fe22ad7e912aa1e40aeffdff5e8180977", size = 1678675, upload-time = "2025-10-17T14:01:43.779Z" }, + { url = "https://files.pythonhosted.org/packages/38/b9/7f3e32a81c08b6d29ea15060c377e1f038ad96cd9923a85f30e817afff22/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:f1dfad638b9c91ff225162b2824db0e99ae2d1abe0dc7272b5919701f0a1e685", size = 1726829, upload-time = "2025-10-17T14:01:46.546Z" }, + { url = "https://files.pythonhosted.org/packages/23/ce/610b1f77525a0a46639aea91377b12348e9f9412cc5ddcb17502aa4681c7/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:8fa09ab6dd567cb105db4e8ac4d60f377a7a94f67cf669cac79982f626360f32", size = 1542985, upload-time = "2025-10-17T14:01:49.082Z" }, + { url = "https://files.pythonhosted.org/packages/53/39/3ac8dfdad5de38c401846fa071fcd24cb3b88ccfb024854df6cbd9b4a07e/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:4159fae827f9b5f655538a4f99b7cbc3a2187e5ca2eee82f876ef1da802ccfa9", size = 1741556, upload-time = "2025-10-17T14:01:51.846Z" }, + { url = "https://files.pythonhosted.org/packages/2a/48/b1948b74fea7930b0f29595d1956842324336de200593d49a51a40607fdc/aiohttp-3.13.1-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:ad671118c19e9cfafe81a7a05c294449fe0ebb0d0c6d5bb445cd2190023f5cef", size = 1696175, upload-time = "2025-10-17T14:01:54.232Z" }, + { url = "https://files.pythonhosted.org/packages/96/26/063bba38e4b27b640f56cc89fe83cc3546a7ae162c2e30ca345f0ccdc3d1/aiohttp-3.13.1-cp314-cp314-win32.whl", hash = "sha256:c5c970c148c48cf6acb65224ca3c87a47f74436362dde75c27bc44155ccf7dfc", size = 430254, upload-time = "2025-10-17T14:01:56.451Z" }, + { url = "https://files.pythonhosted.org/packages/88/aa/25fd764384dc4eab714023112d3548a8dd69a058840d61d816ea736097a2/aiohttp-3.13.1-cp314-cp314-win_amd64.whl", hash = "sha256:748a00167b7a88385756fa615417d24081cba7e58c8727d2e28817068b97c18c", size = 456256, upload-time = "2025-10-17T14:01:58.752Z" }, + { url = "https://files.pythonhosted.org/packages/d4/9f/9ba6059de4bad25c71cd88e3da53f93e9618ea369cf875c9f924b1c167e2/aiohttp-3.13.1-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:390b73e99d7a1f0f658b3f626ba345b76382f3edc65f49d6385e326e777ed00e", size = 765956, upload-time = "2025-10-17T14:02:01.515Z" }, + { url = "https://files.pythonhosted.org/packages/1f/30/b86da68b494447d3060f45c7ebb461347535dab4af9162a9267d9d86ca31/aiohttp-3.13.1-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:27e83abb330e687e019173d8fc1fd6a1cf471769624cf89b1bb49131198a810a", size = 503206, upload-time = "2025-10-17T14:02:03.818Z" }, + { url = "https://files.pythonhosted.org/packages/c1/21/d27a506552843ff9eeb9fcc2d45f943b09eefdfdf205aab044f4f1f39f6a/aiohttp-3.13.1-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:2b20eed07131adbf3e873e009c2869b16a579b236e9d4b2f211bf174d8bef44a", size = 507719, upload-time = "2025-10-17T14:02:05.947Z" }, + { url = "https://files.pythonhosted.org/packages/58/23/4042230ec7e4edc7ba43d0342b5a3d2fe0222ca046933c4251a35aaf17f5/aiohttp-3.13.1-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:58fee9ef8477fd69e823b92cfd1f590ee388521b5ff8f97f3497e62ee0656212", size = 1862758, upload-time = "2025-10-17T14:02:08.469Z" }, + { url = "https://files.pythonhosted.org/packages/df/88/525c45bea7cbb9f65df42cadb4ff69f6a0dbf95931b0ff7d1fdc40a1cb5f/aiohttp-3.13.1-cp314-cp314t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:1f62608fcb7b3d034d5e9496bea52d94064b7b62b06edba82cd38191336bbeda", size = 1717790, upload-time = "2025-10-17T14:02:11.37Z" }, + { url = "https://files.pythonhosted.org/packages/1d/80/21e9b5eb77df352a5788713f37359b570a793f0473f3a72db2e46df379b9/aiohttp-3.13.1-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:fdc4d81c3dfc999437f23e36d197e8b557a3f779625cd13efe563a9cfc2ce712", size = 1842088, upload-time = "2025-10-17T14:02:13.872Z" }, + { url = "https://files.pythonhosted.org/packages/d2/bf/d1738f6d63fe8b2a0ad49533911b3347f4953cd001bf3223cb7b61f18dff/aiohttp-3.13.1-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:601d7ec812f746fd80ff8af38eeb3f196e1bab4a4d39816ccbc94c222d23f1d0", size = 1934292, upload-time = "2025-10-17T14:02:16.624Z" }, + { url = "https://files.pythonhosted.org/packages/04/e6/26cab509b42610ca49573f2fc2867810f72bd6a2070182256c31b14f2e98/aiohttp-3.13.1-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:47c3f21c469b840d9609089435c0d9918ae89f41289bf7cc4afe5ff7af5458db", size = 1791328, upload-time = "2025-10-17T14:02:19.051Z" }, + { url = "https://files.pythonhosted.org/packages/8a/6d/baf7b462852475c9d045bee8418d9cdf280efb687752b553e82d0c58bcc2/aiohttp-3.13.1-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:d6c6cdc0750db88520332d4aaa352221732b0cafe89fd0e42feec7cb1b5dc236", size = 1622663, upload-time = "2025-10-17T14:02:21.397Z" }, + { url = "https://files.pythonhosted.org/packages/c8/48/396a97318af9b5f4ca8b3dc14a67976f71c6400a9609c622f96da341453f/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:58a12299eeb1fca2414ee2bc345ac69b0f765c20b82c3ab2a75d91310d95a9f6", size = 1787791, upload-time = "2025-10-17T14:02:24.212Z" }, + { url = "https://files.pythonhosted.org/packages/a8/e2/6925f6784134ce3ff3ce1a8502ab366432a3b5605387618c1a939ce778d9/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:0989cbfc195a4de1bb48f08454ef1cb47424b937e53ed069d08404b9d3c7aea1", size = 1775459, upload-time = "2025-10-17T14:02:26.971Z" }, + { url = "https://files.pythonhosted.org/packages/c3/e3/b372047ba739fc39f199b99290c4cc5578ce5fd125f69168c967dac44021/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:feb5ee664300e2435e0d1bc3443a98925013dfaf2cae9699c1f3606b88544898", size = 1789250, upload-time = "2025-10-17T14:02:29.686Z" }, + { url = "https://files.pythonhosted.org/packages/02/8c/9f48b93d7d57fc9ef2ad4adace62e4663ea1ce1753806c4872fb36b54c39/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:58a6f8702da0c3606fb5cf2e669cce0ca681d072fe830968673bb4c69eb89e88", size = 1616139, upload-time = "2025-10-17T14:02:32.151Z" }, + { url = "https://files.pythonhosted.org/packages/5c/c6/c64e39d61aaa33d7de1be5206c0af3ead4b369bf975dac9fdf907a4291c1/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:a417ceb433b9d280e2368ffea22d4bc6e3e0d894c4bc7768915124d57d0964b6", size = 1815829, upload-time = "2025-10-17T14:02:34.635Z" }, + { url = "https://files.pythonhosted.org/packages/22/75/e19e93965ea675f1151753b409af97a14f1d888588a555e53af1e62b83eb/aiohttp-3.13.1-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:8ac8854f7b0466c5d6a9ea49249b3f6176013859ac8f4bb2522ad8ed6b94ded2", size = 1760923, upload-time = "2025-10-17T14:02:37.364Z" }, + { url = "https://files.pythonhosted.org/packages/6c/a4/06ed38f1dabd98ea136fd116cba1d02c9b51af5a37d513b6850a9a567d86/aiohttp-3.13.1-cp314-cp314t-win32.whl", hash = "sha256:be697a5aeff42179ed13b332a411e674994bcd406c81642d014ace90bf4bb968", size = 463318, upload-time = "2025-10-17T14:02:39.924Z" }, + { url = "https://files.pythonhosted.org/packages/04/0f/27e4fdde899e1e90e35eeff56b54ed63826435ad6cdb06b09ed312d1b3fa/aiohttp-3.13.1-cp314-cp314t-win_amd64.whl", hash = "sha256:f1d6aa90546a4e8f20c3500cb68ab14679cd91f927fa52970035fd3207dfb3da", size = 496721, upload-time = "2025-10-17T14:02:42.199Z" }, + { url = "https://files.pythonhosted.org/packages/1d/c8/76ef829954d9b08c0b785f43394e6cdce5c87673bf5cd6bc6dd04e1c3a04/aiohttp-3.13.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:a5dc5c3b086adc232fd07e691dcc452e8e407bf7c810e6f7e18fd3941a24c5c0", size = 737600, upload-time = "2025-10-17T14:02:44.458Z" }, + { url = "https://files.pythonhosted.org/packages/32/82/2e73cd55d35d6ecc4c5180d9388d8255c07010553badaa3634d2a77ab05a/aiohttp-3.13.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:fb7c5f0b35f5a3a06bd5e1a7b46204c2dca734cd839da830db81f56ce60981fe", size = 493893, upload-time = "2025-10-17T14:02:47.003Z" }, + { url = "https://files.pythonhosted.org/packages/0e/4f/b57cb7a1d16844ec58fa436fa8f8d80afd6de9281162f93a4087cd69a962/aiohttp-3.13.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:cb1e557bd1a90f28dc88a6e31332753795cd471f8d18da749c35930e53d11880", size = 489224, upload-time = "2025-10-17T14:02:49.52Z" }, + { url = "https://files.pythonhosted.org/packages/55/93/ab03a54fa57fbba05a40cae18b704965a5f2b5deec9ae11fb81b554cd393/aiohttp-3.13.1-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e95ea8fb27fbf667d322626a12db708be308b66cd9afd4a997230ded66ffcab4", size = 1661558, upload-time = "2025-10-17T14:02:51.897Z" }, + { url = "https://files.pythonhosted.org/packages/27/de/c50d56b8a2baaf2178743758cb013aae343fec47a4f3e261459142df7b83/aiohttp-3.13.1-cp39-cp39-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:f37da298a486e53f9b5e8ef522719b3787c4fe852639a1edcfcc9f981f2c20ba", size = 1625710, upload-time = "2025-10-17T14:02:54.39Z" }, + { url = "https://files.pythonhosted.org/packages/11/07/1b4db771161afa796a86ffb4787fab7d81199a428d285742b553ba8d3c66/aiohttp-3.13.1-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:37cc1b9773d2a01c3f221c3ebecf0c82b1c93f55f3fde52929e40cf2ed777e6c", size = 1722626, upload-time = "2025-10-17T14:02:56.879Z" }, + { url = "https://files.pythonhosted.org/packages/50/c6/b79780419bde91bac24ca91919f4a2bbd534cf52dc4a80b6a32bf9cfe60d/aiohttp-3.13.1-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:412bfc63a6de4907aae6041da256d183f875bf4dc01e05412b1d19cfc25ee08c", size = 1811491, upload-time = "2025-10-17T14:02:59.765Z" }, + { url = "https://files.pythonhosted.org/packages/79/03/08dc71a29a32ef1b2e1d76c2d72559d0b729eb82933f824f149517f72d02/aiohttp-3.13.1-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d8ccd2946aadf7793643b57d98d5a82598295a37f98d218984039d5179823cd5", size = 1659551, upload-time = "2025-10-17T14:03:02.377Z" }, + { url = "https://files.pythonhosted.org/packages/3b/3d/0c43bef82e5ed691bf7b47bebe7323010bd5c9b685285829d4381e6514a3/aiohttp-3.13.1-cp39-cp39-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:51b3c44434a50bca1763792c6b98b9ba1d614339284780b43107ef37ec3aa1dc", size = 1552454, upload-time = "2025-10-17T14:03:04.894Z" }, + { url = "https://files.pythonhosted.org/packages/09/e4/7f606092153e2f85868462efbe038da97f00459a787ba1dae84e43196041/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:9bff813424c70ad38667edfad4fefe8ca1b09a53621ce7d0fd017e418438f58a", size = 1632849, upload-time = "2025-10-17T14:03:07.301Z" }, + { url = "https://files.pythonhosted.org/packages/f8/e8/7b5dfa9c405022f6e08a97fe1e5f6aa3b8d368c0eaaca03888d4f91ede64/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:ed782a438ff4b66ce29503a1555be51a36e4b5048c3b524929378aa7450c26a9", size = 1638089, upload-time = "2025-10-17T14:03:09.798Z" }, + { url = "https://files.pythonhosted.org/packages/8f/d7/3bdb019f03c95238112eca8a02894b191ca1dd267544971433adc856912a/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:a1d6fd6e9e3578a7aeb0fa11e9a544dceccb840330277bf281325aa0fe37787e", size = 1692679, upload-time = "2025-10-17T14:03:12.274Z" }, + { url = "https://files.pythonhosted.org/packages/29/26/a4910cad8e2699bf64f68bb01a585f164829c0b89036d4141f915c2a26ca/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_riscv64.whl", hash = "sha256:7c5e2660c6d6ab0d85c45bc8bd9f685983ebc63a5c7c0fd3ddeb647712722eca", size = 1539122, upload-time = "2025-10-17T14:03:15.106Z" }, + { url = "https://files.pythonhosted.org/packages/eb/ae/f4d1da25e352bee6a6973adee6f7fd551a5790b1ab2141a630d321c902c1/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:168279a11571a39d689fc7b9725ddcde0dc68f2336b06b69fcea0203f9fb25d8", size = 1708961, upload-time = "2025-10-17T14:03:17.756Z" }, + { url = "https://files.pythonhosted.org/packages/92/40/b355f78d1c8a25568831847bddcb9e0999374b514bae566d53588e21a56f/aiohttp-3.13.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:ff0357fa3dd28cf49ad8c515452a1d1d7ad611b513e0a4f6fa6ad6780abaddfd", size = 1647886, upload-time = "2025-10-17T14:03:20.231Z" }, + { url = "https://files.pythonhosted.org/packages/fd/41/c1a23276181d1f7a59e1569ae1f2ea4b03067753672791bb3587b7202899/aiohttp-3.13.1-cp39-cp39-win32.whl", hash = "sha256:a617769e8294ca58601a579697eae0b0e1b1ef770c5920d55692827d6b330ff9", size = 431590, upload-time = "2025-10-17T14:03:22.939Z" }, + { url = "https://files.pythonhosted.org/packages/86/68/c83126f351ffa50b8b83b88c5e9bc219b6709a74139447d2e05786510873/aiohttp-3.13.1-cp39-cp39-win_amd64.whl", hash = "sha256:f2543eebf890739fd93d06e2c16d97bdf1301d2cda5ffceb7a68441c7b590a92", size = 454627, upload-time = "2025-10-17T14:03:26.698Z" }, +] + +[[package]] +name = "aiosignal" +version = "1.3.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "frozenlist", version = "1.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ae/67/0952ed97a9793b4958e5736f6d2b346b414a2cd63e82d05940032f45b32f/aiosignal-1.3.1.tar.gz", hash = "sha256:54cd96e15e1649b75d6c87526a6ff0b6c1b0dd3459f43d9ca11d48c339b68cfc", size = 19422, upload-time = "2022-11-08T16:03:58.806Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/76/ac/a7305707cb852b7e16ff80eaf5692309bde30e2b1100a1fcacdc8f731d97/aiosignal-1.3.1-py3-none-any.whl", hash = "sha256:f8376fb07dd1e86a584e4fcdec80b36b7f81aac666ebc724e2c090300dd83b17", size = 7617, upload-time = "2022-11-08T16:03:57.483Z" }, +] + +[[package]] +name = "aiosignal" +version = "1.4.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "frozenlist", version = "1.8.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "typing-extensions", version = "4.15.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9' and python_full_version < '3.13'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/61/62/06741b579156360248d1ec624842ad0edf697050bbaf7c3e46394e106ad1/aiosignal-1.4.0.tar.gz", hash = "sha256:f47eecd9468083c2029cc99945502cb7708b082c232f9aca65da147157b251c7", size = 25007, upload-time = "2025-07-03T22:54:43.528Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/fb/76/641ae371508676492379f16e2fa48f4e2c11741bd63c48be4b12a6b09cba/aiosignal-1.4.0-py3-none-any.whl", hash = "sha256:053243f8b92b990551949e63930a839ff0cf0b0ebbe0597b0f3fb19e1a0fe82e", size = 7490, upload-time = "2025-07-03T22:54:42.156Z" }, +] + +[[package]] +name = "anyio" +version = "4.5.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "exceptiongroup", marker = "python_full_version < '3.9'" }, + { name = "idna", marker = "python_full_version < '3.9'" }, + { name = "sniffio", marker = "python_full_version < '3.9'" }, + { name = "typing-extensions", version = "4.13.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/4d/f9/9a7ce600ebe7804daf90d4d48b1c0510a4561ddce43a596be46676f82343/anyio-4.5.2.tar.gz", hash = "sha256:23009af4ed04ce05991845451e11ef02fc7c5ed29179ac9a420e5ad0ac7ddc5b", size = 171293, upload-time = "2024-10-13T22:18:03.307Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1b/b4/f7e396030e3b11394436358ca258a81d6010106582422f23443c16ca1873/anyio-4.5.2-py3-none-any.whl", hash = "sha256:c011ee36bc1e8ba40e5a81cb9df91925c218fe9b778554e0b56a21e1b5d4716f", size = 89766, upload-time = "2024-10-13T22:18:01.524Z" }, +] + +[[package]] +name = "anyio" +version = "4.11.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "exceptiongroup", marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, + { name = "idna", marker = "python_full_version >= '3.9'" }, + { name = "sniffio", marker = "python_full_version >= '3.9'" }, + { name = "typing-extensions", version = "4.15.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9' and python_full_version < '3.13'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c6/78/7d432127c41b50bccba979505f272c16cbcadcc33645d5fa3a738110ae75/anyio-4.11.0.tar.gz", hash = "sha256:82a8d0b81e318cc5ce71a5f1f8b5c4e63619620b63141ef8c995fa0db95a57c4", size = 219094, upload-time = "2025-09-23T09:19:12.58Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/15/b3/9b1a8074496371342ec1e796a96f99c82c945a339cd81a8e73de28b4cf9e/anyio-4.11.0-py3-none-any.whl", hash = "sha256:0287e96f4d26d4149305414d4e3bc32f0dcd0862365a4bddea19d7a1ec38c4fc", size = 109097, upload-time = "2025-09-23T09:19:10.601Z" }, +] + +[[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.3.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/5a/b0/1367933a8532ee6ff8d63537de4f1177af4bff9f3e829baf7331f595bb24/attrs-25.3.0.tar.gz", hash = "sha256:75d7cefc7fb576747b2c81b4442d4d4a1ce0900973527c011d1030fd3bf4af1b", size = 812032, upload-time = "2025-03-13T11:10:22.779Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/77/06/bb80f5f86020c4551da315d78b3ab75e8228f89f0162f2c3a819e407941a/attrs-25.3.0-py3-none-any.whl", hash = "sha256:427318ce031701fea540783410126f03899a97ffc6f61596ad581ac2e40e3bc3", size = 63815, upload-time = "2025-03-13T11:10:21.14Z" }, +] + +[[package]] +name = "attrs" +version = "25.4.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/6b/5c/685e6633917e101e5dcb62b9dd76946cbb57c26e133bae9e0cd36033c0a9/attrs-25.4.0.tar.gz", hash = "sha256:16d5969b87f0859ef33a48b35d55ac1be6e42ae49d5e853b597db70c35c57e11", size = 934251, upload-time = "2025-10-06T13:54:44.725Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/3a/2a/7cc015f5b9f5db42b7d48157e23356022889fc354a2813c15934b7cb5c0e/attrs-25.4.0-py3-none-any.whl", hash = "sha256:adcf7e2a1fb3b36ac48d97835bb6d8ade15b8dcce26aba8bf1d14847b57a3373", size = 67615, upload-time = "2025-10-06T13:54:43.17Z" }, +] + +[[package]] +name = "black" +version = "24.8.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "click", version = "8.1.8", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "mypy-extensions", marker = "python_full_version < '3.9'" }, + { name = "packaging", marker = "python_full_version < '3.9'" }, + { name = "pathspec", marker = "python_full_version < '3.9'" }, + { name = "platformdirs", version = "4.3.6", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "tomli", marker = "python_full_version < '3.9'" }, + { name = "typing-extensions", version = "4.13.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/04/b0/46fb0d4e00372f4a86a6f8efa3cb193c9f64863615e39010b1477e010578/black-24.8.0.tar.gz", hash = "sha256:2500945420b6784c38b9ee885af039f5e7471ef284ab03fa35ecdde4688cd83f", size = 644810, upload-time = "2024-08-02T17:43:18.405Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/47/6e/74e29edf1fba3887ed7066930a87f698ffdcd52c5dbc263eabb06061672d/black-24.8.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:09cdeb74d494ec023ded657f7092ba518e8cf78fa8386155e4a03fdcc44679e6", size = 1632092, upload-time = "2024-08-02T17:47:26.911Z" }, + { url = "https://files.pythonhosted.org/packages/ab/49/575cb6c3faee690b05c9d11ee2e8dba8fbd6d6c134496e644c1feb1b47da/black-24.8.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:81c6742da39f33b08e791da38410f32e27d632260e599df7245cccee2064afeb", size = 1457529, upload-time = "2024-08-02T17:47:29.109Z" }, + { url = "https://files.pythonhosted.org/packages/7a/b4/d34099e95c437b53d01c4aa37cf93944b233066eb034ccf7897fa4e5f286/black-24.8.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:707a1ca89221bc8a1a64fb5e15ef39cd755633daa672a9db7498d1c19de66a42", size = 1757443, upload-time = "2024-08-02T17:46:20.306Z" }, + { url = "https://files.pythonhosted.org/packages/87/a0/6d2e4175ef364b8c4b64f8441ba041ed65c63ea1db2720d61494ac711c15/black-24.8.0-cp310-cp310-win_amd64.whl", hash = "sha256:d6417535d99c37cee4091a2f24eb2b6d5ec42b144d50f1f2e436d9fe1916fe1a", size = 1418012, upload-time = "2024-08-02T17:47:20.33Z" }, + { url = "https://files.pythonhosted.org/packages/08/a6/0a3aa89de9c283556146dc6dbda20cd63a9c94160a6fbdebaf0918e4a3e1/black-24.8.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:fb6e2c0b86bbd43dee042e48059c9ad7830abd5c94b0bc518c0eeec57c3eddc1", size = 1615080, upload-time = "2024-08-02T17:48:05.467Z" }, + { url = "https://files.pythonhosted.org/packages/db/94/b803d810e14588bb297e565821a947c108390a079e21dbdcb9ab6956cd7a/black-24.8.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:837fd281f1908d0076844bc2b801ad2d369c78c45cf800cad7b61686051041af", size = 1438143, upload-time = "2024-08-02T17:47:30.247Z" }, + { url = "https://files.pythonhosted.org/packages/a5/b5/f485e1bbe31f768e2e5210f52ea3f432256201289fd1a3c0afda693776b0/black-24.8.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:62e8730977f0b77998029da7971fa896ceefa2c4c4933fcd593fa599ecbf97a4", size = 1738774, upload-time = "2024-08-02T17:46:17.837Z" }, + { url = "https://files.pythonhosted.org/packages/a8/69/a000fc3736f89d1bdc7f4a879f8aaf516fb03613bb51a0154070383d95d9/black-24.8.0-cp311-cp311-win_amd64.whl", hash = "sha256:72901b4913cbac8972ad911dc4098d5753704d1f3c56e44ae8dce99eecb0e3af", size = 1427503, upload-time = "2024-08-02T17:46:22.654Z" }, + { url = "https://files.pythonhosted.org/packages/a2/a8/05fb14195cfef32b7c8d4585a44b7499c2a4b205e1662c427b941ed87054/black-24.8.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:7c046c1d1eeb7aea9335da62472481d3bbf3fd986e093cffd35f4385c94ae368", size = 1646132, upload-time = "2024-08-02T17:49:52.843Z" }, + { url = "https://files.pythonhosted.org/packages/41/77/8d9ce42673e5cb9988f6df73c1c5c1d4e9e788053cccd7f5fb14ef100982/black-24.8.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:649f6d84ccbae73ab767e206772cc2d7a393a001070a4c814a546afd0d423aed", size = 1448665, upload-time = "2024-08-02T17:47:54.479Z" }, + { url = "https://files.pythonhosted.org/packages/cc/94/eff1ddad2ce1d3cc26c162b3693043c6b6b575f538f602f26fe846dfdc75/black-24.8.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2b59b250fdba5f9a9cd9d0ece6e6d993d91ce877d121d161e4698af3eb9c1018", size = 1762458, upload-time = "2024-08-02T17:46:19.384Z" }, + { url = "https://files.pythonhosted.org/packages/28/ea/18b8d86a9ca19a6942e4e16759b2fa5fc02bbc0eb33c1b866fcd387640ab/black-24.8.0-cp312-cp312-win_amd64.whl", hash = "sha256:6e55d30d44bed36593c3163b9bc63bf58b3b30e4611e4d88a0c3c239930ed5b2", size = 1436109, upload-time = "2024-08-02T17:46:52.97Z" }, + { url = "https://files.pythonhosted.org/packages/9f/d4/ae03761ddecc1a37d7e743b89cccbcf3317479ff4b88cfd8818079f890d0/black-24.8.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:505289f17ceda596658ae81b61ebbe2d9b25aa78067035184ed0a9d855d18afd", size = 1617322, upload-time = "2024-08-02T17:51:20.203Z" }, + { url = "https://files.pythonhosted.org/packages/14/4b/4dfe67eed7f9b1ddca2ec8e4418ea74f0d1dc84d36ea874d618ffa1af7d4/black-24.8.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b19c9ad992c7883ad84c9b22aaa73562a16b819c1d8db7a1a1a49fb7ec13c7d2", size = 1442108, upload-time = "2024-08-02T17:50:40.824Z" }, + { url = "https://files.pythonhosted.org/packages/97/14/95b3f91f857034686cae0e73006b8391d76a8142d339b42970eaaf0416ea/black-24.8.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:1f13f7f386f86f8121d76599114bb8c17b69d962137fc70efe56137727c7047e", size = 1745786, upload-time = "2024-08-02T17:46:02.939Z" }, + { url = "https://files.pythonhosted.org/packages/95/54/68b8883c8aa258a6dde958cd5bdfada8382bec47c5162f4a01e66d839af1/black-24.8.0-cp38-cp38-win_amd64.whl", hash = "sha256:f490dbd59680d809ca31efdae20e634f3fae27fba3ce0ba3208333b713bc3920", size = 1426754, upload-time = "2024-08-02T17:46:38.603Z" }, + { url = "https://files.pythonhosted.org/packages/13/b2/b3f24fdbb46f0e7ef6238e131f13572ee8279b70f237f221dd168a9dba1a/black-24.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:eab4dd44ce80dea27dc69db40dab62d4ca96112f87996bca68cd75639aeb2e4c", size = 1631706, upload-time = "2024-08-02T17:49:57.606Z" }, + { url = "https://files.pythonhosted.org/packages/d9/35/31010981e4a05202a84a3116423970fd1a59d2eda4ac0b3570fbb7029ddc/black-24.8.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:3c4285573d4897a7610054af5a890bde7c65cb466040c5f0c8b732812d7f0e5e", size = 1457429, upload-time = "2024-08-02T17:49:12.764Z" }, + { url = "https://files.pythonhosted.org/packages/27/25/3f706b4f044dd569a20a4835c3b733dedea38d83d2ee0beb8178a6d44945/black-24.8.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9e84e33b37be070ba135176c123ae52a51f82306def9f7d063ee302ecab2cf47", size = 1756488, upload-time = "2024-08-02T17:46:08.067Z" }, + { url = "https://files.pythonhosted.org/packages/63/72/79375cd8277cbf1c5670914e6bd4c1b15dea2c8f8e906dc21c448d0535f0/black-24.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:73bbf84ed136e45d451a260c6b73ed674652f90a2b3211d6a35e78054563a9bb", size = 1417721, upload-time = "2024-08-02T17:46:42.637Z" }, + { url = "https://files.pythonhosted.org/packages/27/1e/83fa8a787180e1632c3d831f7e58994d7aaf23a0961320d21e84f922f919/black-24.8.0-py3-none-any.whl", hash = "sha256:972085c618ee94f402da1af548a4f218c754ea7e5dc70acb168bfaca4c2542ed", size = 206504, upload-time = "2024-08-02T17:43:15.747Z" }, +] + +[[package]] +name = "black" +version = "25.9.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "click", version = "8.1.8", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version == '3.9.*'" }, + { name = "click", version = "8.3.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.10'" }, + { name = "mypy-extensions", marker = "python_full_version >= '3.9'" }, + { name = "packaging", marker = "python_full_version >= '3.9'" }, + { name = "pathspec", marker = "python_full_version >= '3.9'" }, + { name = "platformdirs", version = "4.4.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version == '3.9.*'" }, + { name = "platformdirs", version = "4.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.10'" }, + { name = "pytokens", marker = "python_full_version >= '3.9'" }, + { name = "tomli", marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, + { name = "typing-extensions", version = "4.15.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/4b/43/20b5c90612d7bdb2bdbcceeb53d588acca3bb8f0e4c5d5c751a2c8fdd55a/black-25.9.0.tar.gz", hash = "sha256:0474bca9a0dd1b51791fcc507a4e02078a1c63f6d4e4ae5544b9848c7adfb619", size = 648393, upload-time = "2025-09-19T00:27:37.758Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/25/40/dbe31fc56b218a858c8fc6f5d8d3ba61c1fa7e989d43d4a4574b8b992840/black-25.9.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:ce41ed2614b706fd55fd0b4a6909d06b5bab344ffbfadc6ef34ae50adba3d4f7", size = 1715605, upload-time = "2025-09-19T00:36:13.483Z" }, + { url = "https://files.pythonhosted.org/packages/92/b2/f46800621200eab6479b1f4c0e3ede5b4c06b768e79ee228bc80270bcc74/black-25.9.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:2ab0ce111ef026790e9b13bd216fa7bc48edd934ffc4cbf78808b235793cbc92", size = 1571829, upload-time = "2025-09-19T00:32:42.13Z" }, + { url = "https://files.pythonhosted.org/packages/4e/64/5c7f66bd65af5c19b4ea86062bb585adc28d51d37babf70969e804dbd5c2/black-25.9.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:f96b6726d690c96c60ba682955199f8c39abc1ae0c3a494a9c62c0184049a713", size = 1631888, upload-time = "2025-09-19T00:30:54.212Z" }, + { url = "https://files.pythonhosted.org/packages/3b/64/0b9e5bfcf67db25a6eef6d9be6726499a8a72ebab3888c2de135190853d3/black-25.9.0-cp310-cp310-win_amd64.whl", hash = "sha256:d119957b37cc641596063cd7db2656c5be3752ac17877017b2ffcdb9dfc4d2b1", size = 1327056, upload-time = "2025-09-19T00:31:08.877Z" }, + { url = "https://files.pythonhosted.org/packages/b7/f4/7531d4a336d2d4ac6cc101662184c8e7d068b548d35d874415ed9f4116ef/black-25.9.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:456386fe87bad41b806d53c062e2974615825c7a52159cde7ccaeb0695fa28fa", size = 1698727, upload-time = "2025-09-19T00:31:14.264Z" }, + { url = "https://files.pythonhosted.org/packages/28/f9/66f26bfbbf84b949cc77a41a43e138d83b109502cd9c52dfc94070ca51f2/black-25.9.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:a16b14a44c1af60a210d8da28e108e13e75a284bf21a9afa6b4571f96ab8bb9d", size = 1555679, upload-time = "2025-09-19T00:31:29.265Z" }, + { url = "https://files.pythonhosted.org/packages/bf/59/61475115906052f415f518a648a9ac679d7afbc8da1c16f8fdf68a8cebed/black-25.9.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:aaf319612536d502fdd0e88ce52d8f1352b2c0a955cc2798f79eeca9d3af0608", size = 1617453, upload-time = "2025-09-19T00:30:42.24Z" }, + { url = "https://files.pythonhosted.org/packages/7f/5b/20fd5c884d14550c911e4fb1b0dae00d4abb60a4f3876b449c4d3a9141d5/black-25.9.0-cp311-cp311-win_amd64.whl", hash = "sha256:c0372a93e16b3954208417bfe448e09b0de5cc721d521866cd9e0acac3c04a1f", size = 1333655, upload-time = "2025-09-19T00:30:56.715Z" }, + { url = "https://files.pythonhosted.org/packages/fb/8e/319cfe6c82f7e2d5bfb4d3353c6cc85b523d677ff59edc61fdb9ee275234/black-25.9.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:1b9dc70c21ef8b43248f1d86aedd2aaf75ae110b958a7909ad8463c4aa0880b0", size = 1742012, upload-time = "2025-09-19T00:33:08.678Z" }, + { url = "https://files.pythonhosted.org/packages/94/cc/f562fe5d0a40cd2a4e6ae3f685e4c36e365b1f7e494af99c26ff7f28117f/black-25.9.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:8e46eecf65a095fa62e53245ae2795c90bdecabd53b50c448d0a8bcd0d2e74c4", size = 1581421, upload-time = "2025-09-19T00:35:25.937Z" }, + { url = "https://files.pythonhosted.org/packages/84/67/6db6dff1ebc8965fd7661498aea0da5d7301074b85bba8606a28f47ede4d/black-25.9.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9101ee58ddc2442199a25cb648d46ba22cd580b00ca4b44234a324e3ec7a0f7e", size = 1655619, upload-time = "2025-09-19T00:30:49.241Z" }, + { url = "https://files.pythonhosted.org/packages/10/10/3faef9aa2a730306cf469d76f7f155a8cc1f66e74781298df0ba31f8b4c8/black-25.9.0-cp312-cp312-win_amd64.whl", hash = "sha256:77e7060a00c5ec4b3367c55f39cf9b06e68965a4f2e61cecacd6d0d9b7ec945a", size = 1342481, upload-time = "2025-09-19T00:31:29.625Z" }, + { url = "https://files.pythonhosted.org/packages/48/99/3acfea65f5e79f45472c45f87ec13037b506522719cd9d4ac86484ff51ac/black-25.9.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:0172a012f725b792c358d57fe7b6b6e8e67375dd157f64fa7a3097b3ed3e2175", size = 1742165, upload-time = "2025-09-19T00:34:10.402Z" }, + { url = "https://files.pythonhosted.org/packages/3a/18/799285282c8236a79f25d590f0222dbd6850e14b060dfaa3e720241fd772/black-25.9.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:3bec74ee60f8dfef564b573a96b8930f7b6a538e846123d5ad77ba14a8d7a64f", size = 1581259, upload-time = "2025-09-19T00:32:49.685Z" }, + { url = "https://files.pythonhosted.org/packages/f1/ce/883ec4b6303acdeca93ee06b7622f1fa383c6b3765294824165d49b1a86b/black-25.9.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:b756fc75871cb1bcac5499552d771822fd9db5a2bb8db2a7247936ca48f39831", size = 1655583, upload-time = "2025-09-19T00:30:44.505Z" }, + { url = "https://files.pythonhosted.org/packages/21/17/5c253aa80a0639ccc427a5c7144534b661505ae2b5a10b77ebe13fa25334/black-25.9.0-cp313-cp313-win_amd64.whl", hash = "sha256:846d58e3ce7879ec1ffe816bb9df6d006cd9590515ed5d17db14e17666b2b357", size = 1343428, upload-time = "2025-09-19T00:32:13.839Z" }, + { url = "https://files.pythonhosted.org/packages/c4/26/0f724eb152bc9fc03029a9c903ddd77a288285042222a381050d27e64ac1/black-25.9.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:ef69351df3c84485a8beb6f7b8f9721e2009e20ef80a8d619e2d1788b7816d47", size = 1715243, upload-time = "2025-09-19T00:34:14.216Z" }, + { url = "https://files.pythonhosted.org/packages/fb/be/cb986ea2f0fabd0ee58668367724ba16c3a042842e9ebe009c139f8221c9/black-25.9.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e3c1f4cd5e93842774d9ee4ef6cd8d17790e65f44f7cdbaab5f2cf8ccf22a823", size = 1571246, upload-time = "2025-09-19T00:31:39.624Z" }, + { url = "https://files.pythonhosted.org/packages/82/ce/74cf4d66963fca33ab710e4c5817ceeff843c45649f61f41d88694c2e5db/black-25.9.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:154b06d618233fe468236ba1f0e40823d4eb08b26f5e9261526fde34916b9140", size = 1631265, upload-time = "2025-09-19T00:31:05.341Z" }, + { url = "https://files.pythonhosted.org/packages/ff/f3/9b11e001e84b4d1721f75e20b3c058854a748407e6fc1abe6da0aa22014f/black-25.9.0-cp39-cp39-win_amd64.whl", hash = "sha256:e593466de7b998374ea2585a471ba90553283fb9beefcfa430d84a2651ed5933", size = 1326615, upload-time = "2025-09-19T00:31:25.347Z" }, + { url = "https://files.pythonhosted.org/packages/1b/46/863c90dcd3f9d41b109b7f19032ae0db021f0b2a81482ba0a1e28c84de86/black-25.9.0-py3-none-any.whl", hash = "sha256:474b34c1342cdc157d307b56c4c65bce916480c4a8f6551fdc6bf9b486a7c4ae", size = 203363, upload-time = "2025-09-19T00:27:35.724Z" }, +] + +[[package]] +name = "blinker" +version = "1.8.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/1e/57/a6a1721eff09598fb01f3c7cda070c1b6a0f12d63c83236edf79a440abcc/blinker-1.8.2.tar.gz", hash = "sha256:8f77b09d3bf7c795e969e9486f39c2c5e9c39d4ee07424be2bc594ece9642d83", size = 23161, upload-time = "2024-05-06T17:04:10.101Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/bb/2a/10164ed1f31196a2f7f3799368a821765c62851ead0e630ab52b8e14b4d0/blinker-1.8.2-py3-none-any.whl", hash = "sha256:1779309f71bf239144b9399d06ae925637cf6634cf6bd131104184531bf67c01", size = 9456, upload-time = "2024-05-06T17:04:08.444Z" }, +] + +[[package]] +name = "blinker" +version = "1.9.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/21/28/9b3f50ce0e048515135495f198351908d99540d69bfdc8c1d15b73dc55ce/blinker-1.9.0.tar.gz", hash = "sha256:b4ce2265a7abece45e7cc896e98dbebe6cead56bcf805a3d23136d145f5445bf", size = 22460, upload-time = "2024-11-08T17:25:47.436Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/10/cb/f2ad4230dc2eb1a74edf38f1a38b9b52277f75bef262d8908e60d957e13c/blinker-1.9.0-py3-none-any.whl", hash = "sha256:ba0efaa9080b619ff2f3459d1d500c57bddea4a6b424b60a91141db6fd2f08bc", size = 8458, upload-time = "2024-11-08T17:25:46.184Z" }, +] + +[[package]] +name = "cachetools" +version = "5.5.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/6c/81/3747dad6b14fa2cf53fcf10548cf5aea6913e96fab41a3c198676f8948a5/cachetools-5.5.2.tar.gz", hash = "sha256:1a661caa9175d26759571b2e19580f9d6393969e5dfca11fdb1f947a23e640d4", size = 28380, upload-time = "2025-02-20T21:01:19.524Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/72/76/20fa66124dbe6be5cafeb312ece67de6b61dd91a0247d1ea13db4ebb33c2/cachetools-5.5.2-py3-none-any.whl", hash = "sha256:d26a22bcc62eb95c3beabd9f1ee5e820d3d2704fe2967cbe350e20c8ffcd3f0a", size = 10080, upload-time = "2025-02-20T21:01:16.647Z" }, +] + +[[package]] +name = "cachetools" +version = "6.2.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +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" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/4c/5b/b6ce21586237c77ce67d01dc5507039d444b630dd76611bbca2d8e5dcd91/certifi-2025.10.5.tar.gz", hash = "sha256:47c09d31ccf2acf0be3f701ea53595ee7e0b8fa08801c6624be771df09ae7b43", size = 164519, upload-time = "2025-10-05T04:12:15.808Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e4/37/af0d2ef3967ac0d6113837b44a4f0bfe1328c2b9763bd5b1744520e5cfed/certifi-2025.10.5-py3-none-any.whl", hash = "sha256:0f212c2744a9bb6de0c56639a6f68afe01ecd92d91f14ae897c4fe7bbeeef0de", size = 163286, upload-time = "2025-10-05T04:12:14.03Z" }, +] + +[[package]] +name = "charset-normalizer" +version = "3.4.4" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/13/69/33ddede1939fdd074bce5434295f38fae7136463422fe4fd3e0e89b98062/charset_normalizer-3.4.4.tar.gz", hash = "sha256:94537985111c35f28720e43603b8e7b43a6ecfb2ce1d3058bbe955b73404e21a", size = 129418, upload-time = "2025-10-14T04:42:32.879Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1f/b8/6d51fc1d52cbd52cd4ccedd5b5b2f0f6a11bbf6765c782298b0f3e808541/charset_normalizer-3.4.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:e824f1492727fa856dd6eda4f7cee25f8518a12f3c4a56a74e8095695089cf6d", size = 209709, upload-time = "2025-10-14T04:40:11.385Z" }, + { url = "https://files.pythonhosted.org/packages/5c/af/1f9d7f7faafe2ddfb6f72a2e07a548a629c61ad510fe60f9630309908fef/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4bd5d4137d500351a30687c2d3971758aac9a19208fc110ccb9d7188fbe709e8", size = 148814, upload-time = "2025-10-14T04:40:13.135Z" }, + { url = "https://files.pythonhosted.org/packages/79/3d/f2e3ac2bbc056ca0c204298ea4e3d9db9b4afe437812638759db2c976b5f/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:027f6de494925c0ab2a55eab46ae5129951638a49a34d87f4c3eda90f696b4ad", size = 144467, upload-time = "2025-10-14T04:40:14.728Z" }, + { url = "https://files.pythonhosted.org/packages/ec/85/1bf997003815e60d57de7bd972c57dc6950446a3e4ccac43bc3070721856/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f820802628d2694cb7e56db99213f930856014862f3fd943d290ea8438d07ca8", size = 162280, upload-time = "2025-10-14T04:40:16.14Z" }, + { url = "https://files.pythonhosted.org/packages/3e/8e/6aa1952f56b192f54921c436b87f2aaf7c7a7c3d0d1a765547d64fd83c13/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:798d75d81754988d2565bff1b97ba5a44411867c0cf32b77a7e8f8d84796b10d", size = 159454, upload-time = "2025-10-14T04:40:17.567Z" }, + { url = "https://files.pythonhosted.org/packages/36/3b/60cbd1f8e93aa25d1c669c649b7a655b0b5fb4c571858910ea9332678558/charset_normalizer-3.4.4-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9d1bb833febdff5c8927f922386db610b49db6e0d4f4ee29601d71e7c2694313", size = 153609, upload-time = "2025-10-14T04:40:19.08Z" }, + { url = "https://files.pythonhosted.org/packages/64/91/6a13396948b8fd3c4b4fd5bc74d045f5637d78c9675585e8e9fbe5636554/charset_normalizer-3.4.4-cp310-cp310-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:9cd98cdc06614a2f768d2b7286d66805f94c48cde050acdbbb7db2600ab3197e", size = 151849, upload-time = "2025-10-14T04:40:20.607Z" }, + { url = "https://files.pythonhosted.org/packages/b7/7a/59482e28b9981d105691e968c544cc0df3b7d6133152fb3dcdc8f135da7a/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:077fbb858e903c73f6c9db43374fd213b0b6a778106bc7032446a8e8b5b38b93", size = 151586, upload-time = "2025-10-14T04:40:21.719Z" }, + { url = "https://files.pythonhosted.org/packages/92/59/f64ef6a1c4bdd2baf892b04cd78792ed8684fbc48d4c2afe467d96b4df57/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:244bfb999c71b35de57821b8ea746b24e863398194a4014e4c76adc2bbdfeff0", size = 145290, upload-time = "2025-10-14T04:40:23.069Z" }, + { url = "https://files.pythonhosted.org/packages/6b/63/3bf9f279ddfa641ffa1962b0db6a57a9c294361cc2f5fcac997049a00e9c/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:64b55f9dce520635f018f907ff1b0df1fdc31f2795a922fb49dd14fbcdf48c84", size = 163663, upload-time = "2025-10-14T04:40:24.17Z" }, + { url = "https://files.pythonhosted.org/packages/ed/09/c9e38fc8fa9e0849b172b581fd9803bdf6e694041127933934184e19f8c3/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_riscv64.whl", hash = "sha256:faa3a41b2b66b6e50f84ae4a68c64fcd0c44355741c6374813a800cd6695db9e", size = 151964, upload-time = "2025-10-14T04:40:25.368Z" }, + { url = "https://files.pythonhosted.org/packages/d2/d1/d28b747e512d0da79d8b6a1ac18b7ab2ecfd81b2944c4c710e166d8dd09c/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:6515f3182dbe4ea06ced2d9e8666d97b46ef4c75e326b79bb624110f122551db", size = 161064, upload-time = "2025-10-14T04:40:26.806Z" }, + { url = "https://files.pythonhosted.org/packages/bb/9a/31d62b611d901c3b9e5500c36aab0ff5eb442043fb3a1c254200d3d397d9/charset_normalizer-3.4.4-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:cc00f04ed596e9dc0da42ed17ac5e596c6ccba999ba6bd92b0e0aef2f170f2d6", size = 155015, upload-time = "2025-10-14T04:40:28.284Z" }, + { url = "https://files.pythonhosted.org/packages/1f/f3/107e008fa2bff0c8b9319584174418e5e5285fef32f79d8ee6a430d0039c/charset_normalizer-3.4.4-cp310-cp310-win32.whl", hash = "sha256:f34be2938726fc13801220747472850852fe6b1ea75869a048d6f896838c896f", size = 99792, upload-time = "2025-10-14T04:40:29.613Z" }, + { url = "https://files.pythonhosted.org/packages/eb/66/e396e8a408843337d7315bab30dbf106c38966f1819f123257f5520f8a96/charset_normalizer-3.4.4-cp310-cp310-win_amd64.whl", hash = "sha256:a61900df84c667873b292c3de315a786dd8dac506704dea57bc957bd31e22c7d", size = 107198, upload-time = "2025-10-14T04:40:30.644Z" }, + { url = "https://files.pythonhosted.org/packages/b5/58/01b4f815bf0312704c267f2ccb6e5d42bcc7752340cd487bc9f8c3710597/charset_normalizer-3.4.4-cp310-cp310-win_arm64.whl", hash = "sha256:cead0978fc57397645f12578bfd2d5ea9138ea0fac82b2f63f7f7c6877986a69", size = 100262, upload-time = "2025-10-14T04:40:32.108Z" }, + { url = "https://files.pythonhosted.org/packages/ed/27/c6491ff4954e58a10f69ad90aca8a1b6fe9c5d3c6f380907af3c37435b59/charset_normalizer-3.4.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:6e1fcf0720908f200cd21aa4e6750a48ff6ce4afe7ff5a79a90d5ed8a08296f8", size = 206988, upload-time = "2025-10-14T04:40:33.79Z" }, + { url = "https://files.pythonhosted.org/packages/94/59/2e87300fe67ab820b5428580a53cad894272dbb97f38a7a814a2a1ac1011/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:5f819d5fe9234f9f82d75bdfa9aef3a3d72c4d24a6e57aeaebba32a704553aa0", size = 147324, upload-time = "2025-10-14T04:40:34.961Z" }, + { url = "https://files.pythonhosted.org/packages/07/fb/0cf61dc84b2b088391830f6274cb57c82e4da8bbc2efeac8c025edb88772/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:a59cb51917aa591b1c4e6a43c132f0cdc3c76dbad6155df4e28ee626cc77a0a3", size = 142742, upload-time = "2025-10-14T04:40:36.105Z" }, + { url = "https://files.pythonhosted.org/packages/62/8b/171935adf2312cd745d290ed93cf16cf0dfe320863ab7cbeeae1dcd6535f/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:8ef3c867360f88ac904fd3f5e1f902f13307af9052646963ee08ff4f131adafc", size = 160863, upload-time = "2025-10-14T04:40:37.188Z" }, + { url = "https://files.pythonhosted.org/packages/09/73/ad875b192bda14f2173bfc1bc9a55e009808484a4b256748d931b6948442/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d9e45d7faa48ee908174d8fe84854479ef838fc6a705c9315372eacbc2f02897", size = 157837, upload-time = "2025-10-14T04:40:38.435Z" }, + { url = "https://files.pythonhosted.org/packages/6d/fc/de9cce525b2c5b94b47c70a4b4fb19f871b24995c728e957ee68ab1671ea/charset_normalizer-3.4.4-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:840c25fb618a231545cbab0564a799f101b63b9901f2569faecd6b222ac72381", size = 151550, upload-time = "2025-10-14T04:40:40.053Z" }, + { url = "https://files.pythonhosted.org/packages/55/c2/43edd615fdfba8c6f2dfbd459b25a6b3b551f24ea21981e23fb768503ce1/charset_normalizer-3.4.4-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ca5862d5b3928c4940729dacc329aa9102900382fea192fc5e52eb69d6093815", size = 149162, upload-time = "2025-10-14T04:40:41.163Z" }, + { url = "https://files.pythonhosted.org/packages/03/86/bde4ad8b4d0e9429a4e82c1e8f5c659993a9a863ad62c7df05cf7b678d75/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d9c7f57c3d666a53421049053eaacdd14bbd0a528e2186fcb2e672effd053bb0", size = 150019, upload-time = "2025-10-14T04:40:42.276Z" }, + { url = "https://files.pythonhosted.org/packages/1f/86/a151eb2af293a7e7bac3a739b81072585ce36ccfb4493039f49f1d3cae8c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:277e970e750505ed74c832b4bf75dac7476262ee2a013f5574dd49075879e161", size = 143310, upload-time = "2025-10-14T04:40:43.439Z" }, + { url = "https://files.pythonhosted.org/packages/b5/fe/43dae6144a7e07b87478fdfc4dbe9efd5defb0e7ec29f5f58a55aeef7bf7/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:31fd66405eaf47bb62e8cd575dc621c56c668f27d46a61d975a249930dd5e2a4", size = 162022, upload-time = "2025-10-14T04:40:44.547Z" }, + { url = "https://files.pythonhosted.org/packages/80/e6/7aab83774f5d2bca81f42ac58d04caf44f0cc2b65fc6db2b3b2e8a05f3b3/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:0d3d8f15c07f86e9ff82319b3d9ef6f4bf907608f53fe9d92b28ea9ae3d1fd89", size = 149383, upload-time = "2025-10-14T04:40:46.018Z" }, + { url = "https://files.pythonhosted.org/packages/4f/e8/b289173b4edae05c0dde07f69f8db476a0b511eac556dfe0d6bda3c43384/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:9f7fcd74d410a36883701fafa2482a6af2ff5ba96b9a620e9e0721e28ead5569", size = 159098, upload-time = "2025-10-14T04:40:47.081Z" }, + { url = "https://files.pythonhosted.org/packages/d8/df/fe699727754cae3f8478493c7f45f777b17c3ef0600e28abfec8619eb49c/charset_normalizer-3.4.4-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:ebf3e58c7ec8a8bed6d66a75d7fb37b55e5015b03ceae72a8e7c74495551e224", size = 152991, upload-time = "2025-10-14T04:40:48.246Z" }, + { url = "https://files.pythonhosted.org/packages/1a/86/584869fe4ddb6ffa3bd9f491b87a01568797fb9bd8933f557dba9771beaf/charset_normalizer-3.4.4-cp311-cp311-win32.whl", hash = "sha256:eecbc200c7fd5ddb9a7f16c7decb07b566c29fa2161a16cf67b8d068bd21690a", size = 99456, upload-time = "2025-10-14T04:40:49.376Z" }, + { url = "https://files.pythonhosted.org/packages/65/f6/62fdd5feb60530f50f7e38b4f6a1d5203f4d16ff4f9f0952962c044e919a/charset_normalizer-3.4.4-cp311-cp311-win_amd64.whl", hash = "sha256:5ae497466c7901d54b639cf42d5b8c1b6a4fead55215500d2f486d34db48d016", size = 106978, upload-time = "2025-10-14T04:40:50.844Z" }, + { url = "https://files.pythonhosted.org/packages/7a/9d/0710916e6c82948b3be62d9d398cb4fcf4e97b56d6a6aeccd66c4b2f2bd5/charset_normalizer-3.4.4-cp311-cp311-win_arm64.whl", hash = "sha256:65e2befcd84bc6f37095f5961e68a6f077bf44946771354a28ad434c2cce0ae1", size = 99969, upload-time = "2025-10-14T04:40:52.272Z" }, + { url = "https://files.pythonhosted.org/packages/f3/85/1637cd4af66fa687396e757dec650f28025f2a2f5a5531a3208dc0ec43f2/charset_normalizer-3.4.4-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:0a98e6759f854bd25a58a73fa88833fba3b7c491169f86ce1180c948ab3fd394", size = 208425, upload-time = "2025-10-14T04:40:53.353Z" }, + { url = "https://files.pythonhosted.org/packages/9d/6a/04130023fef2a0d9c62d0bae2649b69f7b7d8d24ea5536feef50551029df/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b5b290ccc2a263e8d185130284f8501e3e36c5e02750fc6b6bdeb2e9e96f1e25", size = 148162, upload-time = "2025-10-14T04:40:54.558Z" }, + { url = "https://files.pythonhosted.org/packages/78/29/62328d79aa60da22c9e0b9a66539feae06ca0f5a4171ac4f7dc285b83688/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:74bb723680f9f7a6234dcf67aea57e708ec1fbdf5699fb91dfd6f511b0a320ef", size = 144558, upload-time = "2025-10-14T04:40:55.677Z" }, + { url = "https://files.pythonhosted.org/packages/86/bb/b32194a4bf15b88403537c2e120b817c61cd4ecffa9b6876e941c3ee38fe/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f1e34719c6ed0b92f418c7c780480b26b5d9c50349e9a9af7d76bf757530350d", size = 161497, upload-time = "2025-10-14T04:40:57.217Z" }, + { url = "https://files.pythonhosted.org/packages/19/89/a54c82b253d5b9b111dc74aca196ba5ccfcca8242d0fb64146d4d3183ff1/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:2437418e20515acec67d86e12bf70056a33abdacb5cb1655042f6538d6b085a8", size = 159240, upload-time = "2025-10-14T04:40:58.358Z" }, + { url = "https://files.pythonhosted.org/packages/c0/10/d20b513afe03acc89ec33948320a5544d31f21b05368436d580dec4e234d/charset_normalizer-3.4.4-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:11d694519d7f29d6cd09f6ac70028dba10f92f6cdd059096db198c283794ac86", size = 153471, upload-time = "2025-10-14T04:40:59.468Z" }, + { url = "https://files.pythonhosted.org/packages/61/fa/fbf177b55bdd727010f9c0a3c49eefa1d10f960e5f09d1d887bf93c2e698/charset_normalizer-3.4.4-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:ac1c4a689edcc530fc9d9aa11f5774b9e2f33f9a0c6a57864e90908f5208d30a", size = 150864, upload-time = "2025-10-14T04:41:00.623Z" }, + { url = "https://files.pythonhosted.org/packages/05/12/9fbc6a4d39c0198adeebbde20b619790e9236557ca59fc40e0e3cebe6f40/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:21d142cc6c0ec30d2efee5068ca36c128a30b0f2c53c1c07bd78cb6bc1d3be5f", size = 150647, upload-time = "2025-10-14T04:41:01.754Z" }, + { url = "https://files.pythonhosted.org/packages/ad/1f/6a9a593d52e3e8c5d2b167daf8c6b968808efb57ef4c210acb907c365bc4/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:5dbe56a36425d26d6cfb40ce79c314a2e4dd6211d51d6d2191c00bed34f354cc", size = 145110, upload-time = "2025-10-14T04:41:03.231Z" }, + { url = "https://files.pythonhosted.org/packages/30/42/9a52c609e72471b0fc54386dc63c3781a387bb4fe61c20231a4ebcd58bdd/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:5bfbb1b9acf3334612667b61bd3002196fe2a1eb4dd74d247e0f2a4d50ec9bbf", size = 162839, upload-time = "2025-10-14T04:41:04.715Z" }, + { url = "https://files.pythonhosted.org/packages/c4/5b/c0682bbf9f11597073052628ddd38344a3d673fda35a36773f7d19344b23/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:d055ec1e26e441f6187acf818b73564e6e6282709e9bcb5b63f5b23068356a15", size = 150667, upload-time = "2025-10-14T04:41:05.827Z" }, + { url = "https://files.pythonhosted.org/packages/e4/24/a41afeab6f990cf2daf6cb8c67419b63b48cf518e4f56022230840c9bfb2/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:af2d8c67d8e573d6de5bc30cdb27e9b95e49115cd9baad5ddbd1a6207aaa82a9", size = 160535, upload-time = "2025-10-14T04:41:06.938Z" }, + { url = "https://files.pythonhosted.org/packages/2a/e5/6a4ce77ed243c4a50a1fecca6aaaab419628c818a49434be428fe24c9957/charset_normalizer-3.4.4-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:780236ac706e66881f3b7f2f32dfe90507a09e67d1d454c762cf642e6e1586e0", size = 154816, upload-time = "2025-10-14T04:41:08.101Z" }, + { url = "https://files.pythonhosted.org/packages/a8/ef/89297262b8092b312d29cdb2517cb1237e51db8ecef2e9af5edbe7b683b1/charset_normalizer-3.4.4-cp312-cp312-win32.whl", hash = "sha256:5833d2c39d8896e4e19b689ffc198f08ea58116bee26dea51e362ecc7cd3ed26", size = 99694, upload-time = "2025-10-14T04:41:09.23Z" }, + { url = "https://files.pythonhosted.org/packages/3d/2d/1e5ed9dd3b3803994c155cd9aacb60c82c331bad84daf75bcb9c91b3295e/charset_normalizer-3.4.4-cp312-cp312-win_amd64.whl", hash = "sha256:a79cfe37875f822425b89a82333404539ae63dbdddf97f84dcbc3d339aae9525", size = 107131, upload-time = "2025-10-14T04:41:10.467Z" }, + { url = "https://files.pythonhosted.org/packages/d0/d9/0ed4c7098a861482a7b6a95603edce4c0d9db2311af23da1fb2b75ec26fc/charset_normalizer-3.4.4-cp312-cp312-win_arm64.whl", hash = "sha256:376bec83a63b8021bb5c8ea75e21c4ccb86e7e45ca4eb81146091b56599b80c3", size = 100390, upload-time = "2025-10-14T04:41:11.915Z" }, + { url = "https://files.pythonhosted.org/packages/97/45/4b3a1239bbacd321068ea6e7ac28875b03ab8bc0aa0966452db17cd36714/charset_normalizer-3.4.4-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:e1f185f86a6f3403aa2420e815904c67b2f9ebc443f045edd0de921108345794", size = 208091, upload-time = "2025-10-14T04:41:13.346Z" }, + { url = "https://files.pythonhosted.org/packages/7d/62/73a6d7450829655a35bb88a88fca7d736f9882a27eacdca2c6d505b57e2e/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b39f987ae8ccdf0d2642338faf2abb1862340facc796048b604ef14919e55ed", size = 147936, upload-time = "2025-10-14T04:41:14.461Z" }, + { url = "https://files.pythonhosted.org/packages/89/c5/adb8c8b3d6625bef6d88b251bbb0d95f8205831b987631ab0c8bb5d937c2/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:3162d5d8ce1bb98dd51af660f2121c55d0fa541b46dff7bb9b9f86ea1d87de72", size = 144180, upload-time = "2025-10-14T04:41:15.588Z" }, + { url = "https://files.pythonhosted.org/packages/91/ed/9706e4070682d1cc219050b6048bfd293ccf67b3d4f5a4f39207453d4b99/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:81d5eb2a312700f4ecaa977a8235b634ce853200e828fbadf3a9c50bab278328", size = 161346, upload-time = "2025-10-14T04:41:16.738Z" }, + { url = "https://files.pythonhosted.org/packages/d5/0d/031f0d95e4972901a2f6f09ef055751805ff541511dc1252ba3ca1f80cf5/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5bd2293095d766545ec1a8f612559f6b40abc0eb18bb2f5d1171872d34036ede", size = 158874, upload-time = "2025-10-14T04:41:17.923Z" }, + { url = "https://files.pythonhosted.org/packages/f5/83/6ab5883f57c9c801ce5e5677242328aa45592be8a00644310a008d04f922/charset_normalizer-3.4.4-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a8a8b89589086a25749f471e6a900d3f662d1d3b6e2e59dcecf787b1cc3a1894", size = 153076, upload-time = "2025-10-14T04:41:19.106Z" }, + { url = "https://files.pythonhosted.org/packages/75/1e/5ff781ddf5260e387d6419959ee89ef13878229732732ee73cdae01800f2/charset_normalizer-3.4.4-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:bc7637e2f80d8530ee4a78e878bce464f70087ce73cf7c1caf142416923b98f1", size = 150601, upload-time = "2025-10-14T04:41:20.245Z" }, + { url = "https://files.pythonhosted.org/packages/d7/57/71be810965493d3510a6ca79b90c19e48696fb1ff964da319334b12677f0/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:f8bf04158c6b607d747e93949aa60618b61312fe647a6369f88ce2ff16043490", size = 150376, upload-time = "2025-10-14T04:41:21.398Z" }, + { url = "https://files.pythonhosted.org/packages/e5/d5/c3d057a78c181d007014feb7e9f2e65905a6c4ef182c0ddf0de2924edd65/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:554af85e960429cf30784dd47447d5125aaa3b99a6f0683589dbd27e2f45da44", size = 144825, upload-time = "2025-10-14T04:41:22.583Z" }, + { url = "https://files.pythonhosted.org/packages/e6/8c/d0406294828d4976f275ffbe66f00266c4b3136b7506941d87c00cab5272/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:74018750915ee7ad843a774364e13a3db91682f26142baddf775342c3f5b1133", size = 162583, upload-time = "2025-10-14T04:41:23.754Z" }, + { url = "https://files.pythonhosted.org/packages/d7/24/e2aa1f18c8f15c4c0e932d9287b8609dd30ad56dbe41d926bd846e22fb8d/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:c0463276121fdee9c49b98908b3a89c39be45d86d1dbaa22957e38f6321d4ce3", size = 150366, upload-time = "2025-10-14T04:41:25.27Z" }, + { url = "https://files.pythonhosted.org/packages/e4/5b/1e6160c7739aad1e2df054300cc618b06bf784a7a164b0f238360721ab86/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:362d61fd13843997c1c446760ef36f240cf81d3ebf74ac62652aebaf7838561e", size = 160300, upload-time = "2025-10-14T04:41:26.725Z" }, + { url = "https://files.pythonhosted.org/packages/7a/10/f882167cd207fbdd743e55534d5d9620e095089d176d55cb22d5322f2afd/charset_normalizer-3.4.4-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:9a26f18905b8dd5d685d6d07b0cdf98a79f3c7a918906af7cc143ea2e164c8bc", size = 154465, upload-time = "2025-10-14T04:41:28.322Z" }, + { url = "https://files.pythonhosted.org/packages/89/66/c7a9e1b7429be72123441bfdbaf2bc13faab3f90b933f664db506dea5915/charset_normalizer-3.4.4-cp313-cp313-win32.whl", hash = "sha256:9b35f4c90079ff2e2edc5b26c0c77925e5d2d255c42c74fdb70fb49b172726ac", size = 99404, upload-time = "2025-10-14T04:41:29.95Z" }, + { url = "https://files.pythonhosted.org/packages/c4/26/b9924fa27db384bdcd97ab83b4f0a8058d96ad9626ead570674d5e737d90/charset_normalizer-3.4.4-cp313-cp313-win_amd64.whl", hash = "sha256:b435cba5f4f750aa6c0a0d92c541fb79f69a387c91e61f1795227e4ed9cece14", size = 107092, upload-time = "2025-10-14T04:41:31.188Z" }, + { url = "https://files.pythonhosted.org/packages/af/8f/3ed4bfa0c0c72a7ca17f0380cd9e4dd842b09f664e780c13cff1dcf2ef1b/charset_normalizer-3.4.4-cp313-cp313-win_arm64.whl", hash = "sha256:542d2cee80be6f80247095cc36c418f7bddd14f4a6de45af91dfad36d817bba2", size = 100408, upload-time = "2025-10-14T04:41:32.624Z" }, + { url = "https://files.pythonhosted.org/packages/2a/35/7051599bd493e62411d6ede36fd5af83a38f37c4767b92884df7301db25d/charset_normalizer-3.4.4-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:da3326d9e65ef63a817ecbcc0df6e94463713b754fe293eaa03da99befb9a5bd", size = 207746, upload-time = "2025-10-14T04:41:33.773Z" }, + { url = "https://files.pythonhosted.org/packages/10/9a/97c8d48ef10d6cd4fcead2415523221624bf58bcf68a802721a6bc807c8f/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:8af65f14dc14a79b924524b1e7fffe304517b2bff5a58bf64f30b98bbc5079eb", size = 147889, upload-time = "2025-10-14T04:41:34.897Z" }, + { url = "https://files.pythonhosted.org/packages/10/bf/979224a919a1b606c82bd2c5fa49b5c6d5727aa47b4312bb27b1734f53cd/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:74664978bb272435107de04e36db5a9735e78232b85b77d45cfb38f758efd33e", size = 143641, upload-time = "2025-10-14T04:41:36.116Z" }, + { url = "https://files.pythonhosted.org/packages/ba/33/0ad65587441fc730dc7bd90e9716b30b4702dc7b617e6ba4997dc8651495/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:752944c7ffbfdd10c074dc58ec2d5a8a4cd9493b314d367c14d24c17684ddd14", size = 160779, upload-time = "2025-10-14T04:41:37.229Z" }, + { url = "https://files.pythonhosted.org/packages/67/ed/331d6b249259ee71ddea93f6f2f0a56cfebd46938bde6fcc6f7b9a3d0e09/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d1f13550535ad8cff21b8d757a3257963e951d96e20ec82ab44bc64aeb62a191", size = 159035, upload-time = "2025-10-14T04:41:38.368Z" }, + { url = "https://files.pythonhosted.org/packages/67/ff/f6b948ca32e4f2a4576aa129d8bed61f2e0543bf9f5f2b7fc3758ed005c9/charset_normalizer-3.4.4-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ecaae4149d99b1c9e7b88bb03e3221956f68fd6d50be2ef061b2381b61d20838", size = 152542, upload-time = "2025-10-14T04:41:39.862Z" }, + { url = "https://files.pythonhosted.org/packages/16/85/276033dcbcc369eb176594de22728541a925b2632f9716428c851b149e83/charset_normalizer-3.4.4-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:cb6254dc36b47a990e59e1068afacdcd02958bdcce30bb50cc1700a8b9d624a6", size = 149524, upload-time = "2025-10-14T04:41:41.319Z" }, + { url = "https://files.pythonhosted.org/packages/9e/f2/6a2a1f722b6aba37050e626530a46a68f74e63683947a8acff92569f979a/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:c8ae8a0f02f57a6e61203a31428fa1d677cbe50c93622b4149d5c0f319c1d19e", size = 150395, upload-time = "2025-10-14T04:41:42.539Z" }, + { url = "https://files.pythonhosted.org/packages/60/bb/2186cb2f2bbaea6338cad15ce23a67f9b0672929744381e28b0592676824/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:47cc91b2f4dd2833fddaedd2893006b0106129d4b94fdb6af1f4ce5a9965577c", size = 143680, upload-time = "2025-10-14T04:41:43.661Z" }, + { url = "https://files.pythonhosted.org/packages/7d/a5/bf6f13b772fbb2a90360eb620d52ed8f796f3c5caee8398c3b2eb7b1c60d/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:82004af6c302b5d3ab2cfc4cc5f29db16123b1a8417f2e25f9066f91d4411090", size = 162045, upload-time = "2025-10-14T04:41:44.821Z" }, + { url = "https://files.pythonhosted.org/packages/df/c5/d1be898bf0dc3ef9030c3825e5d3b83f2c528d207d246cbabe245966808d/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:2b7d8f6c26245217bd2ad053761201e9f9680f8ce52f0fcd8d0755aeae5b2152", size = 149687, upload-time = "2025-10-14T04:41:46.442Z" }, + { url = "https://files.pythonhosted.org/packages/a5/42/90c1f7b9341eef50c8a1cb3f098ac43b0508413f33affd762855f67a410e/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:799a7a5e4fb2d5898c60b640fd4981d6a25f1c11790935a44ce38c54e985f828", size = 160014, upload-time = "2025-10-14T04:41:47.631Z" }, + { url = "https://files.pythonhosted.org/packages/76/be/4d3ee471e8145d12795ab655ece37baed0929462a86e72372fd25859047c/charset_normalizer-3.4.4-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:99ae2cffebb06e6c22bdc25801d7b30f503cc87dbd283479e7b606f70aff57ec", size = 154044, upload-time = "2025-10-14T04:41:48.81Z" }, + { url = "https://files.pythonhosted.org/packages/b0/6f/8f7af07237c34a1defe7defc565a9bc1807762f672c0fde711a4b22bf9c0/charset_normalizer-3.4.4-cp314-cp314-win32.whl", hash = "sha256:f9d332f8c2a2fcbffe1378594431458ddbef721c1769d78e2cbc06280d8155f9", size = 99940, upload-time = "2025-10-14T04:41:49.946Z" }, + { url = "https://files.pythonhosted.org/packages/4b/51/8ade005e5ca5b0d80fb4aff72a3775b325bdc3d27408c8113811a7cbe640/charset_normalizer-3.4.4-cp314-cp314-win_amd64.whl", hash = "sha256:8a6562c3700cce886c5be75ade4a5db4214fda19fede41d9792d100288d8f94c", size = 107104, upload-time = "2025-10-14T04:41:51.051Z" }, + { url = "https://files.pythonhosted.org/packages/da/5f/6b8f83a55bb8278772c5ae54a577f3099025f9ade59d0136ac24a0df4bde/charset_normalizer-3.4.4-cp314-cp314-win_arm64.whl", hash = "sha256:de00632ca48df9daf77a2c65a484531649261ec9f25489917f09e455cb09ddb2", size = 100743, upload-time = "2025-10-14T04:41:52.122Z" }, + { url = "https://files.pythonhosted.org/packages/0a/4e/3926a1c11f0433791985727965263f788af00db3482d89a7545ca5ecc921/charset_normalizer-3.4.4-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:ce8a0633f41a967713a59c4139d29110c07e826d131a316b50ce11b1d79b4f84", size = 198599, upload-time = "2025-10-14T04:41:53.213Z" }, + { url = "https://files.pythonhosted.org/packages/ec/7c/b92d1d1dcffc34592e71ea19c882b6709e43d20fa498042dea8b815638d7/charset_normalizer-3.4.4-cp38-cp38-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:eaabd426fe94daf8fd157c32e571c85cb12e66692f15516a83a03264b08d06c3", size = 143090, upload-time = "2025-10-14T04:41:54.385Z" }, + { url = "https://files.pythonhosted.org/packages/84/ce/61a28d3bb77281eb24107b937a497f3c43089326d27832a63dcedaab0478/charset_normalizer-3.4.4-cp38-cp38-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:c4ef880e27901b6cc782f1b95f82da9313c0eb95c3af699103088fa0ac3ce9ac", size = 139490, upload-time = "2025-10-14T04:41:55.551Z" }, + { url = "https://files.pythonhosted.org/packages/c0/bd/c9e59a91b2061c6f8bb98a150670cb16d4cd7c4ba7d11ad0cdf789155f41/charset_normalizer-3.4.4-cp38-cp38-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:2aaba3b0819274cc41757a1da876f810a3e4d7b6eb25699253a4effef9e8e4af", size = 155334, upload-time = "2025-10-14T04:41:56.724Z" }, + { url = "https://files.pythonhosted.org/packages/bf/37/f17ae176a80f22ff823456af91ba3bc59df308154ff53aef0d39eb3d3419/charset_normalizer-3.4.4-cp38-cp38-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:778d2e08eda00f4256d7f672ca9fef386071c9202f5e4607920b86d7803387f2", size = 152823, upload-time = "2025-10-14T04:41:58.236Z" }, + { url = "https://files.pythonhosted.org/packages/bf/fa/cf5bb2409a385f78750e78c8d2e24780964976acdaaed65dbd6083ae5b40/charset_normalizer-3.4.4-cp38-cp38-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:f155a433c2ec037d4e8df17d18922c3a0d9b3232a396690f17175d2946f0218d", size = 147618, upload-time = "2025-10-14T04:41:59.409Z" }, + { url = "https://files.pythonhosted.org/packages/9b/63/579784a65bc7de2d4518d40bb8f1870900163e86f17f21fd1384318c459d/charset_normalizer-3.4.4-cp38-cp38-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:a8bf8d0f749c5757af2142fe7903a9df1d2e8aa3841559b2bad34b08d0e2bcf3", size = 145516, upload-time = "2025-10-14T04:42:00.579Z" }, + { url = "https://files.pythonhosted.org/packages/a3/a9/94ec6266cd394e8f93a4d69cca651d61bf6ac58d2a0422163b30c698f2c7/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:194f08cbb32dc406d6e1aea671a68be0823673db2832b38405deba2fb0d88f63", size = 145266, upload-time = "2025-10-14T04:42:01.684Z" }, + { url = "https://files.pythonhosted.org/packages/09/14/d6626eb97764b58c2779fa7928fa7d1a49adb8ce687c2dbba4db003c1939/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_armv7l.whl", hash = "sha256:6aee717dcfead04c6eb1ce3bd29ac1e22663cdea57f943c87d1eab9a025438d7", size = 139559, upload-time = "2025-10-14T04:42:02.902Z" }, + { url = "https://files.pythonhosted.org/packages/09/01/ddbe6b01313ba191dbb0a43c7563bc770f2448c18127f9ea4b119c44dff0/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:cd4b7ca9984e5e7985c12bc60a6f173f3c958eae74f3ef6624bb6b26e2abbae4", size = 156653, upload-time = "2025-10-14T04:42:04.005Z" }, + { url = "https://files.pythonhosted.org/packages/95/c8/d05543378bea89296e9af4510b44c704626e191da447235c8fdedfc5b7b2/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_riscv64.whl", hash = "sha256:b7cf1017d601aa35e6bb650b6ad28652c9cd78ee6caff19f3c28d03e1c80acbf", size = 145644, upload-time = "2025-10-14T04:42:05.211Z" }, + { url = "https://files.pythonhosted.org/packages/72/01/2866c4377998ef8a1f6802f6431e774a4c8ebe75b0a6e569ceec55c9cbfb/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:e912091979546adf63357d7e2ccff9b44f026c075aeaf25a52d0e95ad2281074", size = 153964, upload-time = "2025-10-14T04:42:06.341Z" }, + { url = "https://files.pythonhosted.org/packages/4a/66/66c72468a737b4cbd7851ba2c522fe35c600575fbeac944460b4fd4a06fe/charset_normalizer-3.4.4-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:5cb4d72eea50c8868f5288b7f7f33ed276118325c1dfd3957089f6b519e1382a", size = 148777, upload-time = "2025-10-14T04:42:07.535Z" }, + { url = "https://files.pythonhosted.org/packages/50/94/d0d56677fdddbffa8ca00ec411f67bb8c947f9876374ddc9d160d4f2c4b3/charset_normalizer-3.4.4-cp38-cp38-win32.whl", hash = "sha256:837c2ce8c5a65a2035be9b3569c684358dfbf109fd3b6969630a87535495ceaa", size = 98687, upload-time = "2025-10-14T04:42:08.678Z" }, + { url = "https://files.pythonhosted.org/packages/00/64/c3bc303d1b586480b1c8e6e1e2191a6d6dd40255244e5cf16763dcec52e6/charset_normalizer-3.4.4-cp38-cp38-win_amd64.whl", hash = "sha256:44c2a8734b333e0578090c4cd6b16f275e07aa6614ca8715e6c038e865e70576", size = 106115, upload-time = "2025-10-14T04:42:09.793Z" }, + { url = "https://files.pythonhosted.org/packages/46/7c/0c4760bccf082737ca7ab84a4c2034fcc06b1f21cf3032ea98bd6feb1725/charset_normalizer-3.4.4-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:a9768c477b9d7bd54bc0c86dbaebdec6f03306675526c9927c0e8a04e8f94af9", size = 209609, upload-time = "2025-10-14T04:42:10.922Z" }, + { url = "https://files.pythonhosted.org/packages/bb/a4/69719daef2f3d7f1819de60c9a6be981b8eeead7542d5ec4440f3c80e111/charset_normalizer-3.4.4-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1bee1e43c28aa63cb16e5c14e582580546b08e535299b8b6158a7c9c768a1f3d", size = 149029, upload-time = "2025-10-14T04:42:12.38Z" }, + { url = "https://files.pythonhosted.org/packages/e6/21/8d4e1d6c1e6070d3672908b8e4533a71b5b53e71d16828cc24d0efec564c/charset_normalizer-3.4.4-cp39-cp39-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:fd44c878ea55ba351104cb93cc85e74916eb8fa440ca7903e57575e97394f608", size = 144580, upload-time = "2025-10-14T04:42:13.549Z" }, + { url = "https://files.pythonhosted.org/packages/a7/0a/a616d001b3f25647a9068e0b9199f697ce507ec898cacb06a0d5a1617c99/charset_normalizer-3.4.4-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:0f04b14ffe5fdc8c4933862d8306109a2c51e0704acfa35d51598eb45a1e89fc", size = 162340, upload-time = "2025-10-14T04:42:14.892Z" }, + { url = "https://files.pythonhosted.org/packages/85/93/060b52deb249a5450460e0585c88a904a83aec474ab8e7aba787f45e79f2/charset_normalizer-3.4.4-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:cd09d08005f958f370f539f186d10aec3377d55b9eeb0d796025d4886119d76e", size = 159619, upload-time = "2025-10-14T04:42:16.676Z" }, + { url = "https://files.pythonhosted.org/packages/dd/21/0274deb1cc0632cd587a9a0ec6b4674d9108e461cb4cd40d457adaeb0564/charset_normalizer-3.4.4-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4fe7859a4e3e8457458e2ff592f15ccb02f3da787fcd31e0183879c3ad4692a1", size = 153980, upload-time = "2025-10-14T04:42:17.917Z" }, + { url = "https://files.pythonhosted.org/packages/28/2b/e3d7d982858dccc11b31906976323d790dded2017a0572f093ff982d692f/charset_normalizer-3.4.4-cp39-cp39-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:fa09f53c465e532f4d3db095e0c55b615f010ad81803d383195b6b5ca6cbf5f3", size = 152174, upload-time = "2025-10-14T04:42:19.018Z" }, + { url = "https://files.pythonhosted.org/packages/6e/ff/4a269f8e35f1e58b2df52c131a1fa019acb7ef3f8697b7d464b07e9b492d/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:7fa17817dc5625de8a027cb8b26d9fefa3ea28c8253929b8d6649e705d2835b6", size = 151666, upload-time = "2025-10-14T04:42:20.171Z" }, + { url = "https://files.pythonhosted.org/packages/da/c9/ec39870f0b330d58486001dd8e532c6b9a905f5765f58a6f8204926b4a93/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:5947809c8a2417be3267efc979c47d76a079758166f7d43ef5ae8e9f92751f88", size = 145550, upload-time = "2025-10-14T04:42:21.324Z" }, + { url = "https://files.pythonhosted.org/packages/75/8f/d186ab99e40e0ed9f82f033d6e49001701c81244d01905dd4a6924191a30/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:4902828217069c3c5c71094537a8e623f5d097858ac6ca8252f7b4d10b7560f1", size = 163721, upload-time = "2025-10-14T04:42:22.46Z" }, + { url = "https://files.pythonhosted.org/packages/96/b1/6047663b9744df26a7e479ac1e77af7134b1fcf9026243bb48ee2d18810f/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_riscv64.whl", hash = "sha256:7c308f7e26e4363d79df40ca5b2be1c6ba9f02bdbccfed5abddb7859a6ce72cf", size = 152127, upload-time = "2025-10-14T04:42:23.712Z" }, + { url = "https://files.pythonhosted.org/packages/59/78/e5a6eac9179f24f704d1be67d08704c3c6ab9f00963963524be27c18ed87/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:2c9d3c380143a1fedbff95a312aa798578371eb29da42106a29019368a475318", size = 161175, upload-time = "2025-10-14T04:42:24.87Z" }, + { url = "https://files.pythonhosted.org/packages/e5/43/0e626e42d54dd2f8dd6fc5e1c5ff00f05fbca17cb699bedead2cae69c62f/charset_normalizer-3.4.4-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:cb01158d8b88ee68f15949894ccc6712278243d95f344770fa7593fa2d94410c", size = 155375, upload-time = "2025-10-14T04:42:27.246Z" }, + { url = "https://files.pythonhosted.org/packages/e9/91/d9615bf2e06f35e4997616ff31248c3657ed649c5ab9d35ea12fce54e380/charset_normalizer-3.4.4-cp39-cp39-win32.whl", hash = "sha256:2677acec1a2f8ef614c6888b5b4ae4060cc184174a938ed4e8ef690e15d3e505", size = 99692, upload-time = "2025-10-14T04:42:28.425Z" }, + { url = "https://files.pythonhosted.org/packages/d1/a9/6c040053909d9d1ef4fcab45fddec083aedc9052c10078339b47c8573ea8/charset_normalizer-3.4.4-cp39-cp39-win_amd64.whl", hash = "sha256:f8e160feb2aed042cd657a72acc0b481212ed28b1b9a95c0cee1621b524e1966", size = 107192, upload-time = "2025-10-14T04:42:29.482Z" }, + { url = "https://files.pythonhosted.org/packages/f0/c6/4fa536b2c0cd3edfb7ccf8469fa0f363ea67b7213a842b90909ca33dd851/charset_normalizer-3.4.4-cp39-cp39-win_arm64.whl", hash = "sha256:b5d84d37db046c5ca74ee7bb47dd6cbc13f80665fdde3e8040bdd3fb015ecb50", size = 100220, upload-time = "2025-10-14T04:42:30.632Z" }, + { url = "https://files.pythonhosted.org/packages/0a/4c/925909008ed5a988ccbb72dcc897407e5d6d3bd72410d69e051fc0c14647/charset_normalizer-3.4.4-py3-none-any.whl", hash = "sha256:7a32c560861a02ff789ad905a2fe94e3f840803362c84fecf1851cb4cf3dc37f", size = 53402, upload-time = "2025-10-14T04:42:31.76Z" }, +] + +[[package]] +name = "click" +version = "8.1.8" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version == '3.9.*'", + "python_full_version < '3.9'", +] +dependencies = [ + { name = "colorama", marker = "python_full_version < '3.10' and sys_platform == 'win32'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/b9/2e/0090cbf739cee7d23781ad4b89a9894a41538e4fcf4c31dcdd705b78eb8b/click-8.1.8.tar.gz", hash = "sha256:ed53c9d8990d83c2a27deae68e4ee337473f6330c040a31d4225c9574d16096a", size = 226593, upload-time = "2024-12-21T18:38:44.339Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/7e/d4/7ebdbd03970677812aac39c869717059dbb71a4cfc033ca6e5221787892c/click-8.1.8-py3-none-any.whl", hash = "sha256:63c132bbbed01578a06712a2d1f497bb62d9c1c0d329b7903a866228027263b2", size = 98188, upload-time = "2024-12-21T18:38:41.666Z" }, +] + +[[package]] +name = "click" +version = "8.3.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", +] +dependencies = [ + { name = "colorama", marker = "python_full_version >= '3.10' and sys_platform == 'win32'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/46/61/de6cd827efad202d7057d93e0fed9294b96952e188f7384832791c7b2254/click-8.3.0.tar.gz", hash = "sha256:e7b8232224eba16f4ebe410c25ced9f7875cb5f3263ffc93cc3e8da705e229c4", size = 276943, upload-time = "2025-09-18T17:32:23.696Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/db/d3/9dcc0f5797f070ec8edf30fbadfb200e71d9db6b84d211e3b2085a7589a0/click-8.3.0-py3-none-any.whl", hash = "sha256:9b9f285302c6e3064f4330c05f05b81945b2a39544279343e6e7c5f27a9baddc", size = 107295, upload-time = "2025-09-18T17:32:22.42Z" }, +] + +[[package]] +name = "colorama" +version = "0.4.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d8/53/6f443c9a4a8358a93a6792e2acffb9d9d5cb0a5cfd8802644b7b1c9a02e4/colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44", size = 27697, upload-time = "2022-10-25T02:36:22.414Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/d6/3965ed04c63042e047cb6a3e6ed1a63a35087b6a609aa3a15ed8ac56c221/colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6", size = 25335, upload-time = "2022-10-25T02:36:20.889Z" }, +] + +[[package]] +name = "databricks-sdk" +version = "0.70.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "google-auth" }, + { name = "protobuf", version = "5.29.5", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "protobuf", version = "6.33.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "requests", version = "2.32.4", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "requests", version = "2.32.5", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/0e/c0/7bca00fcf265bc1fc8ac9452f8fc80779ca56225e11ffce5fbbcd1b47e17/databricks_sdk-0.70.0.tar.gz", hash = "sha256:a4e2141972a5aebca7f4cda0a8e7e3ea444d150fea9bb28fcbd1746e62f65735", size = 798157, upload-time = "2025-10-23T13:44:18.217Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/45/b9/202b3ff6f2c53736aa45b68870da0de1226e1abe15fc3f2222278cb8193c/databricks_sdk-0.70.0-py3-none-any.whl", hash = "sha256:f573d76cd6960d390253929950210145e9175242196c6f192facd8ea00bc91f2", size = 752568, upload-time = "2025-10-23T13:44:16.474Z" }, +] + +[[package]] +name = "databricks-slack-bot" +version = "0.1.0" +source = { editable = "." } +dependencies = [ + { name = "aiohttp", version = "3.10.11", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "aiohttp", version = "3.13.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "databricks-sdk" }, + { name = "flask", version = "3.0.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "flask", version = "3.1.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "httpx" }, + { name = "python-dotenv", version = "1.0.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "python-dotenv", version = "1.1.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "slack-bolt" }, + { name = "slack-sdk" }, +] + +[package.optional-dependencies] +dev = [ + { name = "black", version = "24.8.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "black", version = "25.9.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "pytest", version = "8.3.5", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "pytest", version = "8.4.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "ruff" }, +] + +[package.metadata] +requires-dist = [ + { name = "aiohttp", specifier = ">=3.9.0" }, + { name = "black", marker = "extra == 'dev'", specifier = ">=23.0.0" }, + { name = "databricks-sdk", specifier = ">=0.18.0" }, + { name = "flask", specifier = ">=3.0.0" }, + { name = "httpx", specifier = ">=0.25.0" }, + { name = "pytest", marker = "extra == 'dev'", specifier = ">=7.0.0" }, + { name = "python-dotenv", specifier = ">=1.0.0" }, + { name = "ruff", marker = "extra == 'dev'", specifier = ">=0.1.0" }, + { name = "slack-bolt", specifier = ">=1.18.0" }, + { name = "slack-sdk", specifier = ">=3.23.0" }, +] +provides-extras = ["dev"] + +[[package]] +name = "exceptiongroup" +version = "1.3.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "typing-extensions", version = "4.13.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "typing-extensions", version = "4.15.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9' and python_full_version < '3.13'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/0b/9f/a65090624ecf468cdca03533906e7c69ed7588582240cfe7cc9e770b50eb/exceptiongroup-1.3.0.tar.gz", hash = "sha256:b241f5885f560bc56a59ee63ca4c6a8bfa46ae4ad651af316d4e81817bb9fd88", size = 29749, upload-time = "2025-05-10T17:42:51.123Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/36/f4/c6e662dade71f56cd2f3735141b265c3c79293c109549c1e6933b0651ffc/exceptiongroup-1.3.0-py3-none-any.whl", hash = "sha256:4d111e6e0c13d0644cad6ddaa7ed0261a0b36971f6d23e7ec9b4b9097da78a10", size = 16674, upload-time = "2025-05-10T17:42:49.33Z" }, +] + +[[package]] +name = "flask" +version = "3.0.3" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "blinker", version = "1.8.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "click", version = "8.1.8", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "importlib-metadata", version = "8.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "itsdangerous", marker = "python_full_version < '3.9'" }, + { name = "jinja2", marker = "python_full_version < '3.9'" }, + { name = "werkzeug", version = "3.0.6", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/41/e1/d104c83026f8d35dfd2c261df7d64738341067526406b40190bc063e829a/flask-3.0.3.tar.gz", hash = "sha256:ceb27b0af3823ea2737928a4d99d125a06175b8512c445cbd9a9ce200ef76842", size = 676315, upload-time = "2024-04-07T19:26:11.035Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/61/80/ffe1da13ad9300f87c93af113edd0638c75138c42a0994becfacac078c06/flask-3.0.3-py3-none-any.whl", hash = "sha256:34e815dfaa43340d1d15a5c3a02b8476004037eb4840b34910c6e21679d288f3", size = 101735, upload-time = "2024-04-07T19:26:08.569Z" }, +] + +[[package]] +name = "flask" +version = "3.1.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "blinker", version = "1.9.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "click", version = "8.1.8", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version == '3.9.*'" }, + { name = "click", version = "8.3.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.10'" }, + { name = "importlib-metadata", version = "8.7.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version == '3.9.*'" }, + { name = "itsdangerous", marker = "python_full_version >= '3.9'" }, + { name = "jinja2", marker = "python_full_version >= '3.9'" }, + { name = "markupsafe", version = "3.0.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "werkzeug", version = "3.1.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/dc/6d/cfe3c0fcc5e477df242b98bfe186a4c34357b4847e87ecaef04507332dab/flask-3.1.2.tar.gz", hash = "sha256:bf656c15c80190ed628ad08cdfd3aaa35beb087855e2f494910aa3774cc4fd87", size = 720160, upload-time = "2025-08-19T21:03:21.205Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ec/f9/7f9263c5695f4bd0023734af91bedb2ff8209e8de6ead162f35d8dc762fd/flask-3.1.2-py3-none-any.whl", hash = "sha256:ca1d8112ec8a6158cc29ea4858963350011b5c846a414cdb7a954aa9e967d03c", size = 103308, upload-time = "2025-08-19T21:03:19.499Z" }, +] + +[[package]] +name = "frozenlist" +version = "1.5.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/8f/ed/0f4cec13a93c02c47ec32d81d11c0c1efbadf4a471e3f3ce7cad366cbbd3/frozenlist-1.5.0.tar.gz", hash = "sha256:81d5af29e61b9c8348e876d442253723928dce6433e0e76cd925cd83f1b4b817", size = 39930, upload-time = "2024-10-23T09:48:29.903Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/54/79/29d44c4af36b2b240725dce566b20f63f9b36ef267aaaa64ee7466f4f2f8/frozenlist-1.5.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:5b6a66c18b5b9dd261ca98dffcb826a525334b2f29e7caa54e182255c5f6a65a", size = 94451, upload-time = "2024-10-23T09:46:20.558Z" }, + { url = "https://files.pythonhosted.org/packages/47/47/0c999aeace6ead8a44441b4f4173e2261b18219e4ad1fe9a479871ca02fc/frozenlist-1.5.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d1b3eb7b05ea246510b43a7e53ed1653e55c2121019a97e60cad7efb881a97bb", size = 54301, upload-time = "2024-10-23T09:46:21.759Z" }, + { url = "https://files.pythonhosted.org/packages/8d/60/107a38c1e54176d12e06e9d4b5d755b677d71d1219217cee063911b1384f/frozenlist-1.5.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:15538c0cbf0e4fa11d1e3a71f823524b0c46299aed6e10ebb4c2089abd8c3bec", size = 52213, upload-time = "2024-10-23T09:46:22.993Z" }, + { url = "https://files.pythonhosted.org/packages/17/62/594a6829ac5679c25755362a9dc93486a8a45241394564309641425d3ff6/frozenlist-1.5.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e79225373c317ff1e35f210dd5f1344ff31066ba8067c307ab60254cd3a78ad5", size = 240946, upload-time = "2024-10-23T09:46:24.661Z" }, + { url = "https://files.pythonhosted.org/packages/7e/75/6c8419d8f92c80dd0ee3f63bdde2702ce6398b0ac8410ff459f9b6f2f9cb/frozenlist-1.5.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9272fa73ca71266702c4c3e2d4a28553ea03418e591e377a03b8e3659d94fa76", size = 264608, upload-time = "2024-10-23T09:46:26.017Z" }, + { url = "https://files.pythonhosted.org/packages/88/3e/82a6f0b84bc6fb7e0be240e52863c6d4ab6098cd62e4f5b972cd31e002e8/frozenlist-1.5.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:498524025a5b8ba81695761d78c8dd7382ac0b052f34e66939c42df860b8ff17", size = 261361, upload-time = "2024-10-23T09:46:27.787Z" }, + { url = "https://files.pythonhosted.org/packages/fd/85/14e5f9ccac1b64ff2f10c927b3ffdf88772aea875882406f9ba0cec8ad84/frozenlist-1.5.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:92b5278ed9d50fe610185ecd23c55d8b307d75ca18e94c0e7de328089ac5dcba", size = 231649, upload-time = "2024-10-23T09:46:28.992Z" }, + { url = "https://files.pythonhosted.org/packages/ee/59/928322800306f6529d1852323014ee9008551e9bb027cc38d276cbc0b0e7/frozenlist-1.5.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f3c8c1dacd037df16e85227bac13cca58c30da836c6f936ba1df0c05d046d8d", size = 241853, upload-time = "2024-10-23T09:46:30.211Z" }, + { url = "https://files.pythonhosted.org/packages/7d/bd/e01fa4f146a6f6c18c5d34cab8abdc4013774a26c4ff851128cd1bd3008e/frozenlist-1.5.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:f2ac49a9bedb996086057b75bf93538240538c6d9b38e57c82d51f75a73409d2", size = 243652, upload-time = "2024-10-23T09:46:31.758Z" }, + { url = "https://files.pythonhosted.org/packages/a5/bd/e4771fd18a8ec6757033f0fa903e447aecc3fbba54e3630397b61596acf0/frozenlist-1.5.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:e66cc454f97053b79c2ab09c17fbe3c825ea6b4de20baf1be28919460dd7877f", size = 241734, upload-time = "2024-10-23T09:46:33.044Z" }, + { url = "https://files.pythonhosted.org/packages/21/13/c83821fa5544af4f60c5d3a65d054af3213c26b14d3f5f48e43e5fb48556/frozenlist-1.5.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:5a3ba5f9a0dfed20337d3e966dc359784c9f96503674c2faf015f7fe8e96798c", size = 260959, upload-time = "2024-10-23T09:46:34.916Z" }, + { url = "https://files.pythonhosted.org/packages/71/f3/1f91c9a9bf7ed0e8edcf52698d23f3c211d8d00291a53c9f115ceb977ab1/frozenlist-1.5.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:6321899477db90bdeb9299ac3627a6a53c7399c8cd58d25da094007402b039ab", size = 262706, upload-time = "2024-10-23T09:46:36.159Z" }, + { url = "https://files.pythonhosted.org/packages/4c/22/4a256fdf5d9bcb3ae32622c796ee5ff9451b3a13a68cfe3f68e2c95588ce/frozenlist-1.5.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:76e4753701248476e6286f2ef492af900ea67d9706a0155335a40ea21bf3b2f5", size = 250401, upload-time = "2024-10-23T09:46:37.327Z" }, + { url = "https://files.pythonhosted.org/packages/af/89/c48ebe1f7991bd2be6d5f4ed202d94960c01b3017a03d6954dd5fa9ea1e8/frozenlist-1.5.0-cp310-cp310-win32.whl", hash = "sha256:977701c081c0241d0955c9586ffdd9ce44f7a7795df39b9151cd9a6fd0ce4cfb", size = 45498, upload-time = "2024-10-23T09:46:38.552Z" }, + { url = "https://files.pythonhosted.org/packages/28/2f/cc27d5f43e023d21fe5c19538e08894db3d7e081cbf582ad5ed366c24446/frozenlist-1.5.0-cp310-cp310-win_amd64.whl", hash = "sha256:189f03b53e64144f90990d29a27ec4f7997d91ed3d01b51fa39d2dbe77540fd4", size = 51622, upload-time = "2024-10-23T09:46:39.513Z" }, + { url = "https://files.pythonhosted.org/packages/79/43/0bed28bf5eb1c9e4301003b74453b8e7aa85fb293b31dde352aac528dafc/frozenlist-1.5.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:fd74520371c3c4175142d02a976aee0b4cb4a7cc912a60586ffd8d5929979b30", size = 94987, upload-time = "2024-10-23T09:46:40.487Z" }, + { url = "https://files.pythonhosted.org/packages/bb/bf/b74e38f09a246e8abbe1e90eb65787ed745ccab6eaa58b9c9308e052323d/frozenlist-1.5.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:2f3f7a0fbc219fb4455264cae4d9f01ad41ae6ee8524500f381de64ffaa077d5", size = 54584, upload-time = "2024-10-23T09:46:41.463Z" }, + { url = "https://files.pythonhosted.org/packages/2c/31/ab01375682f14f7613a1ade30149f684c84f9b8823a4391ed950c8285656/frozenlist-1.5.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f47c9c9028f55a04ac254346e92977bf0f166c483c74b4232bee19a6697e4778", size = 52499, upload-time = "2024-10-23T09:46:42.451Z" }, + { url = "https://files.pythonhosted.org/packages/98/a8/d0ac0b9276e1404f58fec3ab6e90a4f76b778a49373ccaf6a563f100dfbc/frozenlist-1.5.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0996c66760924da6e88922756d99b47512a71cfd45215f3570bf1e0b694c206a", size = 276357, upload-time = "2024-10-23T09:46:44.166Z" }, + { url = "https://files.pythonhosted.org/packages/ad/c9/c7761084fa822f07dac38ac29f841d4587570dd211e2262544aa0b791d21/frozenlist-1.5.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a2fe128eb4edeabe11896cb6af88fca5346059f6c8d807e3b910069f39157869", size = 287516, upload-time = "2024-10-23T09:46:45.369Z" }, + { url = "https://files.pythonhosted.org/packages/a1/ff/cd7479e703c39df7bdab431798cef89dc75010d8aa0ca2514c5b9321db27/frozenlist-1.5.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1a8ea951bbb6cacd492e3948b8da8c502a3f814f5d20935aae74b5df2b19cf3d", size = 283131, upload-time = "2024-10-23T09:46:46.654Z" }, + { url = "https://files.pythonhosted.org/packages/59/a0/370941beb47d237eca4fbf27e4e91389fd68699e6f4b0ebcc95da463835b/frozenlist-1.5.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:de537c11e4aa01d37db0d403b57bd6f0546e71a82347a97c6a9f0dcc532b3a45", size = 261320, upload-time = "2024-10-23T09:46:47.825Z" }, + { url = "https://files.pythonhosted.org/packages/b8/5f/c10123e8d64867bc9b4f2f510a32042a306ff5fcd7e2e09e5ae5100ee333/frozenlist-1.5.0-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9c2623347b933fcb9095841f1cc5d4ff0b278addd743e0e966cb3d460278840d", size = 274877, upload-time = "2024-10-23T09:46:48.989Z" }, + { url = "https://files.pythonhosted.org/packages/fa/79/38c505601ae29d4348f21706c5d89755ceded02a745016ba2f58bd5f1ea6/frozenlist-1.5.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:cee6798eaf8b1416ef6909b06f7dc04b60755206bddc599f52232606e18179d3", size = 269592, upload-time = "2024-10-23T09:46:50.235Z" }, + { url = "https://files.pythonhosted.org/packages/19/e2/39f3a53191b8204ba9f0bb574b926b73dd2efba2a2b9d2d730517e8f7622/frozenlist-1.5.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:f5f9da7f5dbc00a604fe74aa02ae7c98bcede8a3b8b9666f9f86fc13993bc71a", size = 265934, upload-time = "2024-10-23T09:46:51.829Z" }, + { url = "https://files.pythonhosted.org/packages/d5/c9/3075eb7f7f3a91f1a6b00284af4de0a65a9ae47084930916f5528144c9dd/frozenlist-1.5.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:90646abbc7a5d5c7c19461d2e3eeb76eb0b204919e6ece342feb6032c9325ae9", size = 283859, upload-time = "2024-10-23T09:46:52.947Z" }, + { url = "https://files.pythonhosted.org/packages/05/f5/549f44d314c29408b962fa2b0e69a1a67c59379fb143b92a0a065ffd1f0f/frozenlist-1.5.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:bdac3c7d9b705d253b2ce370fde941836a5f8b3c5c2b8fd70940a3ea3af7f4f2", size = 287560, upload-time = "2024-10-23T09:46:54.162Z" }, + { url = "https://files.pythonhosted.org/packages/9d/f8/cb09b3c24a3eac02c4c07a9558e11e9e244fb02bf62c85ac2106d1eb0c0b/frozenlist-1.5.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:03d33c2ddbc1816237a67f66336616416e2bbb6beb306e5f890f2eb22b959cdf", size = 277150, upload-time = "2024-10-23T09:46:55.361Z" }, + { url = "https://files.pythonhosted.org/packages/37/48/38c2db3f54d1501e692d6fe058f45b6ad1b358d82cd19436efab80cfc965/frozenlist-1.5.0-cp311-cp311-win32.whl", hash = "sha256:237f6b23ee0f44066219dae14c70ae38a63f0440ce6750f868ee08775073f942", size = 45244, upload-time = "2024-10-23T09:46:56.578Z" }, + { url = "https://files.pythonhosted.org/packages/ca/8c/2ddffeb8b60a4bce3b196c32fcc30d8830d4615e7b492ec2071da801b8ad/frozenlist-1.5.0-cp311-cp311-win_amd64.whl", hash = "sha256:0cc974cc93d32c42e7b0f6cf242a6bd941c57c61b618e78b6c0a96cb72788c1d", size = 51634, upload-time = "2024-10-23T09:46:57.6Z" }, + { url = "https://files.pythonhosted.org/packages/79/73/fa6d1a96ab7fd6e6d1c3500700963eab46813847f01ef0ccbaa726181dd5/frozenlist-1.5.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:31115ba75889723431aa9a4e77d5f398f5cf976eea3bdf61749731f62d4a4a21", size = 94026, upload-time = "2024-10-23T09:46:58.601Z" }, + { url = "https://files.pythonhosted.org/packages/ab/04/ea8bf62c8868b8eada363f20ff1b647cf2e93377a7b284d36062d21d81d1/frozenlist-1.5.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:7437601c4d89d070eac8323f121fcf25f88674627505334654fd027b091db09d", size = 54150, upload-time = "2024-10-23T09:46:59.608Z" }, + { url = "https://files.pythonhosted.org/packages/d0/9a/8e479b482a6f2070b26bda572c5e6889bb3ba48977e81beea35b5ae13ece/frozenlist-1.5.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:7948140d9f8ece1745be806f2bfdf390127cf1a763b925c4a805c603df5e697e", size = 51927, upload-time = "2024-10-23T09:47:00.625Z" }, + { url = "https://files.pythonhosted.org/packages/e3/12/2aad87deb08a4e7ccfb33600871bbe8f0e08cb6d8224371387f3303654d7/frozenlist-1.5.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:feeb64bc9bcc6b45c6311c9e9b99406660a9c05ca8a5b30d14a78555088b0b3a", size = 282647, upload-time = "2024-10-23T09:47:01.992Z" }, + { url = "https://files.pythonhosted.org/packages/77/f2/07f06b05d8a427ea0060a9cef6e63405ea9e0d761846b95ef3fb3be57111/frozenlist-1.5.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:683173d371daad49cffb8309779e886e59c2f369430ad28fe715f66d08d4ab1a", size = 289052, upload-time = "2024-10-23T09:47:04.039Z" }, + { url = "https://files.pythonhosted.org/packages/bd/9f/8bf45a2f1cd4aa401acd271b077989c9267ae8463e7c8b1eb0d3f561b65e/frozenlist-1.5.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7d57d8f702221405a9d9b40f9da8ac2e4a1a8b5285aac6100f3393675f0a85ee", size = 291719, upload-time = "2024-10-23T09:47:05.58Z" }, + { url = "https://files.pythonhosted.org/packages/41/d1/1f20fd05a6c42d3868709b7604c9f15538a29e4f734c694c6bcfc3d3b935/frozenlist-1.5.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:30c72000fbcc35b129cb09956836c7d7abf78ab5416595e4857d1cae8d6251a6", size = 267433, upload-time = "2024-10-23T09:47:07.807Z" }, + { url = "https://files.pythonhosted.org/packages/af/f2/64b73a9bb86f5a89fb55450e97cd5c1f84a862d4ff90d9fd1a73ab0f64a5/frozenlist-1.5.0-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:000a77d6034fbad9b6bb880f7ec073027908f1b40254b5d6f26210d2dab1240e", size = 283591, upload-time = "2024-10-23T09:47:09.645Z" }, + { url = "https://files.pythonhosted.org/packages/29/e2/ffbb1fae55a791fd6c2938dd9ea779509c977435ba3940b9f2e8dc9d5316/frozenlist-1.5.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:5d7f5a50342475962eb18b740f3beecc685a15b52c91f7d975257e13e029eca9", size = 273249, upload-time = "2024-10-23T09:47:10.808Z" }, + { url = "https://files.pythonhosted.org/packages/2e/6e/008136a30798bb63618a114b9321b5971172a5abddff44a100c7edc5ad4f/frozenlist-1.5.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:87f724d055eb4785d9be84e9ebf0f24e392ddfad00b3fe036e43f489fafc9039", size = 271075, upload-time = "2024-10-23T09:47:11.938Z" }, + { url = "https://files.pythonhosted.org/packages/ae/f0/4e71e54a026b06724cec9b6c54f0b13a4e9e298cc8db0f82ec70e151f5ce/frozenlist-1.5.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:6e9080bb2fb195a046e5177f10d9d82b8a204c0736a97a153c2466127de87784", size = 285398, upload-time = "2024-10-23T09:47:14.071Z" }, + { url = "https://files.pythonhosted.org/packages/4d/36/70ec246851478b1c0b59f11ef8ade9c482ff447c1363c2bd5fad45098b12/frozenlist-1.5.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:9b93d7aaa36c966fa42efcaf716e6b3900438632a626fb09c049f6a2f09fc631", size = 294445, upload-time = "2024-10-23T09:47:15.318Z" }, + { url = "https://files.pythonhosted.org/packages/37/e0/47f87544055b3349b633a03c4d94b405956cf2437f4ab46d0928b74b7526/frozenlist-1.5.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:52ef692a4bc60a6dd57f507429636c2af8b6046db8b31b18dac02cbc8f507f7f", size = 280569, upload-time = "2024-10-23T09:47:17.149Z" }, + { url = "https://files.pythonhosted.org/packages/f9/7c/490133c160fb6b84ed374c266f42800e33b50c3bbab1652764e6e1fc498a/frozenlist-1.5.0-cp312-cp312-win32.whl", hash = "sha256:29d94c256679247b33a3dc96cce0f93cbc69c23bf75ff715919332fdbb6a32b8", size = 44721, upload-time = "2024-10-23T09:47:19.012Z" }, + { url = "https://files.pythonhosted.org/packages/b1/56/4e45136ffc6bdbfa68c29ca56ef53783ef4c2fd395f7cbf99a2624aa9aaa/frozenlist-1.5.0-cp312-cp312-win_amd64.whl", hash = "sha256:8969190d709e7c48ea386db202d708eb94bdb29207a1f269bab1196ce0dcca1f", size = 51329, upload-time = "2024-10-23T09:47:20.177Z" }, + { url = "https://files.pythonhosted.org/packages/da/3b/915f0bca8a7ea04483622e84a9bd90033bab54bdf485479556c74fd5eaf5/frozenlist-1.5.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:7a1a048f9215c90973402e26c01d1cff8a209e1f1b53f72b95c13db61b00f953", size = 91538, upload-time = "2024-10-23T09:47:21.176Z" }, + { url = "https://files.pythonhosted.org/packages/c7/d1/a7c98aad7e44afe5306a2b068434a5830f1470675f0e715abb86eb15f15b/frozenlist-1.5.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:dd47a5181ce5fcb463b5d9e17ecfdb02b678cca31280639255ce9d0e5aa67af0", size = 52849, upload-time = "2024-10-23T09:47:22.439Z" }, + { url = "https://files.pythonhosted.org/packages/3a/c8/76f23bf9ab15d5f760eb48701909645f686f9c64fbb8982674c241fbef14/frozenlist-1.5.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:1431d60b36d15cda188ea222033eec8e0eab488f39a272461f2e6d9e1a8e63c2", size = 50583, upload-time = "2024-10-23T09:47:23.44Z" }, + { url = "https://files.pythonhosted.org/packages/1f/22/462a3dd093d11df623179d7754a3b3269de3b42de2808cddef50ee0f4f48/frozenlist-1.5.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6482a5851f5d72767fbd0e507e80737f9c8646ae7fd303def99bfe813f76cf7f", size = 265636, upload-time = "2024-10-23T09:47:24.82Z" }, + { url = "https://files.pythonhosted.org/packages/80/cf/e075e407fc2ae7328155a1cd7e22f932773c8073c1fc78016607d19cc3e5/frozenlist-1.5.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:44c49271a937625619e862baacbd037a7ef86dd1ee215afc298a417ff3270608", size = 270214, upload-time = "2024-10-23T09:47:26.156Z" }, + { url = "https://files.pythonhosted.org/packages/a1/58/0642d061d5de779f39c50cbb00df49682832923f3d2ebfb0fedf02d05f7f/frozenlist-1.5.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:12f78f98c2f1c2429d42e6a485f433722b0061d5c0b0139efa64f396efb5886b", size = 273905, upload-time = "2024-10-23T09:47:27.741Z" }, + { url = "https://files.pythonhosted.org/packages/ab/66/3fe0f5f8f2add5b4ab7aa4e199f767fd3b55da26e3ca4ce2cc36698e50c4/frozenlist-1.5.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ce3aa154c452d2467487765e3adc730a8c153af77ad84096bc19ce19a2400840", size = 250542, upload-time = "2024-10-23T09:47:28.938Z" }, + { url = "https://files.pythonhosted.org/packages/f6/b8/260791bde9198c87a465224e0e2bb62c4e716f5d198fc3a1dacc4895dbd1/frozenlist-1.5.0-cp313-cp313-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9b7dc0c4338e6b8b091e8faf0db3168a37101943e687f373dce00959583f7439", size = 267026, upload-time = "2024-10-23T09:47:30.283Z" }, + { url = "https://files.pythonhosted.org/packages/2e/a4/3d24f88c527f08f8d44ade24eaee83b2627793fa62fa07cbb7ff7a2f7d42/frozenlist-1.5.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:45e0896250900b5aa25180f9aec243e84e92ac84bd4a74d9ad4138ef3f5c97de", size = 257690, upload-time = "2024-10-23T09:47:32.388Z" }, + { url = "https://files.pythonhosted.org/packages/de/9a/d311d660420b2beeff3459b6626f2ab4fb236d07afbdac034a4371fe696e/frozenlist-1.5.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:561eb1c9579d495fddb6da8959fd2a1fca2c6d060d4113f5844b433fc02f2641", size = 253893, upload-time = "2024-10-23T09:47:34.274Z" }, + { url = "https://files.pythonhosted.org/packages/c6/23/e491aadc25b56eabd0f18c53bb19f3cdc6de30b2129ee0bc39cd387cd560/frozenlist-1.5.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:df6e2f325bfee1f49f81aaac97d2aa757c7646534a06f8f577ce184afe2f0a9e", size = 267006, upload-time = "2024-10-23T09:47:35.499Z" }, + { url = "https://files.pythonhosted.org/packages/08/c4/ab918ce636a35fb974d13d666dcbe03969592aeca6c3ab3835acff01f79c/frozenlist-1.5.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:140228863501b44b809fb39ec56b5d4071f4d0aa6d216c19cbb08b8c5a7eadb9", size = 276157, upload-time = "2024-10-23T09:47:37.522Z" }, + { url = "https://files.pythonhosted.org/packages/c0/29/3b7a0bbbbe5a34833ba26f686aabfe982924adbdcafdc294a7a129c31688/frozenlist-1.5.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:7707a25d6a77f5d27ea7dc7d1fc608aa0a478193823f88511ef5e6b8a48f9d03", size = 264642, upload-time = "2024-10-23T09:47:38.75Z" }, + { url = "https://files.pythonhosted.org/packages/ab/42/0595b3dbffc2e82d7fe658c12d5a5bafcd7516c6bf2d1d1feb5387caa9c1/frozenlist-1.5.0-cp313-cp313-win32.whl", hash = "sha256:31a9ac2b38ab9b5a8933b693db4939764ad3f299fcaa931a3e605bc3460e693c", size = 44914, upload-time = "2024-10-23T09:47:40.145Z" }, + { url = "https://files.pythonhosted.org/packages/17/c4/b7db1206a3fea44bf3b838ca61deb6f74424a8a5db1dd53ecb21da669be6/frozenlist-1.5.0-cp313-cp313-win_amd64.whl", hash = "sha256:11aabdd62b8b9c4b84081a3c246506d1cddd2dd93ff0ad53ede5defec7886b28", size = 51167, upload-time = "2024-10-23T09:47:41.812Z" }, + { url = "https://files.pythonhosted.org/packages/33/b5/00fcbe8e7e7e172829bf4addc8227d8f599a3d5def3a4e9aa2b54b3145aa/frozenlist-1.5.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:dd94994fc91a6177bfaafd7d9fd951bc8689b0a98168aa26b5f543868548d3ca", size = 95648, upload-time = "2024-10-23T09:47:43.118Z" }, + { url = "https://files.pythonhosted.org/packages/1e/69/e4a32fc4b2fa8e9cb6bcb1bad9c7eeb4b254bc34da475b23f93264fdc306/frozenlist-1.5.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:2d0da8bbec082bf6bf18345b180958775363588678f64998c2b7609e34719b10", size = 54888, upload-time = "2024-10-23T09:47:44.832Z" }, + { url = "https://files.pythonhosted.org/packages/76/a3/c08322a91e73d1199901a77ce73971cffa06d3c74974270ff97aed6e152a/frozenlist-1.5.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:73f2e31ea8dd7df61a359b731716018c2be196e5bb3b74ddba107f694fbd7604", size = 52975, upload-time = "2024-10-23T09:47:46.579Z" }, + { url = "https://files.pythonhosted.org/packages/fc/60/a315321d8ada167b578ff9d2edc147274ead6129523b3a308501b6621b4f/frozenlist-1.5.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:828afae9f17e6de596825cf4228ff28fbdf6065974e5ac1410cecc22f699d2b3", size = 241912, upload-time = "2024-10-23T09:47:47.687Z" }, + { url = "https://files.pythonhosted.org/packages/bd/d0/1f0980987bca4f94f9e8bae01980b23495ffc2e5049a3da4d9b7d2762bee/frozenlist-1.5.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f1577515d35ed5649d52ab4319db757bb881ce3b2b796d7283e6634d99ace307", size = 259433, upload-time = "2024-10-23T09:47:49.339Z" }, + { url = "https://files.pythonhosted.org/packages/28/e7/d00600c072eec8f18a606e281afdf0e8606e71a4882104d0438429b02468/frozenlist-1.5.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2150cc6305a2c2ab33299453e2968611dacb970d2283a14955923062c8d00b10", size = 255576, upload-time = "2024-10-23T09:47:50.519Z" }, + { url = "https://files.pythonhosted.org/packages/82/71/993c5f45dba7be347384ddec1ebc1b4d998291884e7690c06aa6ba755211/frozenlist-1.5.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a72b7a6e3cd2725eff67cd64c8f13335ee18fc3c7befc05aed043d24c7b9ccb9", size = 233349, upload-time = "2024-10-23T09:47:53.197Z" }, + { url = "https://files.pythonhosted.org/packages/66/30/f9c006223feb2ac87f1826b57f2367b60aacc43092f562dab60d2312562e/frozenlist-1.5.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c16d2fa63e0800723139137d667e1056bee1a1cf7965153d2d104b62855e9b99", size = 243126, upload-time = "2024-10-23T09:47:54.432Z" }, + { url = "https://files.pythonhosted.org/packages/b5/34/e4219c9343f94b81068d0018cbe37948e66c68003b52bf8a05e9509d09ec/frozenlist-1.5.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:17dcc32fc7bda7ce5875435003220a457bcfa34ab7924a49a1c19f55b6ee185c", size = 241261, upload-time = "2024-10-23T09:47:56.01Z" }, + { url = "https://files.pythonhosted.org/packages/48/96/9141758f6a19f2061a51bb59b9907c92f9bda1ac7b2baaf67a6e352b280f/frozenlist-1.5.0-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:97160e245ea33d8609cd2b8fd997c850b56db147a304a262abc2b3be021a9171", size = 240203, upload-time = "2024-10-23T09:47:57.337Z" }, + { url = "https://files.pythonhosted.org/packages/f9/71/0ef5970e68d181571a050958e84c76a061ca52f9c6f50257d9bfdd84c7f7/frozenlist-1.5.0-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:f1e6540b7fa044eee0bb5111ada694cf3dc15f2b0347ca125ee9ca984d5e9e6e", size = 267539, upload-time = "2024-10-23T09:47:58.874Z" }, + { url = "https://files.pythonhosted.org/packages/ab/bd/6e7d450c5d993b413591ad9cdab6dcdfa2c6ab2cd835b2b5c1cfeb0323bf/frozenlist-1.5.0-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:91d6c171862df0a6c61479d9724f22efb6109111017c87567cfeb7b5d1449fdf", size = 268518, upload-time = "2024-10-23T09:48:00.771Z" }, + { url = "https://files.pythonhosted.org/packages/cc/3d/5a7c4dfff1ae57ca2cbbe9041521472ecd9446d49e7044a0e9bfd0200fd0/frozenlist-1.5.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:c1fac3e2ace2eb1052e9f7c7db480818371134410e1f5c55d65e8f3ac6d1407e", size = 248114, upload-time = "2024-10-23T09:48:02.625Z" }, + { url = "https://files.pythonhosted.org/packages/f7/41/2342ec4c714349793f1a1e7bd5c4aeec261e24e697fa9a5499350c3a2415/frozenlist-1.5.0-cp38-cp38-win32.whl", hash = "sha256:b97f7b575ab4a8af9b7bc1d2ef7f29d3afee2226bd03ca3875c16451ad5a7723", size = 45648, upload-time = "2024-10-23T09:48:03.895Z" }, + { url = "https://files.pythonhosted.org/packages/0c/90/85bb3547c327f5975078c1be018478d5e8d250a540c828f8f31a35d2a1bd/frozenlist-1.5.0-cp38-cp38-win_amd64.whl", hash = "sha256:374ca2dabdccad8e2a76d40b1d037f5bd16824933bf7bcea3e59c891fd4a0923", size = 51930, upload-time = "2024-10-23T09:48:05.293Z" }, + { url = "https://files.pythonhosted.org/packages/da/4d/d94ff0fb0f5313902c132817c62d19cdc5bdcd0c195d392006ef4b779fc6/frozenlist-1.5.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:9bbcdfaf4af7ce002694a4e10a0159d5a8d20056a12b05b45cea944a4953f972", size = 95319, upload-time = "2024-10-23T09:48:06.405Z" }, + { url = "https://files.pythonhosted.org/packages/8c/1b/d90e554ca2b483d31cb2296e393f72c25bdc38d64526579e95576bfda587/frozenlist-1.5.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1893f948bf6681733aaccf36c5232c231e3b5166d607c5fa77773611df6dc336", size = 54749, upload-time = "2024-10-23T09:48:07.48Z" }, + { url = "https://files.pythonhosted.org/packages/f8/66/7fdecc9ef49f8db2aa4d9da916e4ecf357d867d87aea292efc11e1b2e932/frozenlist-1.5.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:2b5e23253bb709ef57a8e95e6ae48daa9ac5f265637529e4ce6b003a37b2621f", size = 52718, upload-time = "2024-10-23T09:48:08.725Z" }, + { url = "https://files.pythonhosted.org/packages/08/04/e2fddc92135276e07addbc1cf413acffa0c2d848b3e54cacf684e146df49/frozenlist-1.5.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0f253985bb515ecd89629db13cb58d702035ecd8cfbca7d7a7e29a0e6d39af5f", size = 241756, upload-time = "2024-10-23T09:48:09.843Z" }, + { url = "https://files.pythonhosted.org/packages/c6/52/be5ff200815d8a341aee5b16b6b707355e0ca3652953852238eb92b120c2/frozenlist-1.5.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:04a5c6babd5e8fb7d3c871dc8b321166b80e41b637c31a995ed844a6139942b6", size = 267718, upload-time = "2024-10-23T09:48:11.828Z" }, + { url = "https://files.pythonhosted.org/packages/88/be/4bd93a58be57a3722fc544c36debdf9dcc6758f761092e894d78f18b8f20/frozenlist-1.5.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a9fe0f1c29ba24ba6ff6abf688cb0b7cf1efab6b6aa6adc55441773c252f7411", size = 263494, upload-time = "2024-10-23T09:48:13.424Z" }, + { url = "https://files.pythonhosted.org/packages/32/ba/58348b90193caa096ce9e9befea6ae67f38dabfd3aacb47e46137a6250a8/frozenlist-1.5.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:226d72559fa19babe2ccd920273e767c96a49b9d3d38badd7c91a0fdeda8ea08", size = 232838, upload-time = "2024-10-23T09:48:14.792Z" }, + { url = "https://files.pythonhosted.org/packages/f6/33/9f152105227630246135188901373c4f322cc026565ca6215b063f4c82f4/frozenlist-1.5.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:15b731db116ab3aedec558573c1a5eec78822b32292fe4f2f0345b7f697745c2", size = 242912, upload-time = "2024-10-23T09:48:16.249Z" }, + { url = "https://files.pythonhosted.org/packages/a0/10/3db38fb3ccbafadd80a1b0d6800c987b0e3fe3ef2d117c6ced0246eea17a/frozenlist-1.5.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:366d8f93e3edfe5a918c874702f78faac300209a4d5bf38352b2c1bdc07a766d", size = 244763, upload-time = "2024-10-23T09:48:17.781Z" }, + { url = "https://files.pythonhosted.org/packages/e2/cd/1df468fdce2f66a4608dffe44c40cdc35eeaa67ef7fd1d813f99a9a37842/frozenlist-1.5.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:1b96af8c582b94d381a1c1f51ffaedeb77c821c690ea5f01da3d70a487dd0a9b", size = 242841, upload-time = "2024-10-23T09:48:19.507Z" }, + { url = "https://files.pythonhosted.org/packages/ee/5f/16097a5ca0bb6b6779c02cc9379c72fe98d56115d4c54d059fb233168fb6/frozenlist-1.5.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:c03eff4a41bd4e38415cbed054bbaff4a075b093e2394b6915dca34a40d1e38b", size = 263407, upload-time = "2024-10-23T09:48:21.467Z" }, + { url = "https://files.pythonhosted.org/packages/0f/f7/58cd220ee1c2248ee65a32f5b4b93689e3fe1764d85537eee9fc392543bc/frozenlist-1.5.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:50cf5e7ee9b98f22bdecbabf3800ae78ddcc26e4a435515fc72d97903e8488e0", size = 265083, upload-time = "2024-10-23T09:48:22.725Z" }, + { url = "https://files.pythonhosted.org/packages/62/b8/49768980caabf81ac4a2d156008f7cbd0107e6b36d08a313bb31035d9201/frozenlist-1.5.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:1e76bfbc72353269c44e0bc2cfe171900fbf7f722ad74c9a7b638052afe6a00c", size = 251564, upload-time = "2024-10-23T09:48:24.272Z" }, + { url = "https://files.pythonhosted.org/packages/cb/83/619327da3b86ef957ee7a0cbf3c166a09ed1e87a3f7f1ff487d7d0284683/frozenlist-1.5.0-cp39-cp39-win32.whl", hash = "sha256:666534d15ba8f0fda3f53969117383d5dc021266b3c1a42c9ec4855e4b58b9d3", size = 45691, upload-time = "2024-10-23T09:48:26.317Z" }, + { url = "https://files.pythonhosted.org/packages/8b/28/407bc34a745151ed2322c690b6e7d83d7101472e81ed76e1ebdac0b70a78/frozenlist-1.5.0-cp39-cp39-win_amd64.whl", hash = "sha256:5c28f4b5dbef8a0d8aad0d4de24d1e9e981728628afaf4ea0792f5d0939372f0", size = 51767, upload-time = "2024-10-23T09:48:27.427Z" }, + { url = "https://files.pythonhosted.org/packages/c6/c8/a5be5b7550c10858fcf9b0ea054baccab474da77d37f1e828ce043a3a5d4/frozenlist-1.5.0-py3-none-any.whl", hash = "sha256:d994863bba198a4a518b467bb971c56e1db3f180a25c6cf7bb1949c267f748c3", size = 11901, upload-time = "2024-10-23T09:48:28.851Z" }, +] + +[[package]] +name = "frozenlist" +version = "1.8.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/2d/f5/c831fac6cc817d26fd54c7eaccd04ef7e0288806943f7cc5bbf69f3ac1f0/frozenlist-1.8.0.tar.gz", hash = "sha256:3ede829ed8d842f6cd48fc7081d7a41001a56f1f38603f9d49bf3020d59a31ad", size = 45875, upload-time = "2025-10-06T05:38:17.865Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/83/4a/557715d5047da48d54e659203b9335be7bfaafda2c3f627b7c47e0b3aaf3/frozenlist-1.8.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:b37f6d31b3dcea7deb5e9696e529a6aa4a898adc33db82da12e4c60a7c4d2011", size = 86230, upload-time = "2025-10-06T05:35:23.699Z" }, + { url = "https://files.pythonhosted.org/packages/a2/fb/c85f9fed3ea8fe8740e5b46a59cc141c23b842eca617da8876cfce5f760e/frozenlist-1.8.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:ef2b7b394f208233e471abc541cc6991f907ffd47dc72584acee3147899d6565", size = 49621, upload-time = "2025-10-06T05:35:25.341Z" }, + { url = "https://files.pythonhosted.org/packages/63/70/26ca3f06aace16f2352796b08704338d74b6d1a24ca38f2771afbb7ed915/frozenlist-1.8.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a88f062f072d1589b7b46e951698950e7da00442fc1cacbe17e19e025dc327ad", size = 49889, upload-time = "2025-10-06T05:35:26.797Z" }, + { url = "https://files.pythonhosted.org/packages/5d/ed/c7895fd2fde7f3ee70d248175f9b6cdf792fb741ab92dc59cd9ef3bd241b/frozenlist-1.8.0-cp310-cp310-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:f57fb59d9f385710aa7060e89410aeb5058b99e62f4d16b08b91986b9a2140c2", size = 219464, upload-time = "2025-10-06T05:35:28.254Z" }, + { url = "https://files.pythonhosted.org/packages/6b/83/4d587dccbfca74cb8b810472392ad62bfa100bf8108c7223eb4c4fa2f7b3/frozenlist-1.8.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:799345ab092bee59f01a915620b5d014698547afd011e691a208637312db9186", size = 221649, upload-time = "2025-10-06T05:35:29.454Z" }, + { url = "https://files.pythonhosted.org/packages/6a/c6/fd3b9cd046ec5fff9dab66831083bc2077006a874a2d3d9247dea93ddf7e/frozenlist-1.8.0-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:c23c3ff005322a6e16f71bf8692fcf4d5a304aaafe1e262c98c6d4adc7be863e", size = 219188, upload-time = "2025-10-06T05:35:30.951Z" }, + { url = "https://files.pythonhosted.org/packages/ce/80/6693f55eb2e085fc8afb28cf611448fb5b90e98e068fa1d1b8d8e66e5c7d/frozenlist-1.8.0-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:8a76ea0f0b9dfa06f254ee06053d93a600865b3274358ca48a352ce4f0798450", size = 231748, upload-time = "2025-10-06T05:35:32.101Z" }, + { url = "https://files.pythonhosted.org/packages/97/d6/e9459f7c5183854abd989ba384fe0cc1a0fb795a83c033f0571ec5933ca4/frozenlist-1.8.0-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:c7366fe1418a6133d5aa824ee53d406550110984de7637d65a178010f759c6ef", size = 236351, upload-time = "2025-10-06T05:35:33.834Z" }, + { url = "https://files.pythonhosted.org/packages/97/92/24e97474b65c0262e9ecd076e826bfd1d3074adcc165a256e42e7b8a7249/frozenlist-1.8.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:13d23a45c4cebade99340c4165bd90eeb4a56c6d8a9d8aa49568cac19a6d0dc4", size = 218767, upload-time = "2025-10-06T05:35:35.205Z" }, + { url = "https://files.pythonhosted.org/packages/ee/bf/dc394a097508f15abff383c5108cb8ad880d1f64a725ed3b90d5c2fbf0bb/frozenlist-1.8.0-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:e4a3408834f65da56c83528fb52ce7911484f0d1eaf7b761fc66001db1646eff", size = 235887, upload-time = "2025-10-06T05:35:36.354Z" }, + { url = "https://files.pythonhosted.org/packages/40/90/25b201b9c015dbc999a5baf475a257010471a1fa8c200c843fd4abbee725/frozenlist-1.8.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:42145cd2748ca39f32801dad54aeea10039da6f86e303659db90db1c4b614c8c", size = 228785, upload-time = "2025-10-06T05:35:37.949Z" }, + { url = "https://files.pythonhosted.org/packages/84/f4/b5bc148df03082f05d2dd30c089e269acdbe251ac9a9cf4e727b2dbb8a3d/frozenlist-1.8.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:e2de870d16a7a53901e41b64ffdf26f2fbb8917b3e6ebf398098d72c5b20bd7f", size = 230312, upload-time = "2025-10-06T05:35:39.178Z" }, + { url = "https://files.pythonhosted.org/packages/db/4b/87e95b5d15097c302430e647136b7d7ab2398a702390cf4c8601975709e7/frozenlist-1.8.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:20e63c9493d33ee48536600d1a5c95eefc870cd71e7ab037763d1fbb89cc51e7", size = 217650, upload-time = "2025-10-06T05:35:40.377Z" }, + { url = "https://files.pythonhosted.org/packages/e5/70/78a0315d1fea97120591a83e0acd644da638c872f142fd72a6cebee825f3/frozenlist-1.8.0-cp310-cp310-win32.whl", hash = "sha256:adbeebaebae3526afc3c96fad434367cafbfd1b25d72369a9e5858453b1bb71a", size = 39659, upload-time = "2025-10-06T05:35:41.863Z" }, + { url = "https://files.pythonhosted.org/packages/66/aa/3f04523fb189a00e147e60c5b2205126118f216b0aa908035c45336e27e4/frozenlist-1.8.0-cp310-cp310-win_amd64.whl", hash = "sha256:667c3777ca571e5dbeb76f331562ff98b957431df140b54c85fd4d52eea8d8f6", size = 43837, upload-time = "2025-10-06T05:35:43.205Z" }, + { url = "https://files.pythonhosted.org/packages/39/75/1135feecdd7c336938bd55b4dc3b0dfc46d85b9be12ef2628574b28de776/frozenlist-1.8.0-cp310-cp310-win_arm64.whl", hash = "sha256:80f85f0a7cc86e7a54c46d99c9e1318ff01f4687c172ede30fd52d19d1da1c8e", size = 39989, upload-time = "2025-10-06T05:35:44.596Z" }, + { url = "https://files.pythonhosted.org/packages/bc/03/077f869d540370db12165c0aa51640a873fb661d8b315d1d4d67b284d7ac/frozenlist-1.8.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:09474e9831bc2b2199fad6da3c14c7b0fbdd377cce9d3d77131be28906cb7d84", size = 86912, upload-time = "2025-10-06T05:35:45.98Z" }, + { url = "https://files.pythonhosted.org/packages/df/b5/7610b6bd13e4ae77b96ba85abea1c8cb249683217ef09ac9e0ae93f25a91/frozenlist-1.8.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:17c883ab0ab67200b5f964d2b9ed6b00971917d5d8a92df149dc2c9779208ee9", size = 50046, upload-time = "2025-10-06T05:35:47.009Z" }, + { url = "https://files.pythonhosted.org/packages/6e/ef/0e8f1fe32f8a53dd26bdd1f9347efe0778b0fddf62789ea683f4cc7d787d/frozenlist-1.8.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:fa47e444b8ba08fffd1c18e8cdb9a75db1b6a27f17507522834ad13ed5922b93", size = 50119, upload-time = "2025-10-06T05:35:48.38Z" }, + { url = "https://files.pythonhosted.org/packages/11/b1/71a477adc7c36e5fb628245dfbdea2166feae310757dea848d02bd0689fd/frozenlist-1.8.0-cp311-cp311-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:2552f44204b744fba866e573be4c1f9048d6a324dfe14475103fd51613eb1d1f", size = 231067, upload-time = "2025-10-06T05:35:49.97Z" }, + { url = "https://files.pythonhosted.org/packages/45/7e/afe40eca3a2dc19b9904c0f5d7edfe82b5304cb831391edec0ac04af94c2/frozenlist-1.8.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:957e7c38f250991e48a9a73e6423db1bb9dd14e722a10f6b8bb8e16a0f55f695", size = 233160, upload-time = "2025-10-06T05:35:51.729Z" }, + { url = "https://files.pythonhosted.org/packages/a6/aa/7416eac95603ce428679d273255ffc7c998d4132cfae200103f164b108aa/frozenlist-1.8.0-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:8585e3bb2cdea02fc88ffa245069c36555557ad3609e83be0ec71f54fd4abb52", size = 228544, upload-time = "2025-10-06T05:35:53.246Z" }, + { url = "https://files.pythonhosted.org/packages/8b/3d/2a2d1f683d55ac7e3875e4263d28410063e738384d3adc294f5ff3d7105e/frozenlist-1.8.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:edee74874ce20a373d62dc28b0b18b93f645633c2943fd90ee9d898550770581", size = 243797, upload-time = "2025-10-06T05:35:54.497Z" }, + { url = "https://files.pythonhosted.org/packages/78/1e/2d5565b589e580c296d3bb54da08d206e797d941a83a6fdea42af23be79c/frozenlist-1.8.0-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:c9a63152fe95756b85f31186bddf42e4c02c6321207fd6601a1c89ebac4fe567", size = 247923, upload-time = "2025-10-06T05:35:55.861Z" }, + { url = "https://files.pythonhosted.org/packages/aa/c3/65872fcf1d326a7f101ad4d86285c403c87be7d832b7470b77f6d2ed5ddc/frozenlist-1.8.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:b6db2185db9be0a04fecf2f241c70b63b1a242e2805be291855078f2b404dd6b", size = 230886, upload-time = "2025-10-06T05:35:57.399Z" }, + { url = "https://files.pythonhosted.org/packages/a0/76/ac9ced601d62f6956f03cc794f9e04c81719509f85255abf96e2510f4265/frozenlist-1.8.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:f4be2e3d8bc8aabd566f8d5b8ba7ecc09249d74ba3c9ed52e54dc23a293f0b92", size = 245731, upload-time = "2025-10-06T05:35:58.563Z" }, + { url = "https://files.pythonhosted.org/packages/b9/49/ecccb5f2598daf0b4a1415497eba4c33c1e8ce07495eb07d2860c731b8d5/frozenlist-1.8.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:c8d1634419f39ea6f5c427ea2f90ca85126b54b50837f31497f3bf38266e853d", size = 241544, upload-time = "2025-10-06T05:35:59.719Z" }, + { url = "https://files.pythonhosted.org/packages/53/4b/ddf24113323c0bbcc54cb38c8b8916f1da7165e07b8e24a717b4a12cbf10/frozenlist-1.8.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:1a7fa382a4a223773ed64242dbe1c9c326ec09457e6b8428efb4118c685c3dfd", size = 241806, upload-time = "2025-10-06T05:36:00.959Z" }, + { url = "https://files.pythonhosted.org/packages/a7/fb/9b9a084d73c67175484ba2789a59f8eebebd0827d186a8102005ce41e1ba/frozenlist-1.8.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:11847b53d722050808926e785df837353bd4d75f1d494377e59b23594d834967", size = 229382, upload-time = "2025-10-06T05:36:02.22Z" }, + { url = "https://files.pythonhosted.org/packages/95/a3/c8fb25aac55bf5e12dae5c5aa6a98f85d436c1dc658f21c3ac73f9fa95e5/frozenlist-1.8.0-cp311-cp311-win32.whl", hash = "sha256:27c6e8077956cf73eadd514be8fb04d77fc946a7fe9f7fe167648b0b9085cc25", size = 39647, upload-time = "2025-10-06T05:36:03.409Z" }, + { url = "https://files.pythonhosted.org/packages/0a/f5/603d0d6a02cfd4c8f2a095a54672b3cf967ad688a60fb9faf04fc4887f65/frozenlist-1.8.0-cp311-cp311-win_amd64.whl", hash = "sha256:ac913f8403b36a2c8610bbfd25b8013488533e71e62b4b4adce9c86c8cea905b", size = 44064, upload-time = "2025-10-06T05:36:04.368Z" }, + { url = "https://files.pythonhosted.org/packages/5d/16/c2c9ab44e181f043a86f9a8f84d5124b62dbcb3a02c0977ec72b9ac1d3e0/frozenlist-1.8.0-cp311-cp311-win_arm64.whl", hash = "sha256:d4d3214a0f8394edfa3e303136d0575eece0745ff2b47bd2cb2e66dd92d4351a", size = 39937, upload-time = "2025-10-06T05:36:05.669Z" }, + { url = "https://files.pythonhosted.org/packages/69/29/948b9aa87e75820a38650af445d2ef2b6b8a6fab1a23b6bb9e4ef0be2d59/frozenlist-1.8.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:78f7b9e5d6f2fdb88cdde9440dc147259b62b9d3b019924def9f6478be254ac1", size = 87782, upload-time = "2025-10-06T05:36:06.649Z" }, + { url = "https://files.pythonhosted.org/packages/64/80/4f6e318ee2a7c0750ed724fa33a4bdf1eacdc5a39a7a24e818a773cd91af/frozenlist-1.8.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:229bf37d2e4acdaf808fd3f06e854a4a7a3661e871b10dc1f8f1896a3b05f18b", size = 50594, upload-time = "2025-10-06T05:36:07.69Z" }, + { url = "https://files.pythonhosted.org/packages/2b/94/5c8a2b50a496b11dd519f4a24cb5496cf125681dd99e94c604ccdea9419a/frozenlist-1.8.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f833670942247a14eafbb675458b4e61c82e002a148f49e68257b79296e865c4", size = 50448, upload-time = "2025-10-06T05:36:08.78Z" }, + { url = "https://files.pythonhosted.org/packages/6a/bd/d91c5e39f490a49df14320f4e8c80161cfcce09f1e2cde1edd16a551abb3/frozenlist-1.8.0-cp312-cp312-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:494a5952b1c597ba44e0e78113a7266e656b9794eec897b19ead706bd7074383", size = 242411, upload-time = "2025-10-06T05:36:09.801Z" }, + { url = "https://files.pythonhosted.org/packages/8f/83/f61505a05109ef3293dfb1ff594d13d64a2324ac3482be2cedc2be818256/frozenlist-1.8.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:96f423a119f4777a4a056b66ce11527366a8bb92f54e541ade21f2374433f6d4", size = 243014, upload-time = "2025-10-06T05:36:11.394Z" }, + { url = "https://files.pythonhosted.org/packages/d8/cb/cb6c7b0f7d4023ddda30cf56b8b17494eb3a79e3fda666bf735f63118b35/frozenlist-1.8.0-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:3462dd9475af2025c31cc61be6652dfa25cbfb56cbbf52f4ccfe029f38decaf8", size = 234909, upload-time = "2025-10-06T05:36:12.598Z" }, + { url = "https://files.pythonhosted.org/packages/31/c5/cd7a1f3b8b34af009fb17d4123c5a778b44ae2804e3ad6b86204255f9ec5/frozenlist-1.8.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c4c800524c9cd9bac5166cd6f55285957fcfc907db323e193f2afcd4d9abd69b", size = 250049, upload-time = "2025-10-06T05:36:14.065Z" }, + { url = "https://files.pythonhosted.org/packages/c0/01/2f95d3b416c584a1e7f0e1d6d31998c4a795f7544069ee2e0962a4b60740/frozenlist-1.8.0-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d6a5df73acd3399d893dafc71663ad22534b5aa4f94e8a2fabfe856c3c1b6a52", size = 256485, upload-time = "2025-10-06T05:36:15.39Z" }, + { url = "https://files.pythonhosted.org/packages/ce/03/024bf7720b3abaebcff6d0793d73c154237b85bdf67b7ed55e5e9596dc9a/frozenlist-1.8.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:405e8fe955c2280ce66428b3ca55e12b3c4e9c336fb2103a4937e891c69a4a29", size = 237619, upload-time = "2025-10-06T05:36:16.558Z" }, + { url = "https://files.pythonhosted.org/packages/69/fa/f8abdfe7d76b731f5d8bd217827cf6764d4f1d9763407e42717b4bed50a0/frozenlist-1.8.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:908bd3f6439f2fef9e85031b59fd4f1297af54415fb60e4254a95f75b3cab3f3", size = 250320, upload-time = "2025-10-06T05:36:17.821Z" }, + { url = "https://files.pythonhosted.org/packages/f5/3c/b051329f718b463b22613e269ad72138cc256c540f78a6de89452803a47d/frozenlist-1.8.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:294e487f9ec720bd8ffcebc99d575f7eff3568a08a253d1ee1a0378754b74143", size = 246820, upload-time = "2025-10-06T05:36:19.046Z" }, + { url = "https://files.pythonhosted.org/packages/0f/ae/58282e8f98e444b3f4dd42448ff36fa38bef29e40d40f330b22e7108f565/frozenlist-1.8.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:74c51543498289c0c43656701be6b077f4b265868fa7f8a8859c197006efb608", size = 250518, upload-time = "2025-10-06T05:36:20.763Z" }, + { url = "https://files.pythonhosted.org/packages/8f/96/007e5944694d66123183845a106547a15944fbbb7154788cbf7272789536/frozenlist-1.8.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:776f352e8329135506a1d6bf16ac3f87bc25b28e765949282dcc627af36123aa", size = 239096, upload-time = "2025-10-06T05:36:22.129Z" }, + { url = "https://files.pythonhosted.org/packages/66/bb/852b9d6db2fa40be96f29c0d1205c306288f0684df8fd26ca1951d461a56/frozenlist-1.8.0-cp312-cp312-win32.whl", hash = "sha256:433403ae80709741ce34038da08511d4a77062aa924baf411ef73d1146e74faf", size = 39985, upload-time = "2025-10-06T05:36:23.661Z" }, + { url = "https://files.pythonhosted.org/packages/b8/af/38e51a553dd66eb064cdf193841f16f077585d4d28394c2fa6235cb41765/frozenlist-1.8.0-cp312-cp312-win_amd64.whl", hash = "sha256:34187385b08f866104f0c0617404c8eb08165ab1272e884abc89c112e9c00746", size = 44591, upload-time = "2025-10-06T05:36:24.958Z" }, + { url = "https://files.pythonhosted.org/packages/a7/06/1dc65480ab147339fecc70797e9c2f69d9cea9cf38934ce08df070fdb9cb/frozenlist-1.8.0-cp312-cp312-win_arm64.whl", hash = "sha256:fe3c58d2f5db5fbd18c2987cba06d51b0529f52bc3a6cdc33d3f4eab725104bd", size = 40102, upload-time = "2025-10-06T05:36:26.333Z" }, + { url = "https://files.pythonhosted.org/packages/2d/40/0832c31a37d60f60ed79e9dfb5a92e1e2af4f40a16a29abcc7992af9edff/frozenlist-1.8.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:8d92f1a84bb12d9e56f818b3a746f3efba93c1b63c8387a73dde655e1e42282a", size = 85717, upload-time = "2025-10-06T05:36:27.341Z" }, + { url = "https://files.pythonhosted.org/packages/30/ba/b0b3de23f40bc55a7057bd38434e25c34fa48e17f20ee273bbde5e0650f3/frozenlist-1.8.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:96153e77a591c8adc2ee805756c61f59fef4cf4073a9275ee86fe8cba41241f7", size = 49651, upload-time = "2025-10-06T05:36:28.855Z" }, + { url = "https://files.pythonhosted.org/packages/0c/ab/6e5080ee374f875296c4243c381bbdef97a9ac39c6e3ce1d5f7d42cb78d6/frozenlist-1.8.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:f21f00a91358803399890ab167098c131ec2ddd5f8f5fd5fe9c9f2c6fcd91e40", size = 49417, upload-time = "2025-10-06T05:36:29.877Z" }, + { url = "https://files.pythonhosted.org/packages/d5/4e/e4691508f9477ce67da2015d8c00acd751e6287739123113a9fca6f1604e/frozenlist-1.8.0-cp313-cp313-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:fb30f9626572a76dfe4293c7194a09fb1fe93ba94c7d4f720dfae3b646b45027", size = 234391, upload-time = "2025-10-06T05:36:31.301Z" }, + { url = "https://files.pythonhosted.org/packages/40/76/c202df58e3acdf12969a7895fd6f3bc016c642e6726aa63bd3025e0fc71c/frozenlist-1.8.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:eaa352d7047a31d87dafcacbabe89df0aa506abb5b1b85a2fb91bc3faa02d822", size = 233048, upload-time = "2025-10-06T05:36:32.531Z" }, + { url = "https://files.pythonhosted.org/packages/f9/c0/8746afb90f17b73ca5979c7a3958116e105ff796e718575175319b5bb4ce/frozenlist-1.8.0-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:03ae967b4e297f58f8c774c7eabcce57fe3c2434817d4385c50661845a058121", size = 226549, upload-time = "2025-10-06T05:36:33.706Z" }, + { url = "https://files.pythonhosted.org/packages/7e/eb/4c7eefc718ff72f9b6c4893291abaae5fbc0c82226a32dcd8ef4f7a5dbef/frozenlist-1.8.0-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f6292f1de555ffcc675941d65fffffb0a5bcd992905015f85d0592201793e0e5", size = 239833, upload-time = "2025-10-06T05:36:34.947Z" }, + { url = "https://files.pythonhosted.org/packages/c2/4e/e5c02187cf704224f8b21bee886f3d713ca379535f16893233b9d672ea71/frozenlist-1.8.0-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:29548f9b5b5e3460ce7378144c3010363d8035cea44bc0bf02d57f5a685e084e", size = 245363, upload-time = "2025-10-06T05:36:36.534Z" }, + { url = "https://files.pythonhosted.org/packages/1f/96/cb85ec608464472e82ad37a17f844889c36100eed57bea094518bf270692/frozenlist-1.8.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:ec3cc8c5d4084591b4237c0a272cc4f50a5b03396a47d9caaf76f5d7b38a4f11", size = 229314, upload-time = "2025-10-06T05:36:38.582Z" }, + { url = "https://files.pythonhosted.org/packages/5d/6f/4ae69c550e4cee66b57887daeebe006fe985917c01d0fff9caab9883f6d0/frozenlist-1.8.0-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:517279f58009d0b1f2e7c1b130b377a349405da3f7621ed6bfae50b10adf20c1", size = 243365, upload-time = "2025-10-06T05:36:40.152Z" }, + { url = "https://files.pythonhosted.org/packages/7a/58/afd56de246cf11780a40a2c28dc7cbabbf06337cc8ddb1c780a2d97e88d8/frozenlist-1.8.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:db1e72ede2d0d7ccb213f218df6a078a9c09a7de257c2fe8fcef16d5925230b1", size = 237763, upload-time = "2025-10-06T05:36:41.355Z" }, + { url = "https://files.pythonhosted.org/packages/cb/36/cdfaf6ed42e2644740d4a10452d8e97fa1c062e2a8006e4b09f1b5fd7d63/frozenlist-1.8.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:b4dec9482a65c54a5044486847b8a66bf10c9cb4926d42927ec4e8fd5db7fed8", size = 240110, upload-time = "2025-10-06T05:36:42.716Z" }, + { url = "https://files.pythonhosted.org/packages/03/a8/9ea226fbefad669f11b52e864c55f0bd57d3c8d7eb07e9f2e9a0b39502e1/frozenlist-1.8.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:21900c48ae04d13d416f0e1e0c4d81f7931f73a9dfa0b7a8746fb2fe7dd970ed", size = 233717, upload-time = "2025-10-06T05:36:44.251Z" }, + { url = "https://files.pythonhosted.org/packages/1e/0b/1b5531611e83ba7d13ccc9988967ea1b51186af64c42b7a7af465dcc9568/frozenlist-1.8.0-cp313-cp313-win32.whl", hash = "sha256:8b7b94a067d1c504ee0b16def57ad5738701e4ba10cec90529f13fa03c833496", size = 39628, upload-time = "2025-10-06T05:36:45.423Z" }, + { url = "https://files.pythonhosted.org/packages/d8/cf/174c91dbc9cc49bc7b7aab74d8b734e974d1faa8f191c74af9b7e80848e6/frozenlist-1.8.0-cp313-cp313-win_amd64.whl", hash = "sha256:878be833caa6a3821caf85eb39c5ba92d28e85df26d57afb06b35b2efd937231", size = 43882, upload-time = "2025-10-06T05:36:46.796Z" }, + { url = "https://files.pythonhosted.org/packages/c1/17/502cd212cbfa96eb1388614fe39a3fc9ab87dbbe042b66f97acb57474834/frozenlist-1.8.0-cp313-cp313-win_arm64.whl", hash = "sha256:44389d135b3ff43ba8cc89ff7f51f5a0bb6b63d829c8300f79a2fe4fe61bcc62", size = 39676, upload-time = "2025-10-06T05:36:47.8Z" }, + { url = "https://files.pythonhosted.org/packages/d2/5c/3bbfaa920dfab09e76946a5d2833a7cbdf7b9b4a91c714666ac4855b88b4/frozenlist-1.8.0-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:e25ac20a2ef37e91c1b39938b591457666a0fa835c7783c3a8f33ea42870db94", size = 89235, upload-time = "2025-10-06T05:36:48.78Z" }, + { url = "https://files.pythonhosted.org/packages/d2/d6/f03961ef72166cec1687e84e8925838442b615bd0b8854b54923ce5b7b8a/frozenlist-1.8.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:07cdca25a91a4386d2e76ad992916a85038a9b97561bf7a3fd12d5d9ce31870c", size = 50742, upload-time = "2025-10-06T05:36:49.837Z" }, + { url = "https://files.pythonhosted.org/packages/1e/bb/a6d12b7ba4c3337667d0e421f7181c82dda448ce4e7ad7ecd249a16fa806/frozenlist-1.8.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:4e0c11f2cc6717e0a741f84a527c52616140741cd812a50422f83dc31749fb52", size = 51725, upload-time = "2025-10-06T05:36:50.851Z" }, + { url = "https://files.pythonhosted.org/packages/bc/71/d1fed0ffe2c2ccd70b43714c6cab0f4188f09f8a67a7914a6b46ee30f274/frozenlist-1.8.0-cp313-cp313t-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:b3210649ee28062ea6099cfda39e147fa1bc039583c8ee4481cb7811e2448c51", size = 284533, upload-time = "2025-10-06T05:36:51.898Z" }, + { url = "https://files.pythonhosted.org/packages/c9/1f/fb1685a7b009d89f9bf78a42d94461bc06581f6e718c39344754a5d9bada/frozenlist-1.8.0-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:581ef5194c48035a7de2aefc72ac6539823bb71508189e5de01d60c9dcd5fa65", size = 292506, upload-time = "2025-10-06T05:36:53.101Z" }, + { url = "https://files.pythonhosted.org/packages/e6/3b/b991fe1612703f7e0d05c0cf734c1b77aaf7c7d321df4572e8d36e7048c8/frozenlist-1.8.0-cp313-cp313t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:3ef2d026f16a2b1866e1d86fc4e1291e1ed8a387b2c333809419a2f8b3a77b82", size = 274161, upload-time = "2025-10-06T05:36:54.309Z" }, + { url = "https://files.pythonhosted.org/packages/ca/ec/c5c618767bcdf66e88945ec0157d7f6c4a1322f1473392319b7a2501ded7/frozenlist-1.8.0-cp313-cp313t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:5500ef82073f599ac84d888e3a8c1f77ac831183244bfd7f11eaa0289fb30714", size = 294676, upload-time = "2025-10-06T05:36:55.566Z" }, + { url = "https://files.pythonhosted.org/packages/7c/ce/3934758637d8f8a88d11f0585d6495ef54b2044ed6ec84492a91fa3b27aa/frozenlist-1.8.0-cp313-cp313t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:50066c3997d0091c411a66e710f4e11752251e6d2d73d70d8d5d4c76442a199d", size = 300638, upload-time = "2025-10-06T05:36:56.758Z" }, + { url = "https://files.pythonhosted.org/packages/fc/4f/a7e4d0d467298f42de4b41cbc7ddaf19d3cfeabaf9ff97c20c6c7ee409f9/frozenlist-1.8.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:5c1c8e78426e59b3f8005e9b19f6ff46e5845895adbde20ece9218319eca6506", size = 283067, upload-time = "2025-10-06T05:36:57.965Z" }, + { url = "https://files.pythonhosted.org/packages/dc/48/c7b163063d55a83772b268e6d1affb960771b0e203b632cfe09522d67ea5/frozenlist-1.8.0-cp313-cp313t-musllinux_1_2_armv7l.whl", hash = "sha256:eefdba20de0d938cec6a89bd4d70f346a03108a19b9df4248d3cf0d88f1b0f51", size = 292101, upload-time = "2025-10-06T05:36:59.237Z" }, + { url = "https://files.pythonhosted.org/packages/9f/d0/2366d3c4ecdc2fd391e0afa6e11500bfba0ea772764d631bbf82f0136c9d/frozenlist-1.8.0-cp313-cp313t-musllinux_1_2_ppc64le.whl", hash = "sha256:cf253e0e1c3ceb4aaff6df637ce033ff6535fb8c70a764a8f46aafd3d6ab798e", size = 289901, upload-time = "2025-10-06T05:37:00.811Z" }, + { url = "https://files.pythonhosted.org/packages/b8/94/daff920e82c1b70e3618a2ac39fbc01ae3e2ff6124e80739ce5d71c9b920/frozenlist-1.8.0-cp313-cp313t-musllinux_1_2_s390x.whl", hash = "sha256:032efa2674356903cd0261c4317a561a6850f3ac864a63fc1583147fb05a79b0", size = 289395, upload-time = "2025-10-06T05:37:02.115Z" }, + { url = "https://files.pythonhosted.org/packages/e3/20/bba307ab4235a09fdcd3cc5508dbabd17c4634a1af4b96e0f69bfe551ebd/frozenlist-1.8.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:6da155091429aeba16851ecb10a9104a108bcd32f6c1642867eadaee401c1c41", size = 283659, upload-time = "2025-10-06T05:37:03.711Z" }, + { url = "https://files.pythonhosted.org/packages/fd/00/04ca1c3a7a124b6de4f8a9a17cc2fcad138b4608e7a3fc5877804b8715d7/frozenlist-1.8.0-cp313-cp313t-win32.whl", hash = "sha256:0f96534f8bfebc1a394209427d0f8a63d343c9779cda6fc25e8e121b5fd8555b", size = 43492, upload-time = "2025-10-06T05:37:04.915Z" }, + { url = "https://files.pythonhosted.org/packages/59/5e/c69f733a86a94ab10f68e496dc6b7e8bc078ebb415281d5698313e3af3a1/frozenlist-1.8.0-cp313-cp313t-win_amd64.whl", hash = "sha256:5d63a068f978fc69421fb0e6eb91a9603187527c86b7cd3f534a5b77a592b888", size = 48034, upload-time = "2025-10-06T05:37:06.343Z" }, + { url = "https://files.pythonhosted.org/packages/16/6c/be9d79775d8abe79b05fa6d23da99ad6e7763a1d080fbae7290b286093fd/frozenlist-1.8.0-cp313-cp313t-win_arm64.whl", hash = "sha256:bf0a7e10b077bf5fb9380ad3ae8ce20ef919a6ad93b4552896419ac7e1d8e042", size = 41749, upload-time = "2025-10-06T05:37:07.431Z" }, + { url = "https://files.pythonhosted.org/packages/f1/c8/85da824b7e7b9b6e7f7705b2ecaf9591ba6f79c1177f324c2735e41d36a2/frozenlist-1.8.0-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:cee686f1f4cadeb2136007ddedd0aaf928ab95216e7691c63e50a8ec066336d0", size = 86127, upload-time = "2025-10-06T05:37:08.438Z" }, + { url = "https://files.pythonhosted.org/packages/8e/e8/a1185e236ec66c20afd72399522f142c3724c785789255202d27ae992818/frozenlist-1.8.0-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:119fb2a1bd47307e899c2fac7f28e85b9a543864df47aa7ec9d3c1b4545f096f", size = 49698, upload-time = "2025-10-06T05:37:09.48Z" }, + { url = "https://files.pythonhosted.org/packages/a1/93/72b1736d68f03fda5fdf0f2180fb6caaae3894f1b854d006ac61ecc727ee/frozenlist-1.8.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:4970ece02dbc8c3a92fcc5228e36a3e933a01a999f7094ff7c23fbd2beeaa67c", size = 49749, upload-time = "2025-10-06T05:37:10.569Z" }, + { url = "https://files.pythonhosted.org/packages/a7/b2/fabede9fafd976b991e9f1b9c8c873ed86f202889b864756f240ce6dd855/frozenlist-1.8.0-cp314-cp314-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:cba69cb73723c3f329622e34bdbf5ce1f80c21c290ff04256cff1cd3c2036ed2", size = 231298, upload-time = "2025-10-06T05:37:11.993Z" }, + { url = "https://files.pythonhosted.org/packages/3a/3b/d9b1e0b0eed36e70477ffb8360c49c85c8ca8ef9700a4e6711f39a6e8b45/frozenlist-1.8.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:778a11b15673f6f1df23d9586f83c4846c471a8af693a22e066508b77d201ec8", size = 232015, upload-time = "2025-10-06T05:37:13.194Z" }, + { url = "https://files.pythonhosted.org/packages/dc/94/be719d2766c1138148564a3960fc2c06eb688da592bdc25adcf856101be7/frozenlist-1.8.0-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:0325024fe97f94c41c08872db482cf8ac4800d80e79222c6b0b7b162d5b13686", size = 225038, upload-time = "2025-10-06T05:37:14.577Z" }, + { url = "https://files.pythonhosted.org/packages/e4/09/6712b6c5465f083f52f50cf74167b92d4ea2f50e46a9eea0523d658454ae/frozenlist-1.8.0-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:97260ff46b207a82a7567b581ab4190bd4dfa09f4db8a8b49d1a958f6aa4940e", size = 240130, upload-time = "2025-10-06T05:37:15.781Z" }, + { url = "https://files.pythonhosted.org/packages/f8/d4/cd065cdcf21550b54f3ce6a22e143ac9e4836ca42a0de1022da8498eac89/frozenlist-1.8.0-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:54b2077180eb7f83dd52c40b2750d0a9f175e06a42e3213ce047219de902717a", size = 242845, upload-time = "2025-10-06T05:37:17.037Z" }, + { url = "https://files.pythonhosted.org/packages/62/c3/f57a5c8c70cd1ead3d5d5f776f89d33110b1addae0ab010ad774d9a44fb9/frozenlist-1.8.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:2f05983daecab868a31e1da44462873306d3cbfd76d1f0b5b69c473d21dbb128", size = 229131, upload-time = "2025-10-06T05:37:18.221Z" }, + { url = "https://files.pythonhosted.org/packages/6c/52/232476fe9cb64f0742f3fde2b7d26c1dac18b6d62071c74d4ded55e0ef94/frozenlist-1.8.0-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:33f48f51a446114bc5d251fb2954ab0164d5be02ad3382abcbfe07e2531d650f", size = 240542, upload-time = "2025-10-06T05:37:19.771Z" }, + { url = "https://files.pythonhosted.org/packages/5f/85/07bf3f5d0fb5414aee5f47d33c6f5c77bfe49aac680bfece33d4fdf6a246/frozenlist-1.8.0-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:154e55ec0655291b5dd1b8731c637ecdb50975a2ae70c606d100750a540082f7", size = 237308, upload-time = "2025-10-06T05:37:20.969Z" }, + { url = "https://files.pythonhosted.org/packages/11/99/ae3a33d5befd41ac0ca2cc7fd3aa707c9c324de2e89db0e0f45db9a64c26/frozenlist-1.8.0-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:4314debad13beb564b708b4a496020e5306c7333fa9a3ab90374169a20ffab30", size = 238210, upload-time = "2025-10-06T05:37:22.252Z" }, + { url = "https://files.pythonhosted.org/packages/b2/60/b1d2da22f4970e7a155f0adde9b1435712ece01b3cd45ba63702aea33938/frozenlist-1.8.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:073f8bf8becba60aa931eb3bc420b217bb7d5b8f4750e6f8b3be7f3da85d38b7", size = 231972, upload-time = "2025-10-06T05:37:23.5Z" }, + { url = "https://files.pythonhosted.org/packages/3f/ab/945b2f32de889993b9c9133216c068b7fcf257d8595a0ac420ac8677cab0/frozenlist-1.8.0-cp314-cp314-win32.whl", hash = "sha256:bac9c42ba2ac65ddc115d930c78d24ab8d4f465fd3fc473cdedfccadb9429806", size = 40536, upload-time = "2025-10-06T05:37:25.581Z" }, + { url = "https://files.pythonhosted.org/packages/59/ad/9caa9b9c836d9ad6f067157a531ac48b7d36499f5036d4141ce78c230b1b/frozenlist-1.8.0-cp314-cp314-win_amd64.whl", hash = "sha256:3e0761f4d1a44f1d1a47996511752cf3dcec5bbdd9cc2b4fe595caf97754b7a0", size = 44330, upload-time = "2025-10-06T05:37:26.928Z" }, + { url = "https://files.pythonhosted.org/packages/82/13/e6950121764f2676f43534c555249f57030150260aee9dcf7d64efda11dd/frozenlist-1.8.0-cp314-cp314-win_arm64.whl", hash = "sha256:d1eaff1d00c7751b7c6662e9c5ba6eb2c17a2306ba5e2a37f24ddf3cc953402b", size = 40627, upload-time = "2025-10-06T05:37:28.075Z" }, + { url = "https://files.pythonhosted.org/packages/c0/c7/43200656ecc4e02d3f8bc248df68256cd9572b3f0017f0a0c4e93440ae23/frozenlist-1.8.0-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:d3bb933317c52d7ea5004a1c442eef86f426886fba134ef8cf4226ea6ee1821d", size = 89238, upload-time = "2025-10-06T05:37:29.373Z" }, + { url = "https://files.pythonhosted.org/packages/d1/29/55c5f0689b9c0fb765055629f472c0de484dcaf0acee2f7707266ae3583c/frozenlist-1.8.0-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:8009897cdef112072f93a0efdce29cd819e717fd2f649ee3016efd3cd885a7ed", size = 50738, upload-time = "2025-10-06T05:37:30.792Z" }, + { url = "https://files.pythonhosted.org/packages/ba/7d/b7282a445956506fa11da8c2db7d276adcbf2b17d8bb8407a47685263f90/frozenlist-1.8.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:2c5dcbbc55383e5883246d11fd179782a9d07a986c40f49abe89ddf865913930", size = 51739, upload-time = "2025-10-06T05:37:32.127Z" }, + { url = "https://files.pythonhosted.org/packages/62/1c/3d8622e60d0b767a5510d1d3cf21065b9db874696a51ea6d7a43180a259c/frozenlist-1.8.0-cp314-cp314t-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:39ecbc32f1390387d2aa4f5a995e465e9e2f79ba3adcac92d68e3e0afae6657c", size = 284186, upload-time = "2025-10-06T05:37:33.21Z" }, + { url = "https://files.pythonhosted.org/packages/2d/14/aa36d5f85a89679a85a1d44cd7a6657e0b1c75f61e7cad987b203d2daca8/frozenlist-1.8.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:92db2bf818d5cc8d9c1f1fc56b897662e24ea5adb36ad1f1d82875bd64e03c24", size = 292196, upload-time = "2025-10-06T05:37:36.107Z" }, + { url = "https://files.pythonhosted.org/packages/05/23/6bde59eb55abd407d34f77d39a5126fb7b4f109a3f611d3929f14b700c66/frozenlist-1.8.0-cp314-cp314t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:2dc43a022e555de94c3b68a4ef0b11c4f747d12c024a520c7101709a2144fb37", size = 273830, upload-time = "2025-10-06T05:37:37.663Z" }, + { url = "https://files.pythonhosted.org/packages/d2/3f/22cff331bfad7a8afa616289000ba793347fcd7bc275f3b28ecea2a27909/frozenlist-1.8.0-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:cb89a7f2de3602cfed448095bab3f178399646ab7c61454315089787df07733a", size = 294289, upload-time = "2025-10-06T05:37:39.261Z" }, + { url = "https://files.pythonhosted.org/packages/a4/89/5b057c799de4838b6c69aa82b79705f2027615e01be996d2486a69ca99c4/frozenlist-1.8.0-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:33139dc858c580ea50e7e60a1b0ea003efa1fd42e6ec7fdbad78fff65fad2fd2", size = 300318, upload-time = "2025-10-06T05:37:43.213Z" }, + { url = "https://files.pythonhosted.org/packages/30/de/2c22ab3eb2a8af6d69dc799e48455813bab3690c760de58e1bf43b36da3e/frozenlist-1.8.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:168c0969a329b416119507ba30b9ea13688fafffac1b7822802537569a1cb0ef", size = 282814, upload-time = "2025-10-06T05:37:45.337Z" }, + { url = "https://files.pythonhosted.org/packages/59/f7/970141a6a8dbd7f556d94977858cfb36fa9b66e0892c6dd780d2219d8cd8/frozenlist-1.8.0-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:28bd570e8e189d7f7b001966435f9dac6718324b5be2990ac496cf1ea9ddb7fe", size = 291762, upload-time = "2025-10-06T05:37:46.657Z" }, + { url = "https://files.pythonhosted.org/packages/c1/15/ca1adae83a719f82df9116d66f5bb28bb95557b3951903d39135620ef157/frozenlist-1.8.0-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:b2a095d45c5d46e5e79ba1e5b9cb787f541a8dee0433836cea4b96a2c439dcd8", size = 289470, upload-time = "2025-10-06T05:37:47.946Z" }, + { url = "https://files.pythonhosted.org/packages/ac/83/dca6dc53bf657d371fbc88ddeb21b79891e747189c5de990b9dfff2ccba1/frozenlist-1.8.0-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:eab8145831a0d56ec9c4139b6c3e594c7a83c2c8be25d5bcf2d86136a532287a", size = 289042, upload-time = "2025-10-06T05:37:49.499Z" }, + { url = "https://files.pythonhosted.org/packages/96/52/abddd34ca99be142f354398700536c5bd315880ed0a213812bc491cff5e4/frozenlist-1.8.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:974b28cf63cc99dfb2188d8d222bc6843656188164848c4f679e63dae4b0708e", size = 283148, upload-time = "2025-10-06T05:37:50.745Z" }, + { url = "https://files.pythonhosted.org/packages/af/d3/76bd4ed4317e7119c2b7f57c3f6934aba26d277acc6309f873341640e21f/frozenlist-1.8.0-cp314-cp314t-win32.whl", hash = "sha256:342c97bf697ac5480c0a7ec73cd700ecfa5a8a40ac923bd035484616efecc2df", size = 44676, upload-time = "2025-10-06T05:37:52.222Z" }, + { url = "https://files.pythonhosted.org/packages/89/76/c615883b7b521ead2944bb3480398cbb07e12b7b4e4d073d3752eb721558/frozenlist-1.8.0-cp314-cp314t-win_amd64.whl", hash = "sha256:06be8f67f39c8b1dc671f5d83aaefd3358ae5cdcf8314552c57e7ed3e6475bdd", size = 49451, upload-time = "2025-10-06T05:37:53.425Z" }, + { url = "https://files.pythonhosted.org/packages/e0/a3/5982da14e113d07b325230f95060e2169f5311b1017ea8af2a29b374c289/frozenlist-1.8.0-cp314-cp314t-win_arm64.whl", hash = "sha256:102e6314ca4da683dca92e3b1355490fed5f313b768500084fbe6371fddfdb79", size = 42507, upload-time = "2025-10-06T05:37:54.513Z" }, + { url = "https://files.pythonhosted.org/packages/c2/59/ae5cdac87a00962122ea37bb346d41b66aec05f9ce328fa2b9e216f8967b/frozenlist-1.8.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:d8b7138e5cd0647e4523d6685b0eac5d4be9a184ae9634492f25c6eb38c12a47", size = 86967, upload-time = "2025-10-06T05:37:55.607Z" }, + { url = "https://files.pythonhosted.org/packages/8a/10/17059b2db5a032fd9323c41c39e9d1f5f9d0c8f04d1e4e3e788573086e61/frozenlist-1.8.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:a6483e309ca809f1efd154b4d37dc6d9f61037d6c6a81c2dc7a15cb22c8c5dca", size = 49984, upload-time = "2025-10-06T05:37:57.049Z" }, + { url = "https://files.pythonhosted.org/packages/4b/de/ad9d82ca8e5fa8f0c636e64606553c79e2b859ad253030b62a21fe9986f5/frozenlist-1.8.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1b9290cf81e95e93fdf90548ce9d3c1211cf574b8e3f4b3b7cb0537cf2227068", size = 50240, upload-time = "2025-10-06T05:37:58.145Z" }, + { url = "https://files.pythonhosted.org/packages/4e/45/3dfb7767c2a67d123650122b62ce13c731b6c745bc14424eea67678b508c/frozenlist-1.8.0-cp39-cp39-manylinux1_x86_64.manylinux_2_28_x86_64.manylinux_2_5_x86_64.whl", hash = "sha256:59a6a5876ca59d1b63af8cd5e7ffffb024c3dc1e9cf9301b21a2e76286505c95", size = 219472, upload-time = "2025-10-06T05:37:59.239Z" }, + { url = "https://files.pythonhosted.org/packages/0b/bf/5bf23d913a741b960d5c1dac7c1985d8a2a1d015772b2d18ea168b08e7ff/frozenlist-1.8.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6dc4126390929823e2d2d9dc79ab4046ed74680360fc5f38b585c12c66cdf459", size = 221531, upload-time = "2025-10-06T05:38:00.521Z" }, + { url = "https://files.pythonhosted.org/packages/d0/03/27ec393f3b55860859f4b74cdc8c2a4af3dbf3533305e8eacf48a4fd9a54/frozenlist-1.8.0-cp39-cp39-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:332db6b2563333c5671fecacd085141b5800cb866be16d5e3eb15a2086476675", size = 219211, upload-time = "2025-10-06T05:38:01.842Z" }, + { url = "https://files.pythonhosted.org/packages/3a/ad/0fd00c404fa73fe9b169429e9a972d5ed807973c40ab6b3cf9365a33d360/frozenlist-1.8.0-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:9ff15928d62a0b80bb875655c39bf517938c7d589554cbd2669be42d97c2cb61", size = 231775, upload-time = "2025-10-06T05:38:03.384Z" }, + { url = "https://files.pythonhosted.org/packages/8a/c3/86962566154cb4d2995358bc8331bfc4ea19d07db1a96f64935a1607f2b6/frozenlist-1.8.0-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:7bf6cdf8e07c8151fba6fe85735441240ec7f619f935a5205953d58009aef8c6", size = 236631, upload-time = "2025-10-06T05:38:04.609Z" }, + { url = "https://files.pythonhosted.org/packages/ea/9e/6ffad161dbd83782d2c66dc4d378a9103b31770cb1e67febf43aea42d202/frozenlist-1.8.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:48e6d3f4ec5c7273dfe83ff27c91083c6c9065af655dc2684d2c200c94308bb5", size = 218632, upload-time = "2025-10-06T05:38:05.917Z" }, + { url = "https://files.pythonhosted.org/packages/58/b2/4677eee46e0a97f9b30735e6ad0bf6aba3e497986066eb68807ac85cf60f/frozenlist-1.8.0-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:1a7607e17ad33361677adcd1443edf6f5da0ce5e5377b798fba20fae194825f3", size = 235967, upload-time = "2025-10-06T05:38:07.614Z" }, + { url = "https://files.pythonhosted.org/packages/05/f3/86e75f8639c5a93745ca7addbbc9de6af56aebb930d233512b17e46f6493/frozenlist-1.8.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:5a3a935c3a4e89c733303a2d5a7c257ea44af3a56c8202df486b7f5de40f37e1", size = 228799, upload-time = "2025-10-06T05:38:08.845Z" }, + { url = "https://files.pythonhosted.org/packages/30/00/39aad3a7f0d98f5eb1d99a3c311215674ed87061aecee7851974b335c050/frozenlist-1.8.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:940d4a017dbfed9daf46a3b086e1d2167e7012ee297fef9e1c545c4d022f5178", size = 230566, upload-time = "2025-10-06T05:38:10.52Z" }, + { url = "https://files.pythonhosted.org/packages/0d/4d/aa144cac44568d137846ddc4d5210fb5d9719eb1d7ec6fa2728a54b5b94a/frozenlist-1.8.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:b9be22a69a014bc47e78072d0ecae716f5eb56c15238acca0f43d6eb8e4a5bda", size = 217715, upload-time = "2025-10-06T05:38:11.832Z" }, + { url = "https://files.pythonhosted.org/packages/64/4c/8f665921667509d25a0dd72540513bc86b356c95541686f6442a3283019f/frozenlist-1.8.0-cp39-cp39-win32.whl", hash = "sha256:1aa77cb5697069af47472e39612976ed05343ff2e84a3dcf15437b232cbfd087", size = 39933, upload-time = "2025-10-06T05:38:13.061Z" }, + { url = "https://files.pythonhosted.org/packages/79/bd/bcc926f87027fad5e59926ff12d136e1082a115025d33c032d1cd69ab377/frozenlist-1.8.0-cp39-cp39-win_amd64.whl", hash = "sha256:7398c222d1d405e796970320036b1b563892b65809d9e5261487bb2c7f7b5c6a", size = 44121, upload-time = "2025-10-06T05:38:14.572Z" }, + { url = "https://files.pythonhosted.org/packages/4c/07/9c2e4eb7584af4b705237b971b89a4155a8e57599c4483a131a39256a9a0/frozenlist-1.8.0-cp39-cp39-win_arm64.whl", hash = "sha256:b4f3b365f31c6cd4af24545ca0a244a53688cad8834e32f56831c4923b50a103", size = 40312, upload-time = "2025-10-06T05:38:15.699Z" }, + { 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-auth" +version = "2.41.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "cachetools", version = "5.5.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "cachetools", version = "6.2.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "pyasn1-modules" }, + { name = "rsa" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/a8/af/5129ce5b2f9688d2fa49b463e544972a7c82b0fdb50980dafee92e121d9f/google_auth-2.41.1.tar.gz", hash = "sha256:b76b7b1f9e61f0cb7e88870d14f6a94aeef248959ef6992670efee37709cbfd2", size = 292284, upload-time = "2025-09-30T22:51:26.363Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/be/a4/7319a2a8add4cc352be9e3efeff5e2aacee917c85ca2fa1647e29089983c/google_auth-2.41.1-py2.py3-none-any.whl", hash = "sha256:754843be95575b9a19c604a848a41be03f7f2afd8c019f716dc1f51ee41c639d", size = 221302, upload-time = "2025-09-30T22:51:24.212Z" }, +] + +[[package]] +name = "h11" +version = "0.16.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/01/ee/02a2c011bdab74c6fb3c75474d40b3052059d95df7e73351460c8588d963/h11-0.16.0.tar.gz", hash = "sha256:4e35b956cf45792e4caa5885e69fba00bdbc6ffafbfa020300e549b208ee5ff1", size = 101250, upload-time = "2025-04-24T03:35:25.427Z" } +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 = "httpcore" +version = "1.0.9" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "certifi" }, + { name = "h11" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/06/94/82699a10bca87a5556c9c59b5963f2d039dbd239f25bc2a63907a05a14cb/httpcore-1.0.9.tar.gz", hash = "sha256:6e34463af53fd2ab5d807f399a9b45ea31c3dfa2276f15a2c3f00afff6e176e8", size = 85484, upload-time = "2025-04-24T22:06:22.219Z" } +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 = "httpx" +version = "0.28.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "anyio", version = "4.5.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "anyio", version = "4.11.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "certifi" }, + { name = "httpcore" }, + { name = "idna" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/b1/df/48c586a5fe32a0f01324ee087459e112ebb7224f646c0b5023f5e79e9956/httpx-0.28.1.tar.gz", hash = "sha256:75e98c5f16b0f35b567856f597f06ff2270a374470a5c2392242528e3e3e42fc", size = 141406, upload-time = "2024-12-06T15:37:23.222Z" } +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]] +name = "idna" +version = "3.11" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/6f/6d/0703ccc57f3a7233505399edb88de3cbd678da106337b9fcde432b65ed60/idna-3.11.tar.gz", hash = "sha256:795dafcc9c04ed0c1fb032c2aa73654d8e8c5023a7df64a53f39190ada629902", size = 194582, upload-time = "2025-10-12T14:55:20.501Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/0e/61/66938bbb5fc52dbdf84594873d5b51fb1f7c7794e9c0f5bd885f30bc507b/idna-3.11-py3-none-any.whl", hash = "sha256:771a87f49d9defaf64091e6e6fe9c18d4833f140bd19464795bc32d966ca37ea", size = 71008, upload-time = "2025-10-12T14:55:18.883Z" }, +] + +[[package]] +name = "importlib-metadata" +version = "8.5.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "zipp", version = "3.20.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/cd/12/33e59336dca5be0c398a7482335911a33aa0e20776128f038019f1a95f1b/importlib_metadata-8.5.0.tar.gz", hash = "sha256:71522656f0abace1d072b9e5481a48f07c138e00f079c38c8f883823f9c26bd7", size = 55304, upload-time = "2024-09-11T14:56:08.937Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a0/d9/a1e041c5e7caa9a05c925f4bdbdfb7f006d1f74996af53467bc394c97be7/importlib_metadata-8.5.0-py3-none-any.whl", hash = "sha256:45e54197d28b7a7f1559e60b95e7c567032b602131fbd588f1497f47880aa68b", size = 26514, upload-time = "2024-09-11T14:56:07.019Z" }, +] + +[[package]] +name = "importlib-metadata" +version = "8.7.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "zipp", version = "3.23.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/76/66/650a33bd90f786193e4de4b3ad86ea60b53c89b669a5c7be931fac31cdb0/importlib_metadata-8.7.0.tar.gz", hash = "sha256:d13b81ad223b890aa16c5471f2ac3056cf76c5f10f82d6f9292f0b415f389000", size = 56641, upload-time = "2025-04-27T15:29:01.736Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/20/b0/36bd937216ec521246249be3bf9855081de4c5e06a0c9b4219dbeda50373/importlib_metadata-8.7.0-py3-none-any.whl", hash = "sha256:e5dd1551894c77868a30651cef00984d50e1002d06942a7101d34870c5f02afd", size = 27656, upload-time = "2025-04-27T15:29:00.214Z" }, +] + +[[package]] +name = "iniconfig" +version = "2.1.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version == '3.9.*'", + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/f2/97/ebf4da567aa6827c909642694d71c9fcf53e5b504f2d96afea02718862f3/iniconfig-2.1.0.tar.gz", hash = "sha256:3abbd2e30b36733fee78f9c7f7308f2d0050e88f0087fd25c2645f63c773e1c7", size = 4793, upload-time = "2025-03-19T20:09:59.721Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/2c/e1/e6716421ea10d38022b952c159d5161ca1193197fb744506875fbb87ea7b/iniconfig-2.1.0-py3-none-any.whl", hash = "sha256:9deba5723312380e77435581c6bf4935c94cbfab9b1ed33ef8d238ea168eb760", size = 6050, upload-time = "2025-03-19T20:10:01.071Z" }, +] + +[[package]] +name = "iniconfig" +version = "2.3.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", +] +sdist = { url = "https://files.pythonhosted.org/packages/72/34/14ca021ce8e5dfedc35312d08ba8bf51fdd999c576889fc2c24cb97f4f10/iniconfig-2.3.0.tar.gz", hash = "sha256:c76315c77db068650d49c5b56314774a7804df16fee4402c1f19d6d15d8c4730", size = 20503, upload-time = "2025-10-18T21:55:43.219Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/cb/b1/3846dd7f199d53cb17f49cba7e651e9ce294d8497c8c150530ed11865bb8/iniconfig-2.3.0-py3-none-any.whl", hash = "sha256:f631c04d2c48c52b84d0d0549c99ff3859c98df65b3101406327ecc7d53fbf12", size = 7484, upload-time = "2025-10-18T21:55:41.639Z" }, +] + +[[package]] +name = "itsdangerous" +version = "2.2.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/9c/cb/8ac0172223afbccb63986cc25049b154ecfb5e85932587206f42317be31d/itsdangerous-2.2.0.tar.gz", hash = "sha256:e0050c0b7da1eea53ffaf149c0cfbb5c6e2e2b69c4bef22c81fa6eb73e5f6173", size = 54410, upload-time = "2024-04-16T21:28:15.614Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/04/96/92447566d16df59b2a776c0fb82dbc4d9e07cd95062562af01e408583fc4/itsdangerous-2.2.0-py3-none-any.whl", hash = "sha256:c6242fc49e35958c8b15141343aa660db5fc54d4f13a1db01a3f5891b98700ef", size = 16234, upload-time = "2024-04-16T21:28:14.499Z" }, +] + +[[package]] +name = "jinja2" +version = "3.1.6" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "markupsafe", version = "2.1.5", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "markupsafe", version = "3.0.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115, upload-time = "2025-03-05T20:05:02.478Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899, upload-time = "2025-03-05T20:05:00.369Z" }, +] + +[[package]] +name = "markupsafe" +version = "2.1.5" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/87/5b/aae44c6655f3801e81aa3eef09dbbf012431987ba564d7231722f68df02d/MarkupSafe-2.1.5.tar.gz", hash = "sha256:d283d37a890ba4c1ae73ffadf8046435c76e7bc2247bbb63c00bd1a709c6544b", size = 19384, upload-time = "2024-02-02T16:31:22.863Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e4/54/ad5eb37bf9d51800010a74e4665425831a9db4e7c4e0fde4352e391e808e/MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a17a92de5231666cfbe003f0e4b9b3a7ae3afb1ec2845aadc2bacc93ff85febc", size = 18206, upload-time = "2024-02-02T16:30:04.105Z" }, + { url = "https://files.pythonhosted.org/packages/6a/4a/a4d49415e600bacae038c67f9fecc1d5433b9d3c71a4de6f33537b89654c/MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:72b6be590cc35924b02c78ef34b467da4ba07e4e0f0454a2c5907f473fc50ce5", size = 14079, upload-time = "2024-02-02T16:30:06.5Z" }, + { url = "https://files.pythonhosted.org/packages/0a/7b/85681ae3c33c385b10ac0f8dd025c30af83c78cec1c37a6aa3b55e67f5ec/MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e61659ba32cf2cf1481e575d0462554625196a1f2fc06a1c777d3f48e8865d46", size = 26620, upload-time = "2024-02-02T16:30:08.31Z" }, + { url = "https://files.pythonhosted.org/packages/7c/52/2b1b570f6b8b803cef5ac28fdf78c0da318916c7d2fe9402a84d591b394c/MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2174c595a0d73a3080ca3257b40096db99799265e1c27cc5a610743acd86d62f", size = 25818, upload-time = "2024-02-02T16:30:09.577Z" }, + { url = "https://files.pythonhosted.org/packages/29/fe/a36ba8c7ca55621620b2d7c585313efd10729e63ef81e4e61f52330da781/MarkupSafe-2.1.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ae2ad8ae6ebee9d2d94b17fb62763125f3f374c25618198f40cbb8b525411900", size = 25493, upload-time = "2024-02-02T16:30:11.488Z" }, + { url = "https://files.pythonhosted.org/packages/60/ae/9c60231cdfda003434e8bd27282b1f4e197ad5a710c14bee8bea8a9ca4f0/MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:075202fa5b72c86ad32dc7d0b56024ebdbcf2048c0ba09f1cde31bfdd57bcfff", size = 30630, upload-time = "2024-02-02T16:30:13.144Z" }, + { url = "https://files.pythonhosted.org/packages/65/dc/1510be4d179869f5dafe071aecb3f1f41b45d37c02329dfba01ff59e5ac5/MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:598e3276b64aff0e7b3451b72e94fa3c238d452e7ddcd893c3ab324717456bad", size = 29745, upload-time = "2024-02-02T16:30:14.222Z" }, + { url = "https://files.pythonhosted.org/packages/30/39/8d845dd7d0b0613d86e0ef89549bfb5f61ed781f59af45fc96496e897f3a/MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fce659a462a1be54d2ffcacea5e3ba2d74daa74f30f5f143fe0c58636e355fdd", size = 30021, upload-time = "2024-02-02T16:30:16.032Z" }, + { url = "https://files.pythonhosted.org/packages/c7/5c/356a6f62e4f3c5fbf2602b4771376af22a3b16efa74eb8716fb4e328e01e/MarkupSafe-2.1.5-cp310-cp310-win32.whl", hash = "sha256:d9fad5155d72433c921b782e58892377c44bd6252b5af2f67f16b194987338a4", size = 16659, upload-time = "2024-02-02T16:30:17.079Z" }, + { url = "https://files.pythonhosted.org/packages/69/48/acbf292615c65f0604a0c6fc402ce6d8c991276e16c80c46a8f758fbd30c/MarkupSafe-2.1.5-cp310-cp310-win_amd64.whl", hash = "sha256:bf50cd79a75d181c9181df03572cdce0fbb75cc353bc350712073108cba98de5", size = 17213, upload-time = "2024-02-02T16:30:18.251Z" }, + { url = "https://files.pythonhosted.org/packages/11/e7/291e55127bb2ae67c64d66cef01432b5933859dfb7d6949daa721b89d0b3/MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:629ddd2ca402ae6dbedfceeba9c46d5f7b2a61d9749597d4307f943ef198fc1f", size = 18219, upload-time = "2024-02-02T16:30:19.988Z" }, + { url = "https://files.pythonhosted.org/packages/6b/cb/aed7a284c00dfa7c0682d14df85ad4955a350a21d2e3b06d8240497359bf/MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:5b7b716f97b52c5a14bffdf688f971b2d5ef4029127f1ad7a513973cfd818df2", size = 14098, upload-time = "2024-02-02T16:30:21.063Z" }, + { url = "https://files.pythonhosted.org/packages/1c/cf/35fe557e53709e93feb65575c93927942087e9b97213eabc3fe9d5b25a55/MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ec585f69cec0aa07d945b20805be741395e28ac1627333b1c5b0105962ffced", size = 29014, upload-time = "2024-02-02T16:30:22.926Z" }, + { url = "https://files.pythonhosted.org/packages/97/18/c30da5e7a0e7f4603abfc6780574131221d9148f323752c2755d48abad30/MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b91c037585eba9095565a3556f611e3cbfaa42ca1e865f7b8015fe5c7336d5a5", size = 28220, upload-time = "2024-02-02T16:30:24.76Z" }, + { url = "https://files.pythonhosted.org/packages/0c/40/2e73e7d532d030b1e41180807a80d564eda53babaf04d65e15c1cf897e40/MarkupSafe-2.1.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7502934a33b54030eaf1194c21c692a534196063db72176b0c4028e140f8f32c", size = 27756, upload-time = "2024-02-02T16:30:25.877Z" }, + { url = "https://files.pythonhosted.org/packages/18/46/5dca760547e8c59c5311b332f70605d24c99d1303dd9a6e1fc3ed0d73561/MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0e397ac966fdf721b2c528cf028494e86172b4feba51d65f81ffd65c63798f3f", size = 33988, upload-time = "2024-02-02T16:30:26.935Z" }, + { url = "https://files.pythonhosted.org/packages/6d/c5/27febe918ac36397919cd4a67d5579cbbfa8da027fa1238af6285bb368ea/MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:c061bb86a71b42465156a3ee7bd58c8c2ceacdbeb95d05a99893e08b8467359a", size = 32718, upload-time = "2024-02-02T16:30:28.111Z" }, + { url = "https://files.pythonhosted.org/packages/f8/81/56e567126a2c2bc2684d6391332e357589a96a76cb9f8e5052d85cb0ead8/MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3a57fdd7ce31c7ff06cdfbf31dafa96cc533c21e443d57f5b1ecc6cdc668ec7f", size = 33317, upload-time = "2024-02-02T16:30:29.214Z" }, + { url = "https://files.pythonhosted.org/packages/00/0b/23f4b2470accb53285c613a3ab9ec19dc944eaf53592cb6d9e2af8aa24cc/MarkupSafe-2.1.5-cp311-cp311-win32.whl", hash = "sha256:397081c1a0bfb5124355710fe79478cdbeb39626492b15d399526ae53422b906", size = 16670, upload-time = "2024-02-02T16:30:30.915Z" }, + { url = "https://files.pythonhosted.org/packages/b7/a2/c78a06a9ec6d04b3445a949615c4c7ed86a0b2eb68e44e7541b9d57067cc/MarkupSafe-2.1.5-cp311-cp311-win_amd64.whl", hash = "sha256:2b7c57a4dfc4f16f7142221afe5ba4e093e09e728ca65c51f5620c9aaeb9a617", size = 17224, upload-time = "2024-02-02T16:30:32.09Z" }, + { url = "https://files.pythonhosted.org/packages/53/bd/583bf3e4c8d6a321938c13f49d44024dbe5ed63e0a7ba127e454a66da974/MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:8dec4936e9c3100156f8a2dc89c4b88d5c435175ff03413b443469c7c8c5f4d1", size = 18215, upload-time = "2024-02-02T16:30:33.081Z" }, + { url = "https://files.pythonhosted.org/packages/48/d6/e7cd795fc710292c3af3a06d80868ce4b02bfbbf370b7cee11d282815a2a/MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:3c6b973f22eb18a789b1460b4b91bf04ae3f0c4234a0a6aa6b0a92f6f7b951d4", size = 14069, upload-time = "2024-02-02T16:30:34.148Z" }, + { url = "https://files.pythonhosted.org/packages/51/b5/5d8ec796e2a08fc814a2c7d2584b55f889a55cf17dd1a90f2beb70744e5c/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ac07bad82163452a6884fe8fa0963fb98c2346ba78d779ec06bd7a6262132aee", size = 29452, upload-time = "2024-02-02T16:30:35.149Z" }, + { url = "https://files.pythonhosted.org/packages/0a/0d/2454f072fae3b5a137c119abf15465d1771319dfe9e4acbb31722a0fff91/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f5dfb42c4604dddc8e4305050aa6deb084540643ed5804d7455b5df8fe16f5e5", size = 28462, upload-time = "2024-02-02T16:30:36.166Z" }, + { url = "https://files.pythonhosted.org/packages/2d/75/fd6cb2e68780f72d47e6671840ca517bda5ef663d30ada7616b0462ad1e3/MarkupSafe-2.1.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ea3d8a3d18833cf4304cd2fc9cbb1efe188ca9b5efef2bdac7adc20594a0e46b", size = 27869, upload-time = "2024-02-02T16:30:37.834Z" }, + { url = "https://files.pythonhosted.org/packages/b0/81/147c477391c2750e8fc7705829f7351cf1cd3be64406edcf900dc633feb2/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d050b3361367a06d752db6ead6e7edeb0009be66bc3bae0ee9d97fb326badc2a", size = 33906, upload-time = "2024-02-02T16:30:39.366Z" }, + { url = "https://files.pythonhosted.org/packages/8b/ff/9a52b71839d7a256b563e85d11050e307121000dcebc97df120176b3ad93/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:bec0a414d016ac1a18862a519e54b2fd0fc8bbfd6890376898a6c0891dd82e9f", size = 32296, upload-time = "2024-02-02T16:30:40.413Z" }, + { url = "https://files.pythonhosted.org/packages/88/07/2dc76aa51b481eb96a4c3198894f38b480490e834479611a4053fbf08623/MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:58c98fee265677f63a4385256a6d7683ab1832f3ddd1e66fe948d5880c21a169", size = 33038, upload-time = "2024-02-02T16:30:42.243Z" }, + { url = "https://files.pythonhosted.org/packages/96/0c/620c1fb3661858c0e37eb3cbffd8c6f732a67cd97296f725789679801b31/MarkupSafe-2.1.5-cp312-cp312-win32.whl", hash = "sha256:8590b4ae07a35970728874632fed7bd57b26b0102df2d2b233b6d9d82f6c62ad", size = 16572, upload-time = "2024-02-02T16:30:43.326Z" }, + { url = "https://files.pythonhosted.org/packages/3f/14/c3554d512d5f9100a95e737502f4a2323a1959f6d0d01e0d0997b35f7b10/MarkupSafe-2.1.5-cp312-cp312-win_amd64.whl", hash = "sha256:823b65d8706e32ad2df51ed89496147a42a2a6e01c13cfb6ffb8b1e92bc910bb", size = 17127, upload-time = "2024-02-02T16:30:44.418Z" }, + { url = "https://files.pythonhosted.org/packages/f8/ff/2c942a82c35a49df5de3a630ce0a8456ac2969691b230e530ac12314364c/MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:656f7526c69fac7f600bd1f400991cc282b417d17539a1b228617081106feb4a", size = 18192, upload-time = "2024-02-02T16:30:57.715Z" }, + { url = "https://files.pythonhosted.org/packages/4f/14/6f294b9c4f969d0c801a4615e221c1e084722ea6114ab2114189c5b8cbe0/MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:97cafb1f3cbcd3fd2b6fbfb99ae11cdb14deea0736fc2b0952ee177f2b813a46", size = 14072, upload-time = "2024-02-02T16:30:58.844Z" }, + { url = "https://files.pythonhosted.org/packages/81/d4/fd74714ed30a1dedd0b82427c02fa4deec64f173831ec716da11c51a50aa/MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f3fbcb7ef1f16e48246f704ab79d79da8a46891e2da03f8783a5b6fa41a9532", size = 26928, upload-time = "2024-02-02T16:30:59.922Z" }, + { url = "https://files.pythonhosted.org/packages/c7/bd/50319665ce81bb10e90d1cf76f9e1aa269ea6f7fa30ab4521f14d122a3df/MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fa9db3f79de01457b03d4f01b34cf91bc0048eb2c3846ff26f66687c2f6d16ab", size = 26106, upload-time = "2024-02-02T16:31:01.582Z" }, + { url = "https://files.pythonhosted.org/packages/4c/6f/f2b0f675635b05f6afd5ea03c094557bdb8622fa8e673387444fe8d8e787/MarkupSafe-2.1.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffee1f21e5ef0d712f9033568f8344d5da8cc2869dbd08d87c84656e6a2d2f68", size = 25781, upload-time = "2024-02-02T16:31:02.71Z" }, + { url = "https://files.pythonhosted.org/packages/51/e0/393467cf899b34a9d3678e78961c2c8cdf49fb902a959ba54ece01273fb1/MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:5dedb4db619ba5a2787a94d877bc8ffc0566f92a01c0ef214865e54ecc9ee5e0", size = 30518, upload-time = "2024-02-02T16:31:04.392Z" }, + { url = "https://files.pythonhosted.org/packages/f6/02/5437e2ad33047290dafced9df741d9efc3e716b75583bbd73a9984f1b6f7/MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:30b600cf0a7ac9234b2638fbc0fb6158ba5bdcdf46aeb631ead21248b9affbc4", size = 29669, upload-time = "2024-02-02T16:31:05.53Z" }, + { url = "https://files.pythonhosted.org/packages/0e/7d/968284145ffd9d726183ed6237c77938c021abacde4e073020f920e060b2/MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8dd717634f5a044f860435c1d8c16a270ddf0ef8588d4887037c5028b859b0c3", size = 29933, upload-time = "2024-02-02T16:31:06.636Z" }, + { url = "https://files.pythonhosted.org/packages/bf/f3/ecb00fc8ab02b7beae8699f34db9357ae49d9f21d4d3de6f305f34fa949e/MarkupSafe-2.1.5-cp38-cp38-win32.whl", hash = "sha256:daa4ee5a243f0f20d528d939d06670a298dd39b1ad5f8a72a4275124a7819eff", size = 16656, upload-time = "2024-02-02T16:31:07.767Z" }, + { url = "https://files.pythonhosted.org/packages/92/21/357205f03514a49b293e214ac39de01fadd0970a6e05e4bf1ddd0ffd0881/MarkupSafe-2.1.5-cp38-cp38-win_amd64.whl", hash = "sha256:619bc166c4f2de5caa5a633b8b7326fbe98e0ccbfacabd87268a2b15ff73a029", size = 17206, upload-time = "2024-02-02T16:31:08.843Z" }, + { url = "https://files.pythonhosted.org/packages/0f/31/780bb297db036ba7b7bbede5e1d7f1e14d704ad4beb3ce53fb495d22bc62/MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:7a68b554d356a91cce1236aa7682dc01df0edba8d043fd1ce607c49dd3c1edcf", size = 18193, upload-time = "2024-02-02T16:31:10.155Z" }, + { url = "https://files.pythonhosted.org/packages/6c/77/d77701bbef72892affe060cdacb7a2ed7fd68dae3b477a8642f15ad3b132/MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:db0b55e0f3cc0be60c1f19efdde9a637c32740486004f20d1cff53c3c0ece4d2", size = 14073, upload-time = "2024-02-02T16:31:11.442Z" }, + { url = "https://files.pythonhosted.org/packages/d9/a7/1e558b4f78454c8a3a0199292d96159eb4d091f983bc35ef258314fe7269/MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3e53af139f8579a6d5f7b76549125f0d94d7e630761a2111bc431fd820e163b8", size = 26486, upload-time = "2024-02-02T16:31:12.488Z" }, + { url = "https://files.pythonhosted.org/packages/5f/5a/360da85076688755ea0cceb92472923086993e86b5613bbae9fbc14136b0/MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:17b950fccb810b3293638215058e432159d2b71005c74371d784862b7e4683f3", size = 25685, upload-time = "2024-02-02T16:31:13.726Z" }, + { url = "https://files.pythonhosted.org/packages/6a/18/ae5a258e3401f9b8312f92b028c54d7026a97ec3ab20bfaddbdfa7d8cce8/MarkupSafe-2.1.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4c31f53cdae6ecfa91a77820e8b151dba54ab528ba65dfd235c80b086d68a465", size = 25338, upload-time = "2024-02-02T16:31:14.812Z" }, + { url = "https://files.pythonhosted.org/packages/0b/cc/48206bd61c5b9d0129f4d75243b156929b04c94c09041321456fd06a876d/MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bff1b4290a66b490a2f4719358c0cdcd9bafb6b8f061e45c7a2460866bf50c2e", size = 30439, upload-time = "2024-02-02T16:31:15.946Z" }, + { url = "https://files.pythonhosted.org/packages/d1/06/a41c112ab9ffdeeb5f77bc3e331fdadf97fa65e52e44ba31880f4e7f983c/MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:bc1667f8b83f48511b94671e0e441401371dfd0f0a795c7daa4a3cd1dde55bea", size = 29531, upload-time = "2024-02-02T16:31:17.13Z" }, + { url = "https://files.pythonhosted.org/packages/02/8c/ab9a463301a50dab04d5472e998acbd4080597abc048166ded5c7aa768c8/MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5049256f536511ee3f7e1b3f87d1d1209d327e818e6ae1365e8653d7e3abb6a6", size = 29823, upload-time = "2024-02-02T16:31:18.247Z" }, + { url = "https://files.pythonhosted.org/packages/bc/29/9bc18da763496b055d8e98ce476c8e718dcfd78157e17f555ce6dd7d0895/MarkupSafe-2.1.5-cp39-cp39-win32.whl", hash = "sha256:00e046b6dd71aa03a41079792f8473dc494d564611a8f89bbbd7cb93295ebdcf", size = 16658, upload-time = "2024-02-02T16:31:19.583Z" }, + { url = "https://files.pythonhosted.org/packages/f6/f8/4da07de16f10551ca1f640c92b5f316f9394088b183c6a57183df6de5ae4/MarkupSafe-2.1.5-cp39-cp39-win_amd64.whl", hash = "sha256:fa173ec60341d6bb97a89f5ea19c85c5643c1e7dedebc22f5181eb73573142c5", size = 17211, upload-time = "2024-02-02T16:31:20.96Z" }, +] + +[[package]] +name = "markupsafe" +version = "3.0.3" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/7e/99/7690b6d4034fffd95959cbe0c02de8deb3098cc577c67bb6a24fe5d7caa7/markupsafe-3.0.3.tar.gz", hash = "sha256:722695808f4b6457b320fdc131280796bdceb04ab50fe1795cd540799ebe1698", size = 80313, upload-time = "2025-09-27T18:37:40.426Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/e8/4b/3541d44f3937ba468b75da9eebcae497dcf67adb65caa16760b0a6807ebb/markupsafe-3.0.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:2f981d352f04553a7171b8e44369f2af4055f888dfb147d55e42d29e29e74559", size = 11631, upload-time = "2025-09-27T18:36:05.558Z" }, + { url = "https://files.pythonhosted.org/packages/98/1b/fbd8eed11021cabd9226c37342fa6ca4e8a98d8188a8d9b66740494960e4/markupsafe-3.0.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:e1c1493fb6e50ab01d20a22826e57520f1284df32f2d8601fdd90b6304601419", size = 12057, upload-time = "2025-09-27T18:36:07.165Z" }, + { url = "https://files.pythonhosted.org/packages/40/01/e560d658dc0bb8ab762670ece35281dec7b6c1b33f5fbc09ebb57a185519/markupsafe-3.0.3-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1ba88449deb3de88bd40044603fafffb7bc2b055d626a330323a9ed736661695", size = 22050, upload-time = "2025-09-27T18:36:08.005Z" }, + { url = "https://files.pythonhosted.org/packages/af/cd/ce6e848bbf2c32314c9b237839119c5a564a59725b53157c856e90937b7a/markupsafe-3.0.3-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:f42d0984e947b8adf7dd6dde396e720934d12c506ce84eea8476409563607591", size = 20681, upload-time = "2025-09-27T18:36:08.881Z" }, + { url = "https://files.pythonhosted.org/packages/c9/2a/b5c12c809f1c3045c4d580b035a743d12fcde53cf685dbc44660826308da/markupsafe-3.0.3-cp310-cp310-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:c0c0b3ade1c0b13b936d7970b1d37a57acde9199dc2aecc4c336773e1d86049c", size = 20705, upload-time = "2025-09-27T18:36:10.131Z" }, + { url = "https://files.pythonhosted.org/packages/cf/e3/9427a68c82728d0a88c50f890d0fc072a1484de2f3ac1ad0bfc1a7214fd5/markupsafe-3.0.3-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:0303439a41979d9e74d18ff5e2dd8c43ed6c6001fd40e5bf2e43f7bd9bbc523f", size = 21524, upload-time = "2025-09-27T18:36:11.324Z" }, + { url = "https://files.pythonhosted.org/packages/bc/36/23578f29e9e582a4d0278e009b38081dbe363c5e7165113fad546918a232/markupsafe-3.0.3-cp310-cp310-musllinux_1_2_riscv64.whl", hash = "sha256:d2ee202e79d8ed691ceebae8e0486bd9a2cd4794cec4824e1c99b6f5009502f6", size = 20282, upload-time = "2025-09-27T18:36:12.573Z" }, + { url = "https://files.pythonhosted.org/packages/56/21/dca11354e756ebd03e036bd8ad58d6d7168c80ce1fe5e75218e4945cbab7/markupsafe-3.0.3-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:177b5253b2834fe3678cb4a5f0059808258584c559193998be2601324fdeafb1", size = 20745, upload-time = "2025-09-27T18:36:13.504Z" }, + { url = "https://files.pythonhosted.org/packages/87/99/faba9369a7ad6e4d10b6a5fbf71fa2a188fe4a593b15f0963b73859a1bbd/markupsafe-3.0.3-cp310-cp310-win32.whl", hash = "sha256:2a15a08b17dd94c53a1da0438822d70ebcd13f8c3a95abe3a9ef9f11a94830aa", size = 14571, upload-time = "2025-09-27T18:36:14.779Z" }, + { url = "https://files.pythonhosted.org/packages/d6/25/55dc3ab959917602c96985cb1253efaa4ff42f71194bddeb61eb7278b8be/markupsafe-3.0.3-cp310-cp310-win_amd64.whl", hash = "sha256:c4ffb7ebf07cfe8931028e3e4c85f0357459a3f9f9490886198848f4fa002ec8", size = 15056, upload-time = "2025-09-27T18:36:16.125Z" }, + { url = "https://files.pythonhosted.org/packages/d0/9e/0a02226640c255d1da0b8d12e24ac2aa6734da68bff14c05dd53b94a0fc3/markupsafe-3.0.3-cp310-cp310-win_arm64.whl", hash = "sha256:e2103a929dfa2fcaf9bb4e7c091983a49c9ac3b19c9061b6d5427dd7d14d81a1", size = 13932, upload-time = "2025-09-27T18:36:17.311Z" }, + { url = "https://files.pythonhosted.org/packages/08/db/fefacb2136439fc8dd20e797950e749aa1f4997ed584c62cfb8ef7c2be0e/markupsafe-3.0.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:1cc7ea17a6824959616c525620e387f6dd30fec8cb44f649e31712db02123dad", size = 11631, upload-time = "2025-09-27T18:36:18.185Z" }, + { url = "https://files.pythonhosted.org/packages/e1/2e/5898933336b61975ce9dc04decbc0a7f2fee78c30353c5efba7f2d6ff27a/markupsafe-3.0.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:4bd4cd07944443f5a265608cc6aab442e4f74dff8088b0dfc8238647b8f6ae9a", size = 12058, upload-time = "2025-09-27T18:36:19.444Z" }, + { url = "https://files.pythonhosted.org/packages/1d/09/adf2df3699d87d1d8184038df46a9c80d78c0148492323f4693df54e17bb/markupsafe-3.0.3-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b5420a1d9450023228968e7e6a9ce57f65d148ab56d2313fcd589eee96a7a50", size = 24287, upload-time = "2025-09-27T18:36:20.768Z" }, + { url = "https://files.pythonhosted.org/packages/30/ac/0273f6fcb5f42e314c6d8cd99effae6a5354604d461b8d392b5ec9530a54/markupsafe-3.0.3-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0bf2a864d67e76e5c9a34dc26ec616a66b9888e25e7b9460e1c76d3293bd9dbf", size = 22940, upload-time = "2025-09-27T18:36:22.249Z" }, + { url = "https://files.pythonhosted.org/packages/19/ae/31c1be199ef767124c042c6c3e904da327a2f7f0cd63a0337e1eca2967a8/markupsafe-3.0.3-cp311-cp311-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:bc51efed119bc9cfdf792cdeaa4d67e8f6fcccab66ed4bfdd6bde3e59bfcbb2f", size = 21887, upload-time = "2025-09-27T18:36:23.535Z" }, + { url = "https://files.pythonhosted.org/packages/b2/76/7edcab99d5349a4532a459e1fe64f0b0467a3365056ae550d3bcf3f79e1e/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:068f375c472b3e7acbe2d5318dea141359e6900156b5b2ba06a30b169086b91a", size = 23692, upload-time = "2025-09-27T18:36:24.823Z" }, + { url = "https://files.pythonhosted.org/packages/a4/28/6e74cdd26d7514849143d69f0bf2399f929c37dc2b31e6829fd2045b2765/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_riscv64.whl", hash = "sha256:7be7b61bb172e1ed687f1754f8e7484f1c8019780f6f6b0786e76bb01c2ae115", size = 21471, upload-time = "2025-09-27T18:36:25.95Z" }, + { url = "https://files.pythonhosted.org/packages/62/7e/a145f36a5c2945673e590850a6f8014318d5577ed7e5920a4b3448e0865d/markupsafe-3.0.3-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:f9e130248f4462aaa8e2552d547f36ddadbeaa573879158d721bbd33dfe4743a", size = 22923, upload-time = "2025-09-27T18:36:27.109Z" }, + { url = "https://files.pythonhosted.org/packages/0f/62/d9c46a7f5c9adbeeeda52f5b8d802e1094e9717705a645efc71b0913a0a8/markupsafe-3.0.3-cp311-cp311-win32.whl", hash = "sha256:0db14f5dafddbb6d9208827849fad01f1a2609380add406671a26386cdf15a19", size = 14572, upload-time = "2025-09-27T18:36:28.045Z" }, + { url = "https://files.pythonhosted.org/packages/83/8a/4414c03d3f891739326e1783338e48fb49781cc915b2e0ee052aa490d586/markupsafe-3.0.3-cp311-cp311-win_amd64.whl", hash = "sha256:de8a88e63464af587c950061a5e6a67d3632e36df62b986892331d4620a35c01", size = 15077, upload-time = "2025-09-27T18:36:29.025Z" }, + { url = "https://files.pythonhosted.org/packages/35/73/893072b42e6862f319b5207adc9ae06070f095b358655f077f69a35601f0/markupsafe-3.0.3-cp311-cp311-win_arm64.whl", hash = "sha256:3b562dd9e9ea93f13d53989d23a7e775fdfd1066c33494ff43f5418bc8c58a5c", size = 13876, upload-time = "2025-09-27T18:36:29.954Z" }, + { url = "https://files.pythonhosted.org/packages/5a/72/147da192e38635ada20e0a2e1a51cf8823d2119ce8883f7053879c2199b5/markupsafe-3.0.3-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:d53197da72cc091b024dd97249dfc7794d6a56530370992a5e1a08983ad9230e", size = 11615, upload-time = "2025-09-27T18:36:30.854Z" }, + { url = "https://files.pythonhosted.org/packages/9a/81/7e4e08678a1f98521201c3079f77db69fb552acd56067661f8c2f534a718/markupsafe-3.0.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:1872df69a4de6aead3491198eaf13810b565bdbeec3ae2dc8780f14458ec73ce", size = 12020, upload-time = "2025-09-27T18:36:31.971Z" }, + { url = "https://files.pythonhosted.org/packages/1e/2c/799f4742efc39633a1b54a92eec4082e4f815314869865d876824c257c1e/markupsafe-3.0.3-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3a7e8ae81ae39e62a41ec302f972ba6ae23a5c5396c8e60113e9066ef893da0d", size = 24332, upload-time = "2025-09-27T18:36:32.813Z" }, + { url = "https://files.pythonhosted.org/packages/3c/2e/8d0c2ab90a8c1d9a24f0399058ab8519a3279d1bd4289511d74e909f060e/markupsafe-3.0.3-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d6dd0be5b5b189d31db7cda48b91d7e0a9795f31430b7f271219ab30f1d3ac9d", size = 22947, upload-time = "2025-09-27T18:36:33.86Z" }, + { url = "https://files.pythonhosted.org/packages/2c/54/887f3092a85238093a0b2154bd629c89444f395618842e8b0c41783898ea/markupsafe-3.0.3-cp312-cp312-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:94c6f0bb423f739146aec64595853541634bde58b2135f27f61c1ffd1cd4d16a", size = 21962, upload-time = "2025-09-27T18:36:35.099Z" }, + { url = "https://files.pythonhosted.org/packages/c9/2f/336b8c7b6f4a4d95e91119dc8521402461b74a485558d8f238a68312f11c/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:be8813b57049a7dc738189df53d69395eba14fb99345e0a5994914a3864c8a4b", size = 23760, upload-time = "2025-09-27T18:36:36.001Z" }, + { url = "https://files.pythonhosted.org/packages/32/43/67935f2b7e4982ffb50a4d169b724d74b62a3964bc1a9a527f5ac4f1ee2b/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_riscv64.whl", hash = "sha256:83891d0e9fb81a825d9a6d61e3f07550ca70a076484292a70fde82c4b807286f", size = 21529, upload-time = "2025-09-27T18:36:36.906Z" }, + { url = "https://files.pythonhosted.org/packages/89/e0/4486f11e51bbba8b0c041098859e869e304d1c261e59244baa3d295d47b7/markupsafe-3.0.3-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:77f0643abe7495da77fb436f50f8dab76dbc6e5fd25d39589a0f1fe6548bfa2b", size = 23015, upload-time = "2025-09-27T18:36:37.868Z" }, + { url = "https://files.pythonhosted.org/packages/2f/e1/78ee7a023dac597a5825441ebd17170785a9dab23de95d2c7508ade94e0e/markupsafe-3.0.3-cp312-cp312-win32.whl", hash = "sha256:d88b440e37a16e651bda4c7c2b930eb586fd15ca7406cb39e211fcff3bf3017d", size = 14540, upload-time = "2025-09-27T18:36:38.761Z" }, + { url = "https://files.pythonhosted.org/packages/aa/5b/bec5aa9bbbb2c946ca2733ef9c4ca91c91b6a24580193e891b5f7dbe8e1e/markupsafe-3.0.3-cp312-cp312-win_amd64.whl", hash = "sha256:26a5784ded40c9e318cfc2bdb30fe164bdb8665ded9cd64d500a34fb42067b1c", size = 15105, upload-time = "2025-09-27T18:36:39.701Z" }, + { url = "https://files.pythonhosted.org/packages/e5/f1/216fc1bbfd74011693a4fd837e7026152e89c4bcf3e77b6692fba9923123/markupsafe-3.0.3-cp312-cp312-win_arm64.whl", hash = "sha256:35add3b638a5d900e807944a078b51922212fb3dedb01633a8defc4b01a3c85f", size = 13906, upload-time = "2025-09-27T18:36:40.689Z" }, + { url = "https://files.pythonhosted.org/packages/38/2f/907b9c7bbba283e68f20259574b13d005c121a0fa4c175f9bed27c4597ff/markupsafe-3.0.3-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:e1cf1972137e83c5d4c136c43ced9ac51d0e124706ee1c8aa8532c1287fa8795", size = 11622, upload-time = "2025-09-27T18:36:41.777Z" }, + { url = "https://files.pythonhosted.org/packages/9c/d9/5f7756922cdd676869eca1c4e3c0cd0df60ed30199ffd775e319089cb3ed/markupsafe-3.0.3-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:116bb52f642a37c115f517494ea5feb03889e04df47eeff5b130b1808ce7c219", size = 12029, upload-time = "2025-09-27T18:36:43.257Z" }, + { url = "https://files.pythonhosted.org/packages/00/07/575a68c754943058c78f30db02ee03a64b3c638586fba6a6dd56830b30a3/markupsafe-3.0.3-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:133a43e73a802c5562be9bbcd03d090aa5a1fe899db609c29e8c8d815c5f6de6", size = 24374, upload-time = "2025-09-27T18:36:44.508Z" }, + { url = "https://files.pythonhosted.org/packages/a9/21/9b05698b46f218fc0e118e1f8168395c65c8a2c750ae2bab54fc4bd4e0e8/markupsafe-3.0.3-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ccfcd093f13f0f0b7fdd0f198b90053bf7b2f02a3927a30e63f3ccc9df56b676", size = 22980, upload-time = "2025-09-27T18:36:45.385Z" }, + { url = "https://files.pythonhosted.org/packages/7f/71/544260864f893f18b6827315b988c146b559391e6e7e8f7252839b1b846a/markupsafe-3.0.3-cp313-cp313-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:509fa21c6deb7a7a273d629cf5ec029bc209d1a51178615ddf718f5918992ab9", size = 21990, upload-time = "2025-09-27T18:36:46.916Z" }, + { url = "https://files.pythonhosted.org/packages/c2/28/b50fc2f74d1ad761af2f5dcce7492648b983d00a65b8c0e0cb457c82ebbe/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:a4afe79fb3de0b7097d81da19090f4df4f8d3a2b3adaa8764138aac2e44f3af1", size = 23784, upload-time = "2025-09-27T18:36:47.884Z" }, + { url = "https://files.pythonhosted.org/packages/ed/76/104b2aa106a208da8b17a2fb72e033a5a9d7073c68f7e508b94916ed47a9/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_riscv64.whl", hash = "sha256:795e7751525cae078558e679d646ae45574b47ed6e7771863fcc079a6171a0fc", size = 21588, upload-time = "2025-09-27T18:36:48.82Z" }, + { url = "https://files.pythonhosted.org/packages/b5/99/16a5eb2d140087ebd97180d95249b00a03aa87e29cc224056274f2e45fd6/markupsafe-3.0.3-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:8485f406a96febb5140bfeca44a73e3ce5116b2501ac54fe953e488fb1d03b12", size = 23041, upload-time = "2025-09-27T18:36:49.797Z" }, + { url = "https://files.pythonhosted.org/packages/19/bc/e7140ed90c5d61d77cea142eed9f9c303f4c4806f60a1044c13e3f1471d0/markupsafe-3.0.3-cp313-cp313-win32.whl", hash = "sha256:bdd37121970bfd8be76c5fb069c7751683bdf373db1ed6c010162b2a130248ed", size = 14543, upload-time = "2025-09-27T18:36:51.584Z" }, + { url = "https://files.pythonhosted.org/packages/05/73/c4abe620b841b6b791f2edc248f556900667a5a1cf023a6646967ae98335/markupsafe-3.0.3-cp313-cp313-win_amd64.whl", hash = "sha256:9a1abfdc021a164803f4d485104931fb8f8c1efd55bc6b748d2f5774e78b62c5", size = 15113, upload-time = "2025-09-27T18:36:52.537Z" }, + { url = "https://files.pythonhosted.org/packages/f0/3a/fa34a0f7cfef23cf9500d68cb7c32dd64ffd58a12b09225fb03dd37d5b80/markupsafe-3.0.3-cp313-cp313-win_arm64.whl", hash = "sha256:7e68f88e5b8799aa49c85cd116c932a1ac15caaa3f5db09087854d218359e485", size = 13911, upload-time = "2025-09-27T18:36:53.513Z" }, + { url = "https://files.pythonhosted.org/packages/e4/d7/e05cd7efe43a88a17a37b3ae96e79a19e846f3f456fe79c57ca61356ef01/markupsafe-3.0.3-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:218551f6df4868a8d527e3062d0fb968682fe92054e89978594c28e642c43a73", size = 11658, upload-time = "2025-09-27T18:36:54.819Z" }, + { url = "https://files.pythonhosted.org/packages/99/9e/e412117548182ce2148bdeacdda3bb494260c0b0184360fe0d56389b523b/markupsafe-3.0.3-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:3524b778fe5cfb3452a09d31e7b5adefeea8c5be1d43c4f810ba09f2ceb29d37", size = 12066, upload-time = "2025-09-27T18:36:55.714Z" }, + { url = "https://files.pythonhosted.org/packages/bc/e6/fa0ffcda717ef64a5108eaa7b4f5ed28d56122c9a6d70ab8b72f9f715c80/markupsafe-3.0.3-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4e885a3d1efa2eadc93c894a21770e4bc67899e3543680313b09f139e149ab19", size = 25639, upload-time = "2025-09-27T18:36:56.908Z" }, + { url = "https://files.pythonhosted.org/packages/96/ec/2102e881fe9d25fc16cb4b25d5f5cde50970967ffa5dddafdb771237062d/markupsafe-3.0.3-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8709b08f4a89aa7586de0aadc8da56180242ee0ada3999749b183aa23df95025", size = 23569, upload-time = "2025-09-27T18:36:57.913Z" }, + { url = "https://files.pythonhosted.org/packages/4b/30/6f2fce1f1f205fc9323255b216ca8a235b15860c34b6798f810f05828e32/markupsafe-3.0.3-cp313-cp313t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:b8512a91625c9b3da6f127803b166b629725e68af71f8184ae7e7d54686a56d6", size = 23284, upload-time = "2025-09-27T18:36:58.833Z" }, + { url = "https://files.pythonhosted.org/packages/58/47/4a0ccea4ab9f5dcb6f79c0236d954acb382202721e704223a8aafa38b5c8/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:9b79b7a16f7fedff2495d684f2b59b0457c3b493778c9eed31111be64d58279f", size = 24801, upload-time = "2025-09-27T18:36:59.739Z" }, + { url = "https://files.pythonhosted.org/packages/6a/70/3780e9b72180b6fecb83a4814d84c3bf4b4ae4bf0b19c27196104149734c/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_riscv64.whl", hash = "sha256:12c63dfb4a98206f045aa9563db46507995f7ef6d83b2f68eda65c307c6829eb", size = 22769, upload-time = "2025-09-27T18:37:00.719Z" }, + { url = "https://files.pythonhosted.org/packages/98/c5/c03c7f4125180fc215220c035beac6b9cb684bc7a067c84fc69414d315f5/markupsafe-3.0.3-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:8f71bc33915be5186016f675cd83a1e08523649b0e33efdb898db577ef5bb009", size = 23642, upload-time = "2025-09-27T18:37:01.673Z" }, + { url = "https://files.pythonhosted.org/packages/80/d6/2d1b89f6ca4bff1036499b1e29a1d02d282259f3681540e16563f27ebc23/markupsafe-3.0.3-cp313-cp313t-win32.whl", hash = "sha256:69c0b73548bc525c8cb9a251cddf1931d1db4d2258e9599c28c07ef3580ef354", size = 14612, upload-time = "2025-09-27T18:37:02.639Z" }, + { url = "https://files.pythonhosted.org/packages/2b/98/e48a4bfba0a0ffcf9925fe2d69240bfaa19c6f7507b8cd09c70684a53c1e/markupsafe-3.0.3-cp313-cp313t-win_amd64.whl", hash = "sha256:1b4b79e8ebf6b55351f0d91fe80f893b4743f104bff22e90697db1590e47a218", size = 15200, upload-time = "2025-09-27T18:37:03.582Z" }, + { url = "https://files.pythonhosted.org/packages/0e/72/e3cc540f351f316e9ed0f092757459afbc595824ca724cbc5a5d4263713f/markupsafe-3.0.3-cp313-cp313t-win_arm64.whl", hash = "sha256:ad2cf8aa28b8c020ab2fc8287b0f823d0a7d8630784c31e9ee5edea20f406287", size = 13973, upload-time = "2025-09-27T18:37:04.929Z" }, + { url = "https://files.pythonhosted.org/packages/33/8a/8e42d4838cd89b7dde187011e97fe6c3af66d8c044997d2183fbd6d31352/markupsafe-3.0.3-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:eaa9599de571d72e2daf60164784109f19978b327a3910d3e9de8c97b5b70cfe", size = 11619, upload-time = "2025-09-27T18:37:06.342Z" }, + { url = "https://files.pythonhosted.org/packages/b5/64/7660f8a4a8e53c924d0fa05dc3a55c9cee10bbd82b11c5afb27d44b096ce/markupsafe-3.0.3-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:c47a551199eb8eb2121d4f0f15ae0f923d31350ab9280078d1e5f12b249e0026", size = 12029, upload-time = "2025-09-27T18:37:07.213Z" }, + { url = "https://files.pythonhosted.org/packages/da/ef/e648bfd021127bef5fa12e1720ffed0c6cbb8310c8d9bea7266337ff06de/markupsafe-3.0.3-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f34c41761022dd093b4b6896d4810782ffbabe30f2d443ff5f083e0cbbb8c737", size = 24408, upload-time = "2025-09-27T18:37:09.572Z" }, + { url = "https://files.pythonhosted.org/packages/41/3c/a36c2450754618e62008bf7435ccb0f88053e07592e6028a34776213d877/markupsafe-3.0.3-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:457a69a9577064c05a97c41f4e65148652db078a3a509039e64d3467b9e7ef97", size = 23005, upload-time = "2025-09-27T18:37:10.58Z" }, + { url = "https://files.pythonhosted.org/packages/bc/20/b7fdf89a8456b099837cd1dc21974632a02a999ec9bf7ca3e490aacd98e7/markupsafe-3.0.3-cp314-cp314-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:e8afc3f2ccfa24215f8cb28dcf43f0113ac3c37c2f0f0806d8c70e4228c5cf4d", size = 22048, upload-time = "2025-09-27T18:37:11.547Z" }, + { url = "https://files.pythonhosted.org/packages/9a/a7/591f592afdc734f47db08a75793a55d7fbcc6902a723ae4cfbab61010cc5/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:ec15a59cf5af7be74194f7ab02d0f59a62bdcf1a537677ce67a2537c9b87fcda", size = 23821, upload-time = "2025-09-27T18:37:12.48Z" }, + { url = "https://files.pythonhosted.org/packages/7d/33/45b24e4f44195b26521bc6f1a82197118f74df348556594bd2262bda1038/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_riscv64.whl", hash = "sha256:0eb9ff8191e8498cca014656ae6b8d61f39da5f95b488805da4bb029cccbfbaf", size = 21606, upload-time = "2025-09-27T18:37:13.485Z" }, + { url = "https://files.pythonhosted.org/packages/ff/0e/53dfaca23a69fbfbbf17a4b64072090e70717344c52eaaaa9c5ddff1e5f0/markupsafe-3.0.3-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:2713baf880df847f2bece4230d4d094280f4e67b1e813eec43b4c0e144a34ffe", size = 23043, upload-time = "2025-09-27T18:37:14.408Z" }, + { url = "https://files.pythonhosted.org/packages/46/11/f333a06fc16236d5238bfe74daccbca41459dcd8d1fa952e8fbd5dccfb70/markupsafe-3.0.3-cp314-cp314-win32.whl", hash = "sha256:729586769a26dbceff69f7a7dbbf59ab6572b99d94576a5592625d5b411576b9", size = 14747, upload-time = "2025-09-27T18:37:15.36Z" }, + { url = "https://files.pythonhosted.org/packages/28/52/182836104b33b444e400b14f797212f720cbc9ed6ba34c800639d154e821/markupsafe-3.0.3-cp314-cp314-win_amd64.whl", hash = "sha256:bdc919ead48f234740ad807933cdf545180bfbe9342c2bb451556db2ed958581", size = 15341, upload-time = "2025-09-27T18:37:16.496Z" }, + { url = "https://files.pythonhosted.org/packages/6f/18/acf23e91bd94fd7b3031558b1f013adfa21a8e407a3fdb32745538730382/markupsafe-3.0.3-cp314-cp314-win_arm64.whl", hash = "sha256:5a7d5dc5140555cf21a6fefbdbf8723f06fcd2f63ef108f2854de715e4422cb4", size = 14073, upload-time = "2025-09-27T18:37:17.476Z" }, + { url = "https://files.pythonhosted.org/packages/3c/f0/57689aa4076e1b43b15fdfa646b04653969d50cf30c32a102762be2485da/markupsafe-3.0.3-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:1353ef0c1b138e1907ae78e2f6c63ff67501122006b0f9abad68fda5f4ffc6ab", size = 11661, upload-time = "2025-09-27T18:37:18.453Z" }, + { url = "https://files.pythonhosted.org/packages/89/c3/2e67a7ca217c6912985ec766c6393b636fb0c2344443ff9d91404dc4c79f/markupsafe-3.0.3-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:1085e7fbddd3be5f89cc898938f42c0b3c711fdcb37d75221de2666af647c175", size = 12069, upload-time = "2025-09-27T18:37:19.332Z" }, + { url = "https://files.pythonhosted.org/packages/f0/00/be561dce4e6ca66b15276e184ce4b8aec61fe83662cce2f7d72bd3249d28/markupsafe-3.0.3-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1b52b4fb9df4eb9ae465f8d0c228a00624de2334f216f178a995ccdcf82c4634", size = 25670, upload-time = "2025-09-27T18:37:20.245Z" }, + { url = "https://files.pythonhosted.org/packages/50/09/c419f6f5a92e5fadde27efd190eca90f05e1261b10dbd8cbcb39cd8ea1dc/markupsafe-3.0.3-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fed51ac40f757d41b7c48425901843666a6677e3e8eb0abcff09e4ba6e664f50", size = 23598, upload-time = "2025-09-27T18:37:21.177Z" }, + { url = "https://files.pythonhosted.org/packages/22/44/a0681611106e0b2921b3033fc19bc53323e0b50bc70cffdd19f7d679bb66/markupsafe-3.0.3-cp314-cp314t-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:f190daf01f13c72eac4efd5c430a8de82489d9cff23c364c3ea822545032993e", size = 23261, upload-time = "2025-09-27T18:37:22.167Z" }, + { url = "https://files.pythonhosted.org/packages/5f/57/1b0b3f100259dc9fffe780cfb60d4be71375510e435efec3d116b6436d43/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:e56b7d45a839a697b5eb268c82a71bd8c7f6c94d6fd50c3d577fa39a9f1409f5", size = 24835, upload-time = "2025-09-27T18:37:23.296Z" }, + { url = "https://files.pythonhosted.org/packages/26/6a/4bf6d0c97c4920f1597cc14dd720705eca0bf7c787aebc6bb4d1bead5388/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_riscv64.whl", hash = "sha256:f3e98bb3798ead92273dc0e5fd0f31ade220f59a266ffd8a4f6065e0a3ce0523", size = 22733, upload-time = "2025-09-27T18:37:24.237Z" }, + { url = "https://files.pythonhosted.org/packages/14/c7/ca723101509b518797fedc2fdf79ba57f886b4aca8a7d31857ba3ee8281f/markupsafe-3.0.3-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:5678211cb9333a6468fb8d8be0305520aa073f50d17f089b5b4b477ea6e67fdc", size = 23672, upload-time = "2025-09-27T18:37:25.271Z" }, + { url = "https://files.pythonhosted.org/packages/fb/df/5bd7a48c256faecd1d36edc13133e51397e41b73bb77e1a69deab746ebac/markupsafe-3.0.3-cp314-cp314t-win32.whl", hash = "sha256:915c04ba3851909ce68ccc2b8e2cd691618c4dc4c4232fb7982bca3f41fd8c3d", size = 14819, upload-time = "2025-09-27T18:37:26.285Z" }, + { url = "https://files.pythonhosted.org/packages/1a/8a/0402ba61a2f16038b48b39bccca271134be00c5c9f0f623208399333c448/markupsafe-3.0.3-cp314-cp314t-win_amd64.whl", hash = "sha256:4faffd047e07c38848ce017e8725090413cd80cbc23d86e55c587bf979e579c9", size = 15426, upload-time = "2025-09-27T18:37:27.316Z" }, + { url = "https://files.pythonhosted.org/packages/70/bc/6f1c2f612465f5fa89b95bead1f44dcb607670fd42891d8fdcd5d039f4f4/markupsafe-3.0.3-cp314-cp314t-win_arm64.whl", hash = "sha256:32001d6a8fc98c8cb5c947787c5d08b0a50663d139f1305bac5885d98d9b40fa", size = 14146, upload-time = "2025-09-27T18:37:28.327Z" }, + { url = "https://files.pythonhosted.org/packages/56/23/0d8c13a44bde9154821586520840643467aee574d8ce79a17da539ee7fed/markupsafe-3.0.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:15d939a21d546304880945ca1ecb8a039db6b4dc49b2c5a400387cdae6a62e26", size = 11623, upload-time = "2025-09-27T18:37:29.296Z" }, + { url = "https://files.pythonhosted.org/packages/fd/23/07a2cb9a8045d5f3f0890a8c3bc0859d7a47bfd9a560b563899bec7b72ed/markupsafe-3.0.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:f71a396b3bf33ecaa1626c255855702aca4d3d9fea5e051b41ac59a9c1c41edc", size = 12049, upload-time = "2025-09-27T18:37:30.234Z" }, + { url = "https://files.pythonhosted.org/packages/bc/e4/6be85eb81503f8e11b61c0b6369b6e077dcf0a74adbd9ebf6b349937b4e9/markupsafe-3.0.3-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:0f4b68347f8c5eab4a13419215bdfd7f8c9b19f2b25520968adfad23eb0ce60c", size = 21923, upload-time = "2025-09-27T18:37:31.177Z" }, + { url = "https://files.pythonhosted.org/packages/6f/bc/4dc914ead3fe6ddaef035341fee0fc956949bbd27335b611829292b89ee2/markupsafe-3.0.3-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:e8fc20152abba6b83724d7ff268c249fa196d8259ff481f3b1476383f8f24e42", size = 20543, upload-time = "2025-09-27T18:37:32.168Z" }, + { url = "https://files.pythonhosted.org/packages/89/6e/5fe81fbcfba4aef4093d5f856e5c774ec2057946052d18d168219b7bd9f9/markupsafe-3.0.3-cp39-cp39-manylinux_2_31_riscv64.manylinux_2_39_riscv64.whl", hash = "sha256:949b8d66bc381ee8b007cd945914c721d9aba8e27f71959d750a46f7c282b20b", size = 20585, upload-time = "2025-09-27T18:37:33.166Z" }, + { url = "https://files.pythonhosted.org/packages/f6/f6/e0e5a3d3ae9c4020f696cd055f940ef86b64fe88de26f3a0308b9d3d048c/markupsafe-3.0.3-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:3537e01efc9d4dccdf77221fb1cb3b8e1a38d5428920e0657ce299b20324d758", size = 21387, upload-time = "2025-09-27T18:37:34.185Z" }, + { url = "https://files.pythonhosted.org/packages/c8/25/651753ef4dea08ea790f4fbb65146a9a44a014986996ca40102e237aa49a/markupsafe-3.0.3-cp39-cp39-musllinux_1_2_riscv64.whl", hash = "sha256:591ae9f2a647529ca990bc681daebdd52c8791ff06c2bfa05b65163e28102ef2", size = 20133, upload-time = "2025-09-27T18:37:35.138Z" }, + { url = "https://files.pythonhosted.org/packages/dc/0a/c3cf2b4fef5f0426e8a6d7fce3cb966a17817c568ce59d76b92a233fdbec/markupsafe-3.0.3-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:a320721ab5a1aba0a233739394eb907f8c8da5c98c9181d1161e77a0c8e36f2d", size = 20588, upload-time = "2025-09-27T18:37:36.096Z" }, + { url = "https://files.pythonhosted.org/packages/cd/1b/a7782984844bd519ad4ffdbebbba2671ec5d0ebbeac34736c15fb86399e8/markupsafe-3.0.3-cp39-cp39-win32.whl", hash = "sha256:df2449253ef108a379b8b5d6b43f4b1a8e81a061d6537becd5582fba5f9196d7", size = 14566, upload-time = "2025-09-27T18:37:37.09Z" }, + { url = "https://files.pythonhosted.org/packages/18/1f/8d9c20e1c9440e215a44be5ab64359e207fcb4f675543f1cf9a2a7f648d0/markupsafe-3.0.3-cp39-cp39-win_amd64.whl", hash = "sha256:7c3fb7d25180895632e5d3148dbdc29ea38ccb7fd210aa27acbd1201a1902c6e", size = 15053, upload-time = "2025-09-27T18:37:38.054Z" }, + { url = "https://files.pythonhosted.org/packages/4e/d3/fe08482b5cd995033556d45041a4f4e76e7f0521112a9c9991d40d39825f/markupsafe-3.0.3-cp39-cp39-win_arm64.whl", hash = "sha256:38664109c14ffc9e7437e86b4dceb442b0096dfe3541d7864d9cbe1da4cf36c8", size = 13928, upload-time = "2025-09-27T18:37:39.037Z" }, +] + +[[package]] +name = "multidict" +version = "6.1.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "typing-extensions", version = "4.13.2", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/d6/be/504b89a5e9ca731cd47487e91c469064f8ae5af93b7259758dcfc2b9c848/multidict-6.1.0.tar.gz", hash = "sha256:22ae2ebf9b0c69d206c003e2f6a914ea33f0a932d4aa16f236afc049d9958f4a", size = 64002, upload-time = "2024-09-09T23:49:38.163Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/29/68/259dee7fd14cf56a17c554125e534f6274c2860159692a414d0b402b9a6d/multidict-6.1.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:3380252550e372e8511d49481bd836264c009adb826b23fefcc5dd3c69692f60", size = 48628, upload-time = "2024-09-09T23:47:18.278Z" }, + { url = "https://files.pythonhosted.org/packages/50/79/53ba256069fe5386a4a9e80d4e12857ced9de295baf3e20c68cdda746e04/multidict-6.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:99f826cbf970077383d7de805c0681799491cb939c25450b9b5b3ced03ca99f1", size = 29327, upload-time = "2024-09-09T23:47:20.224Z" }, + { url = "https://files.pythonhosted.org/packages/ff/10/71f1379b05b196dae749b5ac062e87273e3f11634f447ebac12a571d90ae/multidict-6.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a114d03b938376557927ab23f1e950827c3b893ccb94b62fd95d430fd0e5cf53", size = 29689, upload-time = "2024-09-09T23:47:21.667Z" }, + { url = "https://files.pythonhosted.org/packages/71/45/70bac4f87438ded36ad4793793c0095de6572d433d98575a5752629ef549/multidict-6.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b1c416351ee6271b2f49b56ad7f308072f6f44b37118d69c2cad94f3fa8a40d5", size = 126639, upload-time = "2024-09-09T23:47:23.333Z" }, + { url = "https://files.pythonhosted.org/packages/80/cf/17f35b3b9509b4959303c05379c4bfb0d7dd05c3306039fc79cf035bbac0/multidict-6.1.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6b5d83030255983181005e6cfbac1617ce9746b219bc2aad52201ad121226581", size = 134315, upload-time = "2024-09-09T23:47:24.99Z" }, + { url = "https://files.pythonhosted.org/packages/ef/1f/652d70ab5effb33c031510a3503d4d6efc5ec93153562f1ee0acdc895a57/multidict-6.1.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3e97b5e938051226dc025ec80980c285b053ffb1e25a3db2a3aa3bc046bf7f56", size = 129471, upload-time = "2024-09-09T23:47:26.305Z" }, + { url = "https://files.pythonhosted.org/packages/a6/64/2dd6c4c681688c0165dea3975a6a4eab4944ea30f35000f8b8af1df3148c/multidict-6.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d618649d4e70ac6efcbba75be98b26ef5078faad23592f9b51ca492953012429", size = 124585, upload-time = "2024-09-09T23:47:27.958Z" }, + { url = "https://files.pythonhosted.org/packages/87/56/e6ee5459894c7e554b57ba88f7257dc3c3d2d379cb15baaa1e265b8c6165/multidict-6.1.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:10524ebd769727ac77ef2278390fb0068d83f3acb7773792a5080f2b0abf7748", size = 116957, upload-time = "2024-09-09T23:47:29.376Z" }, + { url = "https://files.pythonhosted.org/packages/36/9e/616ce5e8d375c24b84f14fc263c7ef1d8d5e8ef529dbc0f1df8ce71bb5b8/multidict-6.1.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:ff3827aef427c89a25cc96ded1759271a93603aba9fb977a6d264648ebf989db", size = 128609, upload-time = "2024-09-09T23:47:31.038Z" }, + { url = "https://files.pythonhosted.org/packages/8c/4f/4783e48a38495d000f2124020dc96bacc806a4340345211b1ab6175a6cb4/multidict-6.1.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:06809f4f0f7ab7ea2cabf9caca7d79c22c0758b58a71f9d32943ae13c7ace056", size = 123016, upload-time = "2024-09-09T23:47:32.47Z" }, + { url = "https://files.pythonhosted.org/packages/3e/b3/4950551ab8fc39862ba5e9907dc821f896aa829b4524b4deefd3e12945ab/multidict-6.1.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:f179dee3b863ab1c59580ff60f9d99f632f34ccb38bf67a33ec6b3ecadd0fd76", size = 133542, upload-time = "2024-09-09T23:47:34.103Z" }, + { url = "https://files.pythonhosted.org/packages/96/4d/f0ce6ac9914168a2a71df117935bb1f1781916acdecbb43285e225b484b8/multidict-6.1.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:aaed8b0562be4a0876ee3b6946f6869b7bcdb571a5d1496683505944e268b160", size = 130163, upload-time = "2024-09-09T23:47:35.716Z" }, + { url = "https://files.pythonhosted.org/packages/be/72/17c9f67e7542a49dd252c5ae50248607dfb780bcc03035907dafefb067e3/multidict-6.1.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:3c8b88a2ccf5493b6c8da9076fb151ba106960a2df90c2633f342f120751a9e7", size = 126832, upload-time = "2024-09-09T23:47:37.116Z" }, + { url = "https://files.pythonhosted.org/packages/71/9f/72d719e248cbd755c8736c6d14780533a1606ffb3fbb0fbd77da9f0372da/multidict-6.1.0-cp310-cp310-win32.whl", hash = "sha256:4a9cb68166a34117d6646c0023c7b759bf197bee5ad4272f420a0141d7eb03a0", size = 26402, upload-time = "2024-09-09T23:47:38.863Z" }, + { url = "https://files.pythonhosted.org/packages/04/5a/d88cd5d00a184e1ddffc82aa2e6e915164a6d2641ed3606e766b5d2f275a/multidict-6.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:20b9b5fbe0b88d0bdef2012ef7dee867f874b72528cf1d08f1d59b0e3850129d", size = 28800, upload-time = "2024-09-09T23:47:40.056Z" }, + { url = "https://files.pythonhosted.org/packages/93/13/df3505a46d0cd08428e4c8169a196131d1b0c4b515c3649829258843dde6/multidict-6.1.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:3efe2c2cb5763f2f1b275ad2bf7a287d3f7ebbef35648a9726e3b69284a4f3d6", size = 48570, upload-time = "2024-09-09T23:47:41.36Z" }, + { url = "https://files.pythonhosted.org/packages/f0/e1/a215908bfae1343cdb72f805366592bdd60487b4232d039c437fe8f5013d/multidict-6.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c7053d3b0353a8b9de430a4f4b4268ac9a4fb3481af37dfe49825bf45ca24156", size = 29316, upload-time = "2024-09-09T23:47:42.612Z" }, + { url = "https://files.pythonhosted.org/packages/70/0f/6dc70ddf5d442702ed74f298d69977f904960b82368532c88e854b79f72b/multidict-6.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:27e5fc84ccef8dfaabb09d82b7d179c7cf1a3fbc8a966f8274fcb4ab2eb4cadb", size = 29640, upload-time = "2024-09-09T23:47:44.028Z" }, + { url = "https://files.pythonhosted.org/packages/d8/6d/9c87b73a13d1cdea30b321ef4b3824449866bd7f7127eceed066ccb9b9ff/multidict-6.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0e2b90b43e696f25c62656389d32236e049568b39320e2735d51f08fd362761b", size = 131067, upload-time = "2024-09-09T23:47:45.617Z" }, + { url = "https://files.pythonhosted.org/packages/cc/1e/1b34154fef373371fd6c65125b3d42ff5f56c7ccc6bfff91b9b3c60ae9e0/multidict-6.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d83a047959d38a7ff552ff94be767b7fd79b831ad1cd9920662db05fec24fe72", size = 138507, upload-time = "2024-09-09T23:47:47.429Z" }, + { url = "https://files.pythonhosted.org/packages/fb/e0/0bc6b2bac6e461822b5f575eae85da6aae76d0e2a79b6665d6206b8e2e48/multidict-6.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d1a9dd711d0877a1ece3d2e4fea11a8e75741ca21954c919406b44e7cf971304", size = 133905, upload-time = "2024-09-09T23:47:48.878Z" }, + { url = "https://files.pythonhosted.org/packages/ba/af/73d13b918071ff9b2205fcf773d316e0f8fefb4ec65354bbcf0b10908cc6/multidict-6.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ec2abea24d98246b94913b76a125e855eb5c434f7c46546046372fe60f666351", size = 129004, upload-time = "2024-09-09T23:47:50.124Z" }, + { url = "https://files.pythonhosted.org/packages/74/21/23960627b00ed39643302d81bcda44c9444ebcdc04ee5bedd0757513f259/multidict-6.1.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4867cafcbc6585e4b678876c489b9273b13e9fff9f6d6d66add5e15d11d926cb", size = 121308, upload-time = "2024-09-09T23:47:51.97Z" }, + { url = "https://files.pythonhosted.org/packages/8b/5c/cf282263ffce4a596ed0bb2aa1a1dddfe1996d6a62d08842a8d4b33dca13/multidict-6.1.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:5b48204e8d955c47c55b72779802b219a39acc3ee3d0116d5080c388970b76e3", size = 132608, upload-time = "2024-09-09T23:47:53.201Z" }, + { url = "https://files.pythonhosted.org/packages/d7/3e/97e778c041c72063f42b290888daff008d3ab1427f5b09b714f5a8eff294/multidict-6.1.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:d8fff389528cad1618fb4b26b95550327495462cd745d879a8c7c2115248e399", size = 127029, upload-time = "2024-09-09T23:47:54.435Z" }, + { url = "https://files.pythonhosted.org/packages/47/ac/3efb7bfe2f3aefcf8d103e9a7162572f01936155ab2f7ebcc7c255a23212/multidict-6.1.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:a7a9541cd308eed5e30318430a9c74d2132e9a8cb46b901326272d780bf2d423", size = 137594, upload-time = "2024-09-09T23:47:55.659Z" }, + { url = "https://files.pythonhosted.org/packages/42/9b/6c6e9e8dc4f915fc90a9b7798c44a30773dea2995fdcb619870e705afe2b/multidict-6.1.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:da1758c76f50c39a2efd5e9859ce7d776317eb1dd34317c8152ac9251fc574a3", size = 134556, upload-time = "2024-09-09T23:47:56.98Z" }, + { url = "https://files.pythonhosted.org/packages/1d/10/8e881743b26aaf718379a14ac58572a240e8293a1c9d68e1418fb11c0f90/multidict-6.1.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:c943a53e9186688b45b323602298ab727d8865d8c9ee0b17f8d62d14b56f0753", size = 130993, upload-time = "2024-09-09T23:47:58.163Z" }, + { url = "https://files.pythonhosted.org/packages/45/84/3eb91b4b557442802d058a7579e864b329968c8d0ea57d907e7023c677f2/multidict-6.1.0-cp311-cp311-win32.whl", hash = "sha256:90f8717cb649eea3504091e640a1b8568faad18bd4b9fcd692853a04475a4b80", size = 26405, upload-time = "2024-09-09T23:47:59.391Z" }, + { url = "https://files.pythonhosted.org/packages/9f/0b/ad879847ecbf6d27e90a6eabb7eff6b62c129eefe617ea45eae7c1f0aead/multidict-6.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:82176036e65644a6cc5bd619f65f6f19781e8ec2e5330f51aa9ada7504cc1926", size = 28795, upload-time = "2024-09-09T23:48:00.359Z" }, + { url = "https://files.pythonhosted.org/packages/fd/16/92057c74ba3b96d5e211b553895cd6dc7cc4d1e43d9ab8fafc727681ef71/multidict-6.1.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:b04772ed465fa3cc947db808fa306d79b43e896beb677a56fb2347ca1a49c1fa", size = 48713, upload-time = "2024-09-09T23:48:01.893Z" }, + { url = "https://files.pythonhosted.org/packages/94/3d/37d1b8893ae79716179540b89fc6a0ee56b4a65fcc0d63535c6f5d96f217/multidict-6.1.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:6180c0ae073bddeb5a97a38c03f30c233e0a4d39cd86166251617d1bbd0af436", size = 29516, upload-time = "2024-09-09T23:48:03.463Z" }, + { url = "https://files.pythonhosted.org/packages/a2/12/adb6b3200c363062f805275b4c1e656be2b3681aada66c80129932ff0bae/multidict-6.1.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:071120490b47aa997cca00666923a83f02c7fbb44f71cf7f136df753f7fa8761", size = 29557, upload-time = "2024-09-09T23:48:04.905Z" }, + { url = "https://files.pythonhosted.org/packages/47/e9/604bb05e6e5bce1e6a5cf80a474e0f072e80d8ac105f1b994a53e0b28c42/multidict-6.1.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50b3a2710631848991d0bf7de077502e8994c804bb805aeb2925a981de58ec2e", size = 130170, upload-time = "2024-09-09T23:48:06.862Z" }, + { url = "https://files.pythonhosted.org/packages/7e/13/9efa50801785eccbf7086b3c83b71a4fb501a4d43549c2f2f80b8787d69f/multidict-6.1.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b58c621844d55e71c1b7f7c498ce5aa6985d743a1a59034c57a905b3f153c1ef", size = 134836, upload-time = "2024-09-09T23:48:08.537Z" }, + { url = "https://files.pythonhosted.org/packages/bf/0f/93808b765192780d117814a6dfcc2e75de6dcc610009ad408b8814dca3ba/multidict-6.1.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:55b6d90641869892caa9ca42ff913f7ff1c5ece06474fbd32fb2cf6834726c95", size = 133475, upload-time = "2024-09-09T23:48:09.865Z" }, + { url = "https://files.pythonhosted.org/packages/d3/c8/529101d7176fe7dfe1d99604e48d69c5dfdcadb4f06561f465c8ef12b4df/multidict-6.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4b820514bfc0b98a30e3d85462084779900347e4d49267f747ff54060cc33925", size = 131049, upload-time = "2024-09-09T23:48:11.115Z" }, + { url = "https://files.pythonhosted.org/packages/ca/0c/fc85b439014d5a58063e19c3a158a889deec399d47b5269a0f3b6a2e28bc/multidict-6.1.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:10a9b09aba0c5b48c53761b7c720aaaf7cf236d5fe394cd399c7ba662d5f9966", size = 120370, upload-time = "2024-09-09T23:48:12.78Z" }, + { url = "https://files.pythonhosted.org/packages/db/46/d4416eb20176492d2258fbd47b4abe729ff3b6e9c829ea4236f93c865089/multidict-6.1.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:1e16bf3e5fc9f44632affb159d30a437bfe286ce9e02754759be5536b169b305", size = 125178, upload-time = "2024-09-09T23:48:14.295Z" }, + { url = "https://files.pythonhosted.org/packages/5b/46/73697ad7ec521df7de5531a32780bbfd908ded0643cbe457f981a701457c/multidict-6.1.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:76f364861c3bfc98cbbcbd402d83454ed9e01a5224bb3a28bf70002a230f73e2", size = 119567, upload-time = "2024-09-09T23:48:16.284Z" }, + { url = "https://files.pythonhosted.org/packages/cd/ed/51f060e2cb0e7635329fa6ff930aa5cffa17f4c7f5c6c3ddc3500708e2f2/multidict-6.1.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:820c661588bd01a0aa62a1283f20d2be4281b086f80dad9e955e690c75fb54a2", size = 129822, upload-time = "2024-09-09T23:48:17.835Z" }, + { url = "https://files.pythonhosted.org/packages/df/9e/ee7d1954b1331da3eddea0c4e08d9142da5f14b1321c7301f5014f49d492/multidict-6.1.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:0e5f362e895bc5b9e67fe6e4ded2492d8124bdf817827f33c5b46c2fe3ffaca6", size = 128656, upload-time = "2024-09-09T23:48:19.576Z" }, + { url = "https://files.pythonhosted.org/packages/77/00/8538f11e3356b5d95fa4b024aa566cde7a38aa7a5f08f4912b32a037c5dc/multidict-6.1.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:3ec660d19bbc671e3a6443325f07263be452c453ac9e512f5eb935e7d4ac28b3", size = 125360, upload-time = "2024-09-09T23:48:20.957Z" }, + { url = "https://files.pythonhosted.org/packages/be/05/5d334c1f2462d43fec2363cd00b1c44c93a78c3925d952e9a71caf662e96/multidict-6.1.0-cp312-cp312-win32.whl", hash = "sha256:58130ecf8f7b8112cdb841486404f1282b9c86ccb30d3519faf301b2e5659133", size = 26382, upload-time = "2024-09-09T23:48:22.351Z" }, + { url = "https://files.pythonhosted.org/packages/a3/bf/f332a13486b1ed0496d624bcc7e8357bb8053823e8cd4b9a18edc1d97e73/multidict-6.1.0-cp312-cp312-win_amd64.whl", hash = "sha256:188215fc0aafb8e03341995e7c4797860181562380f81ed0a87ff455b70bf1f1", size = 28529, upload-time = "2024-09-09T23:48:23.478Z" }, + { url = "https://files.pythonhosted.org/packages/22/67/1c7c0f39fe069aa4e5d794f323be24bf4d33d62d2a348acdb7991f8f30db/multidict-6.1.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:d569388c381b24671589335a3be6e1d45546c2988c2ebe30fdcada8457a31008", size = 48771, upload-time = "2024-09-09T23:48:24.594Z" }, + { url = "https://files.pythonhosted.org/packages/3c/25/c186ee7b212bdf0df2519eacfb1981a017bda34392c67542c274651daf23/multidict-6.1.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:052e10d2d37810b99cc170b785945421141bf7bb7d2f8799d431e7db229c385f", size = 29533, upload-time = "2024-09-09T23:48:26.187Z" }, + { url = "https://files.pythonhosted.org/packages/67/5e/04575fd837e0958e324ca035b339cea174554f6f641d3fb2b4f2e7ff44a2/multidict-6.1.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:f90c822a402cb865e396a504f9fc8173ef34212a342d92e362ca498cad308e28", size = 29595, upload-time = "2024-09-09T23:48:27.305Z" }, + { url = "https://files.pythonhosted.org/packages/d3/b2/e56388f86663810c07cfe4a3c3d87227f3811eeb2d08450b9e5d19d78876/multidict-6.1.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b225d95519a5bf73860323e633a664b0d85ad3d5bede6d30d95b35d4dfe8805b", size = 130094, upload-time = "2024-09-09T23:48:28.544Z" }, + { url = "https://files.pythonhosted.org/packages/6c/ee/30ae9b4186a644d284543d55d491fbd4239b015d36b23fea43b4c94f7052/multidict-6.1.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:23bfd518810af7de1116313ebd9092cb9aa629beb12f6ed631ad53356ed6b86c", size = 134876, upload-time = "2024-09-09T23:48:30.098Z" }, + { url = "https://files.pythonhosted.org/packages/84/c7/70461c13ba8ce3c779503c70ec9d0345ae84de04521c1f45a04d5f48943d/multidict-6.1.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5c09fcfdccdd0b57867577b719c69e347a436b86cd83747f179dbf0cc0d4c1f3", size = 133500, upload-time = "2024-09-09T23:48:31.793Z" }, + { url = "https://files.pythonhosted.org/packages/4a/9f/002af221253f10f99959561123fae676148dd730e2daa2cd053846a58507/multidict-6.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bf6bea52ec97e95560af5ae576bdac3aa3aae0b6758c6efa115236d9e07dae44", size = 131099, upload-time = "2024-09-09T23:48:33.193Z" }, + { url = "https://files.pythonhosted.org/packages/82/42/d1c7a7301d52af79d88548a97e297f9d99c961ad76bbe6f67442bb77f097/multidict-6.1.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:57feec87371dbb3520da6192213c7d6fc892d5589a93db548331954de8248fd2", size = 120403, upload-time = "2024-09-09T23:48:34.942Z" }, + { url = "https://files.pythonhosted.org/packages/68/f3/471985c2c7ac707547553e8f37cff5158030d36bdec4414cb825fbaa5327/multidict-6.1.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:0c3f390dc53279cbc8ba976e5f8035eab997829066756d811616b652b00a23a3", size = 125348, upload-time = "2024-09-09T23:48:36.222Z" }, + { url = "https://files.pythonhosted.org/packages/67/2c/e6df05c77e0e433c214ec1d21ddd203d9a4770a1f2866a8ca40a545869a0/multidict-6.1.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:59bfeae4b25ec05b34f1956eaa1cb38032282cd4dfabc5056d0a1ec4d696d3aa", size = 119673, upload-time = "2024-09-09T23:48:37.588Z" }, + { url = "https://files.pythonhosted.org/packages/c5/cd/bc8608fff06239c9fb333f9db7743a1b2eafe98c2666c9a196e867a3a0a4/multidict-6.1.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:b2f59caeaf7632cc633b5cf6fc449372b83bbdf0da4ae04d5be36118e46cc0aa", size = 129927, upload-time = "2024-09-09T23:48:39.128Z" }, + { url = "https://files.pythonhosted.org/packages/44/8e/281b69b7bc84fc963a44dc6e0bbcc7150e517b91df368a27834299a526ac/multidict-6.1.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:37bb93b2178e02b7b618893990941900fd25b6b9ac0fa49931a40aecdf083fe4", size = 128711, upload-time = "2024-09-09T23:48:40.55Z" }, + { url = "https://files.pythonhosted.org/packages/12/a4/63e7cd38ed29dd9f1881d5119f272c898ca92536cdb53ffe0843197f6c85/multidict-6.1.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:4e9f48f58c2c523d5a06faea47866cd35b32655c46b443f163d08c6d0ddb17d6", size = 125519, upload-time = "2024-09-09T23:48:42.446Z" }, + { url = "https://files.pythonhosted.org/packages/38/e0/4f5855037a72cd8a7a2f60a3952d9aa45feedb37ae7831642102604e8a37/multidict-6.1.0-cp313-cp313-win32.whl", hash = "sha256:3a37ffb35399029b45c6cc33640a92bef403c9fd388acce75cdc88f58bd19a81", size = 26426, upload-time = "2024-09-09T23:48:43.936Z" }, + { url = "https://files.pythonhosted.org/packages/7e/a5/17ee3a4db1e310b7405f5d25834460073a8ccd86198ce044dfaf69eac073/multidict-6.1.0-cp313-cp313-win_amd64.whl", hash = "sha256:e9aa71e15d9d9beaad2c6b9319edcdc0a49a43ef5c0a4c8265ca9ee7d6c67774", size = 28531, upload-time = "2024-09-09T23:48:45.122Z" }, + { url = "https://files.pythonhosted.org/packages/3e/6a/af41f3aaf5f00fd86cc7d470a2f5b25299b0c84691163b8757f4a1a205f2/multidict-6.1.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:db7457bac39421addd0c8449933ac32d8042aae84a14911a757ae6ca3eef1392", size = 48597, upload-time = "2024-09-09T23:48:46.391Z" }, + { url = "https://files.pythonhosted.org/packages/d9/d6/3d4082760ed11b05734f8bf32a0615b99e7d9d2b3730ad698a4d7377c00a/multidict-6.1.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d094ddec350a2fb899fec68d8353c78233debde9b7d8b4beeafa70825f1c281a", size = 29338, upload-time = "2024-09-09T23:48:47.891Z" }, + { url = "https://files.pythonhosted.org/packages/9d/7f/5d1ce7f47d44393d429922910afbe88fcd29ee3069babbb47507a4c3a7ea/multidict-6.1.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:5845c1fd4866bb5dd3125d89b90e57ed3138241540897de748cdf19de8a2fca2", size = 29562, upload-time = "2024-09-09T23:48:49.254Z" }, + { url = "https://files.pythonhosted.org/packages/ce/ec/c425257671af9308a9b626e2e21f7f43841616e4551de94eb3c92aca75b2/multidict-6.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9079dfc6a70abe341f521f78405b8949f96db48da98aeb43f9907f342f627cdc", size = 130980, upload-time = "2024-09-09T23:48:50.606Z" }, + { url = "https://files.pythonhosted.org/packages/d8/d7/d4220ad2633a89b314593e9b85b5bc9287a7c563c7f9108a4a68d9da5374/multidict-6.1.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3914f5aaa0f36d5d60e8ece6a308ee1c9784cd75ec8151062614657a114c4478", size = 136694, upload-time = "2024-09-09T23:48:52.042Z" }, + { url = "https://files.pythonhosted.org/packages/a1/2a/13e554db5830c8d40185a2e22aa8325516a5de9634c3fb2caf3886a829b3/multidict-6.1.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c08be4f460903e5a9d0f76818db3250f12e9c344e79314d1d570fc69d7f4eae4", size = 131616, upload-time = "2024-09-09T23:48:54.283Z" }, + { url = "https://files.pythonhosted.org/packages/2e/a9/83692e37d8152f104333132105b67100aabfb2e96a87f6bed67f566035a7/multidict-6.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d093be959277cb7dee84b801eb1af388b6ad3ca6a6b6bf1ed7585895789d027d", size = 129664, upload-time = "2024-09-09T23:48:55.785Z" }, + { url = "https://files.pythonhosted.org/packages/cc/1c/1718cd518fb9da7e8890d9d1611c1af0ea5e60f68ff415d026e38401ed36/multidict-6.1.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3702ea6872c5a2a4eeefa6ffd36b042e9773f05b1f37ae3ef7264b1163c2dcf6", size = 121855, upload-time = "2024-09-09T23:48:57.333Z" }, + { url = "https://files.pythonhosted.org/packages/2b/92/f6ed67514b0e3894198f0eb42dcde22f0851ea35f4561a1e4acf36c7b1be/multidict-6.1.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:2090f6a85cafc5b2db085124d752757c9d251548cedabe9bd31afe6363e0aff2", size = 127928, upload-time = "2024-09-09T23:48:58.778Z" }, + { url = "https://files.pythonhosted.org/packages/f7/30/c66954115a4dc4dc3c84e02c8ae11bb35a43d79ef93122c3c3a40c4d459b/multidict-6.1.0-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:f67f217af4b1ff66c68a87318012de788dd95fcfeb24cc889011f4e1c7454dfd", size = 122793, upload-time = "2024-09-09T23:49:00.244Z" }, + { url = "https://files.pythonhosted.org/packages/62/c9/d386d01b43871e8e1631eb7b3695f6af071b7ae1ab716caf371100f0eb24/multidict-6.1.0-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:189f652a87e876098bbc67b4da1049afb5f5dfbaa310dd67c594b01c10388db6", size = 132762, upload-time = "2024-09-09T23:49:02.188Z" }, + { url = "https://files.pythonhosted.org/packages/69/ff/f70cb0a2f7a358acf48e32139ce3a150ff18c961ee9c714cc8c0dc7e3584/multidict-6.1.0-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:6bb5992037f7a9eff7991ebe4273ea7f51f1c1c511e6a2ce511d0e7bdb754492", size = 127872, upload-time = "2024-09-09T23:49:04.389Z" }, + { url = "https://files.pythonhosted.org/packages/89/5b/abea7db3ba4cd07752a9b560f9275a11787cd13f86849b5d99c1ceea921d/multidict-6.1.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:ac10f4c2b9e770c4e393876e35a7046879d195cd123b4f116d299d442b335bcd", size = 126161, upload-time = "2024-09-09T23:49:06.306Z" }, + { url = "https://files.pythonhosted.org/packages/22/03/acc77a4667cca4462ee974fc39990803e58fa573d5a923d6e82b7ef6da7e/multidict-6.1.0-cp38-cp38-win32.whl", hash = "sha256:e27bbb6d14416713a8bd7aaa1313c0fc8d44ee48d74497a0ff4c3a1b6ccb5167", size = 26338, upload-time = "2024-09-09T23:49:07.782Z" }, + { url = "https://files.pythonhosted.org/packages/90/bf/3d0c1cc9c8163abc24625fae89c0ade1ede9bccb6eceb79edf8cff3cca46/multidict-6.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:22f3105d4fb15c8f57ff3959a58fcab6ce36814486500cd7485651230ad4d4ef", size = 28736, upload-time = "2024-09-09T23:49:09.126Z" }, + { url = "https://files.pythonhosted.org/packages/e7/c9/9e153a6572b38ac5ff4434113af38acf8d5e9957897cdb1f513b3d6614ed/multidict-6.1.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:4e18b656c5e844539d506a0a06432274d7bd52a7487e6828c63a63d69185626c", size = 48550, upload-time = "2024-09-09T23:49:10.475Z" }, + { url = "https://files.pythonhosted.org/packages/76/f5/79565ddb629eba6c7f704f09a09df085c8dc04643b12506f10f718cee37a/multidict-6.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:a185f876e69897a6f3325c3f19f26a297fa058c5e456bfcff8015e9a27e83ae1", size = 29298, upload-time = "2024-09-09T23:49:12.119Z" }, + { url = "https://files.pythonhosted.org/packages/60/1b/9851878b704bc98e641a3e0bce49382ae9e05743dac6d97748feb5b7baba/multidict-6.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ab7c4ceb38d91570a650dba194e1ca87c2b543488fe9309b4212694174fd539c", size = 29641, upload-time = "2024-09-09T23:49:13.714Z" }, + { url = "https://files.pythonhosted.org/packages/89/87/d451d45aab9e422cb0fb2f7720c31a4c1d3012c740483c37f642eba568fb/multidict-6.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e617fb6b0b6953fffd762669610c1c4ffd05632c138d61ac7e14ad187870669c", size = 126202, upload-time = "2024-09-09T23:49:15.238Z" }, + { url = "https://files.pythonhosted.org/packages/fa/b4/27cbe9f3e2e469359887653f2e45470272eef7295139916cc21107c6b48c/multidict-6.1.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:16e5f4bf4e603eb1fdd5d8180f1a25f30056f22e55ce51fb3d6ad4ab29f7d96f", size = 133925, upload-time = "2024-09-09T23:49:16.786Z" }, + { url = "https://files.pythonhosted.org/packages/4d/a3/afc841899face8adfd004235ce759a37619f6ec99eafd959650c5ce4df57/multidict-6.1.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f4c035da3f544b1882bac24115f3e2e8760f10a0107614fc9839fd232200b875", size = 129039, upload-time = "2024-09-09T23:49:18.381Z" }, + { url = "https://files.pythonhosted.org/packages/5e/41/0d0fb18c1ad574f807196f5f3d99164edf9de3e169a58c6dc2d6ed5742b9/multidict-6.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:957cf8e4b6e123a9eea554fa7ebc85674674b713551de587eb318a2df3e00255", size = 124072, upload-time = "2024-09-09T23:49:20.115Z" }, + { url = "https://files.pythonhosted.org/packages/00/22/defd7a2e71a44e6e5b9a5428f972e5b572e7fe28e404dfa6519bbf057c93/multidict-6.1.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:483a6aea59cb89904e1ceabd2b47368b5600fb7de78a6e4a2c2987b2d256cf30", size = 116532, upload-time = "2024-09-09T23:49:21.685Z" }, + { url = "https://files.pythonhosted.org/packages/91/25/f7545102def0b1d456ab6449388eed2dfd822debba1d65af60194904a23a/multidict-6.1.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:87701f25a2352e5bf7454caa64757642734da9f6b11384c1f9d1a8e699758057", size = 128173, upload-time = "2024-09-09T23:49:23.657Z" }, + { url = "https://files.pythonhosted.org/packages/45/79/3dbe8d35fc99f5ea610813a72ab55f426cb9cf482f860fa8496e5409be11/multidict-6.1.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:682b987361e5fd7a139ed565e30d81fd81e9629acc7d925a205366877d8c8657", size = 122654, upload-time = "2024-09-09T23:49:25.7Z" }, + { url = "https://files.pythonhosted.org/packages/97/cb/209e735eeab96e1b160825b5d0b36c56d3862abff828fc43999bb957dcad/multidict-6.1.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:ce2186a7df133a9c895dea3331ddc5ddad42cdd0d1ea2f0a51e5d161e4762f28", size = 133197, upload-time = "2024-09-09T23:49:27.906Z" }, + { url = "https://files.pythonhosted.org/packages/e4/3a/a13808a7ada62808afccea67837a79d00ad6581440015ef00f726d064c2d/multidict-6.1.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:9f636b730f7e8cb19feb87094949ba54ee5357440b9658b2a32a5ce4bce53972", size = 129754, upload-time = "2024-09-09T23:49:29.508Z" }, + { url = "https://files.pythonhosted.org/packages/77/dd/8540e139eafb240079242da8f8ffdf9d3f4b4ad1aac5a786cd4050923783/multidict-6.1.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:73eae06aa53af2ea5270cc066dcaf02cc60d2994bbb2c4ef5764949257d10f43", size = 126402, upload-time = "2024-09-09T23:49:31.243Z" }, + { url = "https://files.pythonhosted.org/packages/86/99/e82e1a275d8b1ea16d3a251474262258dbbe41c05cce0c01bceda1fc8ea5/multidict-6.1.0-cp39-cp39-win32.whl", hash = "sha256:1ca0083e80e791cffc6efce7660ad24af66c8d4079d2a750b29001b53ff59ada", size = 26421, upload-time = "2024-09-09T23:49:32.648Z" }, + { url = "https://files.pythonhosted.org/packages/86/1c/9fa630272355af7e4446a2c7550c259f11ee422ab2d30ff90a0a71cf3d9e/multidict-6.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:aa466da5b15ccea564bdab9c89175c762bc12825f4659c11227f515cee76fa4a", size = 28791, upload-time = "2024-09-09T23:49:34.725Z" }, + { url = "https://files.pythonhosted.org/packages/99/b7/b9e70fde2c0f0c9af4cc5277782a89b66d35948ea3369ec9f598358c3ac5/multidict-6.1.0-py3-none-any.whl", hash = "sha256:48e171e52d1c4d33888e529b999e5900356b9ae588c2f09a52dcefb158b27506", size = 10051, upload-time = "2024-09-09T23:49:36.506Z" }, +] + +[[package]] +name = "multidict" +version = "6.7.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "typing-extensions", version = "4.15.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/80/1e/5492c365f222f907de1039b91f922b93fa4f764c713ee858d235495d8f50/multidict-6.7.0.tar.gz", hash = "sha256:c6e99d9a65ca282e578dfea819cfa9c0a62b2499d8677392e09feaf305e9e6f5", size = 101834, upload-time = "2025-10-06T14:52:30.657Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a9/63/7bdd4adc330abcca54c85728db2327130e49e52e8c3ce685cec44e0f2e9f/multidict-6.7.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:9f474ad5acda359c8758c8accc22032c6abe6dc87a8be2440d097785e27a9349", size = 77153, upload-time = "2025-10-06T14:48:26.409Z" }, + { url = "https://files.pythonhosted.org/packages/3f/bb/b6c35ff175ed1a3142222b78455ee31be71a8396ed3ab5280fbe3ebe4e85/multidict-6.7.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:4b7a9db5a870f780220e931d0002bbfd88fb53aceb6293251e2c839415c1b20e", size = 44993, upload-time = "2025-10-06T14:48:28.4Z" }, + { url = "https://files.pythonhosted.org/packages/e0/1f/064c77877c5fa6df6d346e68075c0f6998547afe952d6471b4c5f6a7345d/multidict-6.7.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:03ca744319864e92721195fa28c7a3b2bc7b686246b35e4078c1e4d0eb5466d3", size = 44607, upload-time = "2025-10-06T14:48:29.581Z" }, + { url = "https://files.pythonhosted.org/packages/04/7a/bf6aa92065dd47f287690000b3d7d332edfccb2277634cadf6a810463c6a/multidict-6.7.0-cp310-cp310-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:f0e77e3c0008bc9316e662624535b88d360c3a5d3f81e15cf12c139a75250046", size = 241847, upload-time = "2025-10-06T14:48:32.107Z" }, + { url = "https://files.pythonhosted.org/packages/94/39/297a8de920f76eda343e4ce05f3b489f0ab3f9504f2576dfb37b7c08ca08/multidict-6.7.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:08325c9e5367aa379a3496aa9a022fe8837ff22e00b94db256d3a1378c76ab32", size = 242616, upload-time = "2025-10-06T14:48:34.054Z" }, + { url = "https://files.pythonhosted.org/packages/39/3a/d0eee2898cfd9d654aea6cb8c4addc2f9756e9a7e09391cfe55541f917f7/multidict-6.7.0-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:e2862408c99f84aa571ab462d25236ef9cb12a602ea959ba9c9009a54902fc73", size = 222333, upload-time = "2025-10-06T14:48:35.9Z" }, + { url = "https://files.pythonhosted.org/packages/05/48/3b328851193c7a4240815b71eea165b49248867bbb6153a0aee227a0bb47/multidict-6.7.0-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:4d72a9a2d885f5c208b0cb91ff2ed43636bb7e345ec839ff64708e04f69a13cc", size = 253239, upload-time = "2025-10-06T14:48:37.302Z" }, + { url = "https://files.pythonhosted.org/packages/b1/ca/0706a98c8d126a89245413225ca4a3fefc8435014de309cf8b30acb68841/multidict-6.7.0-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:478cc36476687bac1514d651cbbaa94b86b0732fb6855c60c673794c7dd2da62", size = 251618, upload-time = "2025-10-06T14:48:38.963Z" }, + { url = "https://files.pythonhosted.org/packages/5e/4f/9c7992f245554d8b173f6f0a048ad24b3e645d883f096857ec2c0822b8bd/multidict-6.7.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:6843b28b0364dc605f21481c90fadb5f60d9123b442eb8a726bb74feef588a84", size = 241655, upload-time = "2025-10-06T14:48:40.312Z" }, + { url = "https://files.pythonhosted.org/packages/31/79/26a85991ae67efd1c0b1fc2e0c275b8a6aceeb155a68861f63f87a798f16/multidict-6.7.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:23bfeee5316266e5ee2d625df2d2c602b829435fc3a235c2ba2131495706e4a0", size = 239245, upload-time = "2025-10-06T14:48:41.848Z" }, + { url = "https://files.pythonhosted.org/packages/14/1e/75fa96394478930b79d0302eaf9a6c69f34005a1a5251ac8b9c336486ec9/multidict-6.7.0-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:680878b9f3d45c31e1f730eef731f9b0bc1da456155688c6745ee84eb818e90e", size = 233523, upload-time = "2025-10-06T14:48:43.749Z" }, + { url = "https://files.pythonhosted.org/packages/b2/5e/085544cb9f9c4ad2b5d97467c15f856df8d9bac410cffd5c43991a5d878b/multidict-6.7.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:eb866162ef2f45063acc7a53a88ef6fe8bf121d45c30ea3c9cd87ce7e191a8d4", size = 243129, upload-time = "2025-10-06T14:48:45.225Z" }, + { url = "https://files.pythonhosted.org/packages/b9/c3/e9d9e2f20c9474e7a8fcef28f863c5cbd29bb5adce6b70cebe8bdad0039d/multidict-6.7.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:df0e3bf7993bdbeca5ac25aa859cf40d39019e015c9c91809ba7093967f7a648", size = 248999, upload-time = "2025-10-06T14:48:46.703Z" }, + { url = "https://files.pythonhosted.org/packages/b5/3f/df171b6efa3239ae33b97b887e42671cd1d94d460614bfb2c30ffdab3b95/multidict-6.7.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:661709cdcd919a2ece2234f9bae7174e5220c80b034585d7d8a755632d3e2111", size = 243711, upload-time = "2025-10-06T14:48:48.146Z" }, + { url = "https://files.pythonhosted.org/packages/3c/2f/9b5564888c4e14b9af64c54acf149263721a283aaf4aa0ae89b091d5d8c1/multidict-6.7.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:096f52730c3fb8ed419db2d44391932b63891b2c5ed14850a7e215c0ba9ade36", size = 237504, upload-time = "2025-10-06T14:48:49.447Z" }, + { url = "https://files.pythonhosted.org/packages/6c/3a/0bd6ca0f7d96d790542d591c8c3354c1e1b6bfd2024d4d92dc3d87485ec7/multidict-6.7.0-cp310-cp310-win32.whl", hash = "sha256:afa8a2978ec65d2336305550535c9c4ff50ee527914328c8677b3973ade52b85", size = 41422, upload-time = "2025-10-06T14:48:50.789Z" }, + { url = "https://files.pythonhosted.org/packages/00/35/f6a637ea2c75f0d3b7c7d41b1189189acff0d9deeb8b8f35536bb30f5e33/multidict-6.7.0-cp310-cp310-win_amd64.whl", hash = "sha256:b15b3afff74f707b9275d5ba6a91ae8f6429c3ffb29bbfd216b0b375a56f13d7", size = 46050, upload-time = "2025-10-06T14:48:51.938Z" }, + { url = "https://files.pythonhosted.org/packages/e7/b8/f7bf8329b39893d02d9d95cf610c75885d12fc0f402b1c894e1c8e01c916/multidict-6.7.0-cp310-cp310-win_arm64.whl", hash = "sha256:4b73189894398d59131a66ff157837b1fafea9974be486d036bb3d32331fdbf0", size = 43153, upload-time = "2025-10-06T14:48:53.146Z" }, + { url = "https://files.pythonhosted.org/packages/34/9e/5c727587644d67b2ed479041e4b1c58e30afc011e3d45d25bbe35781217c/multidict-6.7.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4d409aa42a94c0b3fa617708ef5276dfe81012ba6753a0370fcc9d0195d0a1fc", size = 76604, upload-time = "2025-10-06T14:48:54.277Z" }, + { url = "https://files.pythonhosted.org/packages/17/e4/67b5c27bd17c085a5ea8f1ec05b8a3e5cba0ca734bfcad5560fb129e70ca/multidict-6.7.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:14c9e076eede3b54c636f8ce1c9c252b5f057c62131211f0ceeec273810c9721", size = 44715, upload-time = "2025-10-06T14:48:55.445Z" }, + { url = "https://files.pythonhosted.org/packages/4d/e1/866a5d77be6ea435711bef2a4291eed11032679b6b28b56b4776ab06ba3e/multidict-6.7.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:4c09703000a9d0fa3c3404b27041e574cc7f4df4c6563873246d0e11812a94b6", size = 44332, upload-time = "2025-10-06T14:48:56.706Z" }, + { url = "https://files.pythonhosted.org/packages/31/61/0c2d50241ada71ff61a79518db85ada85fdabfcf395d5968dae1cbda04e5/multidict-6.7.0-cp311-cp311-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:a265acbb7bb33a3a2d626afbe756371dce0279e7b17f4f4eda406459c2b5ff1c", size = 245212, upload-time = "2025-10-06T14:48:58.042Z" }, + { url = "https://files.pythonhosted.org/packages/ac/e0/919666a4e4b57fff1b57f279be1c9316e6cdc5de8a8b525d76f6598fefc7/multidict-6.7.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:51cb455de290ae462593e5b1cb1118c5c22ea7f0d3620d9940bf695cea5a4bd7", size = 246671, upload-time = "2025-10-06T14:49:00.004Z" }, + { url = "https://files.pythonhosted.org/packages/a1/cc/d027d9c5a520f3321b65adea289b965e7bcbd2c34402663f482648c716ce/multidict-6.7.0-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:db99677b4457c7a5c5a949353e125ba72d62b35f74e26da141530fbb012218a7", size = 225491, upload-time = "2025-10-06T14:49:01.393Z" }, + { url = "https://files.pythonhosted.org/packages/75/c4/bbd633980ce6155a28ff04e6a6492dd3335858394d7bb752d8b108708558/multidict-6.7.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:f470f68adc395e0183b92a2f4689264d1ea4b40504a24d9882c27375e6662bb9", size = 257322, upload-time = "2025-10-06T14:49:02.745Z" }, + { url = "https://files.pythonhosted.org/packages/4c/6d/d622322d344f1f053eae47e033b0b3f965af01212de21b10bcf91be991fb/multidict-6.7.0-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:0db4956f82723cc1c270de9c6e799b4c341d327762ec78ef82bb962f79cc07d8", size = 254694, upload-time = "2025-10-06T14:49:04.15Z" }, + { url = "https://files.pythonhosted.org/packages/a8/9f/78f8761c2705d4c6d7516faed63c0ebdac569f6db1bef95e0d5218fdc146/multidict-6.7.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:3e56d780c238f9e1ae66a22d2adf8d16f485381878250db8d496623cd38b22bd", size = 246715, upload-time = "2025-10-06T14:49:05.967Z" }, + { url = "https://files.pythonhosted.org/packages/78/59/950818e04f91b9c2b95aab3d923d9eabd01689d0dcd889563988e9ea0fd8/multidict-6.7.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:9d14baca2ee12c1a64740d4531356ba50b82543017f3ad6de0deb943c5979abb", size = 243189, upload-time = "2025-10-06T14:49:07.37Z" }, + { url = "https://files.pythonhosted.org/packages/7a/3d/77c79e1934cad2ee74991840f8a0110966d9599b3af95964c0cd79bb905b/multidict-6.7.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:295a92a76188917c7f99cda95858c822f9e4aae5824246bba9b6b44004ddd0a6", size = 237845, upload-time = "2025-10-06T14:49:08.759Z" }, + { url = "https://files.pythonhosted.org/packages/63/1b/834ce32a0a97a3b70f86437f685f880136677ac00d8bce0027e9fd9c2db7/multidict-6.7.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:39f1719f57adbb767ef592a50ae5ebb794220d1188f9ca93de471336401c34d2", size = 246374, upload-time = "2025-10-06T14:49:10.574Z" }, + { url = "https://files.pythonhosted.org/packages/23/ef/43d1c3ba205b5dec93dc97f3fba179dfa47910fc73aaaea4f7ceb41cec2a/multidict-6.7.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:0a13fb8e748dfc94749f622de065dd5c1def7e0d2216dba72b1d8069a389c6ff", size = 253345, upload-time = "2025-10-06T14:49:12.331Z" }, + { url = "https://files.pythonhosted.org/packages/6b/03/eaf95bcc2d19ead522001f6a650ef32811aa9e3624ff0ad37c445c7a588c/multidict-6.7.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:e3aa16de190d29a0ea1b48253c57d99a68492c8dd8948638073ab9e74dc9410b", size = 246940, upload-time = "2025-10-06T14:49:13.821Z" }, + { url = "https://files.pythonhosted.org/packages/e8/df/ec8a5fd66ea6cd6f525b1fcbb23511b033c3e9bc42b81384834ffa484a62/multidict-6.7.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:a048ce45dcdaaf1defb76b2e684f997fb5abf74437b6cb7b22ddad934a964e34", size = 242229, upload-time = "2025-10-06T14:49:15.603Z" }, + { url = "https://files.pythonhosted.org/packages/8a/a2/59b405d59fd39ec86d1142630e9049243015a5f5291ba49cadf3c090c541/multidict-6.7.0-cp311-cp311-win32.whl", hash = "sha256:a90af66facec4cebe4181b9e62a68be65e45ac9b52b67de9eec118701856e7ff", size = 41308, upload-time = "2025-10-06T14:49:16.871Z" }, + { url = "https://files.pythonhosted.org/packages/32/0f/13228f26f8b882c34da36efa776c3b7348455ec383bab4a66390e42963ae/multidict-6.7.0-cp311-cp311-win_amd64.whl", hash = "sha256:95b5ffa4349df2887518bb839409bcf22caa72d82beec453216802f475b23c81", size = 46037, upload-time = "2025-10-06T14:49:18.457Z" }, + { url = "https://files.pythonhosted.org/packages/84/1f/68588e31b000535a3207fd3c909ebeec4fb36b52c442107499c18a896a2a/multidict-6.7.0-cp311-cp311-win_arm64.whl", hash = "sha256:329aa225b085b6f004a4955271a7ba9f1087e39dcb7e65f6284a988264a63912", size = 43023, upload-time = "2025-10-06T14:49:19.648Z" }, + { url = "https://files.pythonhosted.org/packages/c2/9e/9f61ac18d9c8b475889f32ccfa91c9f59363480613fc807b6e3023d6f60b/multidict-6.7.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:8a3862568a36d26e650a19bb5cbbba14b71789032aebc0423f8cc5f150730184", size = 76877, upload-time = "2025-10-06T14:49:20.884Z" }, + { url = "https://files.pythonhosted.org/packages/38/6f/614f09a04e6184f8824268fce4bc925e9849edfa654ddd59f0b64508c595/multidict-6.7.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:960c60b5849b9b4f9dcc9bea6e3626143c252c74113df2c1540aebce70209b45", size = 45467, upload-time = "2025-10-06T14:49:22.054Z" }, + { url = "https://files.pythonhosted.org/packages/b3/93/c4f67a436dd026f2e780c433277fff72be79152894d9fc36f44569cab1a6/multidict-6.7.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2049be98fb57a31b4ccf870bf377af2504d4ae35646a19037ec271e4c07998aa", size = 43834, upload-time = "2025-10-06T14:49:23.566Z" }, + { url = "https://files.pythonhosted.org/packages/7f/f5/013798161ca665e4a422afbc5e2d9e4070142a9ff8905e482139cd09e4d0/multidict-6.7.0-cp312-cp312-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:0934f3843a1860dd465d38895c17fce1f1cb37295149ab05cd1b9a03afacb2a7", size = 250545, upload-time = "2025-10-06T14:49:24.882Z" }, + { url = "https://files.pythonhosted.org/packages/71/2f/91dbac13e0ba94669ea5119ba267c9a832f0cb65419aca75549fcf09a3dc/multidict-6.7.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b3e34f3a1b8131ba06f1a73adab24f30934d148afcd5f5de9a73565a4404384e", size = 258305, upload-time = "2025-10-06T14:49:26.778Z" }, + { url = "https://files.pythonhosted.org/packages/ef/b0/754038b26f6e04488b48ac621f779c341338d78503fb45403755af2df477/multidict-6.7.0-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:efbb54e98446892590dc2458c19c10344ee9a883a79b5cec4bc34d6656e8d546", size = 242363, upload-time = "2025-10-06T14:49:28.562Z" }, + { url = "https://files.pythonhosted.org/packages/87/15/9da40b9336a7c9fa606c4cf2ed80a649dffeb42b905d4f63a1d7eb17d746/multidict-6.7.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:a35c5fc61d4f51eb045061e7967cfe3123d622cd500e8868e7c0c592a09fedc4", size = 268375, upload-time = "2025-10-06T14:49:29.96Z" }, + { url = "https://files.pythonhosted.org/packages/82/72/c53fcade0cc94dfaad583105fd92b3a783af2091eddcb41a6d5a52474000/multidict-6.7.0-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:29fe6740ebccba4175af1b9b87bf553e9c15cd5868ee967e010efcf94e4fd0f1", size = 269346, upload-time = "2025-10-06T14:49:31.404Z" }, + { url = "https://files.pythonhosted.org/packages/0d/e2/9baffdae21a76f77ef8447f1a05a96ec4bc0a24dae08767abc0a2fe680b8/multidict-6.7.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:123e2a72e20537add2f33a79e605f6191fba2afda4cbb876e35c1a7074298a7d", size = 256107, upload-time = "2025-10-06T14:49:32.974Z" }, + { url = "https://files.pythonhosted.org/packages/3c/06/3f06f611087dc60d65ef775f1fb5aca7c6d61c6db4990e7cda0cef9b1651/multidict-6.7.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:b284e319754366c1aee2267a2036248b24eeb17ecd5dc16022095e747f2f4304", size = 253592, upload-time = "2025-10-06T14:49:34.52Z" }, + { url = "https://files.pythonhosted.org/packages/20/24/54e804ec7945b6023b340c412ce9c3f81e91b3bf5fa5ce65558740141bee/multidict-6.7.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:803d685de7be4303b5a657b76e2f6d1240e7e0a8aa2968ad5811fa2285553a12", size = 251024, upload-time = "2025-10-06T14:49:35.956Z" }, + { url = "https://files.pythonhosted.org/packages/14/48/011cba467ea0b17ceb938315d219391d3e421dfd35928e5dbdc3f4ae76ef/multidict-6.7.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:c04a328260dfd5db8c39538f999f02779012268f54614902d0afc775d44e0a62", size = 251484, upload-time = "2025-10-06T14:49:37.631Z" }, + { url = "https://files.pythonhosted.org/packages/0d/2f/919258b43bb35b99fa127435cfb2d91798eb3a943396631ef43e3720dcf4/multidict-6.7.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:8a19cdb57cd3df4cd865849d93ee14920fb97224300c88501f16ecfa2604b4e0", size = 263579, upload-time = "2025-10-06T14:49:39.502Z" }, + { url = "https://files.pythonhosted.org/packages/31/22/a0e884d86b5242b5a74cf08e876bdf299e413016b66e55511f7a804a366e/multidict-6.7.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:9b2fd74c52accced7e75de26023b7dccee62511a600e62311b918ec5c168fc2a", size = 259654, upload-time = "2025-10-06T14:49:41.32Z" }, + { url = "https://files.pythonhosted.org/packages/b2/e5/17e10e1b5c5f5a40f2fcbb45953c9b215f8a4098003915e46a93f5fcaa8f/multidict-6.7.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:3e8bfdd0e487acf992407a140d2589fe598238eaeffa3da8448d63a63cd363f8", size = 251511, upload-time = "2025-10-06T14:49:46.021Z" }, + { url = "https://files.pythonhosted.org/packages/e3/9a/201bb1e17e7af53139597069c375e7b0dcbd47594604f65c2d5359508566/multidict-6.7.0-cp312-cp312-win32.whl", hash = "sha256:dd32a49400a2c3d52088e120ee00c1e3576cbff7e10b98467962c74fdb762ed4", size = 41895, upload-time = "2025-10-06T14:49:48.718Z" }, + { url = "https://files.pythonhosted.org/packages/46/e2/348cd32faad84eaf1d20cce80e2bb0ef8d312c55bca1f7fa9865e7770aaf/multidict-6.7.0-cp312-cp312-win_amd64.whl", hash = "sha256:92abb658ef2d7ef22ac9f8bb88e8b6c3e571671534e029359b6d9e845923eb1b", size = 46073, upload-time = "2025-10-06T14:49:50.28Z" }, + { url = "https://files.pythonhosted.org/packages/25/ec/aad2613c1910dce907480e0c3aa306905830f25df2e54ccc9dea450cb5aa/multidict-6.7.0-cp312-cp312-win_arm64.whl", hash = "sha256:490dab541a6a642ce1a9d61a4781656b346a55c13038f0b1244653828e3a83ec", size = 43226, upload-time = "2025-10-06T14:49:52.304Z" }, + { url = "https://files.pythonhosted.org/packages/d2/86/33272a544eeb36d66e4d9a920602d1a2f57d4ebea4ef3cdfe5a912574c95/multidict-6.7.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:bee7c0588aa0076ce77c0ea5d19a68d76ad81fcd9fe8501003b9a24f9d4000f6", size = 76135, upload-time = "2025-10-06T14:49:54.26Z" }, + { url = "https://files.pythonhosted.org/packages/91/1c/eb97db117a1ebe46d457a3d235a7b9d2e6dcab174f42d1b67663dd9e5371/multidict-6.7.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:7ef6b61cad77091056ce0e7ce69814ef72afacb150b7ac6a3e9470def2198159", size = 45117, upload-time = "2025-10-06T14:49:55.82Z" }, + { url = "https://files.pythonhosted.org/packages/f1/d8/6c3442322e41fb1dd4de8bd67bfd11cd72352ac131f6368315617de752f1/multidict-6.7.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:9c0359b1ec12b1d6849c59f9d319610b7f20ef990a6d454ab151aa0e3b9f78ca", size = 43472, upload-time = "2025-10-06T14:49:57.048Z" }, + { url = "https://files.pythonhosted.org/packages/75/3f/e2639e80325af0b6c6febdf8e57cc07043ff15f57fa1ef808f4ccb5ac4cd/multidict-6.7.0-cp313-cp313-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:cd240939f71c64bd658f186330603aac1a9a81bf6273f523fca63673cb7378a8", size = 249342, upload-time = "2025-10-06T14:49:58.368Z" }, + { url = "https://files.pythonhosted.org/packages/5d/cc/84e0585f805cbeaa9cbdaa95f9a3d6aed745b9d25700623ac89a6ecff400/multidict-6.7.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:a60a4d75718a5efa473ebd5ab685786ba0c67b8381f781d1be14da49f1a2dc60", size = 257082, upload-time = "2025-10-06T14:49:59.89Z" }, + { url = "https://files.pythonhosted.org/packages/b0/9c/ac851c107c92289acbbf5cfb485694084690c1b17e555f44952c26ddc5bd/multidict-6.7.0-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:53a42d364f323275126aff81fb67c5ca1b7a04fda0546245730a55c8c5f24bc4", size = 240704, upload-time = "2025-10-06T14:50:01.485Z" }, + { url = "https://files.pythonhosted.org/packages/50/cc/5f93e99427248c09da95b62d64b25748a5f5c98c7c2ab09825a1d6af0e15/multidict-6.7.0-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:3b29b980d0ddbecb736735ee5bef69bb2ddca56eff603c86f3f29a1128299b4f", size = 266355, upload-time = "2025-10-06T14:50:02.955Z" }, + { url = "https://files.pythonhosted.org/packages/ec/0c/2ec1d883ceb79c6f7f6d7ad90c919c898f5d1c6ea96d322751420211e072/multidict-6.7.0-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:f8a93b1c0ed2d04b97a5e9336fd2d33371b9a6e29ab7dd6503d63407c20ffbaf", size = 267259, upload-time = "2025-10-06T14:50:04.446Z" }, + { url = "https://files.pythonhosted.org/packages/c6/2d/f0b184fa88d6630aa267680bdb8623fb69cb0d024b8c6f0d23f9a0f406d3/multidict-6.7.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9ff96e8815eecacc6645da76c413eb3b3d34cfca256c70b16b286a687d013c32", size = 254903, upload-time = "2025-10-06T14:50:05.98Z" }, + { url = "https://files.pythonhosted.org/packages/06/c9/11ea263ad0df7dfabcad404feb3c0dd40b131bc7f232d5537f2fb1356951/multidict-6.7.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:7516c579652f6a6be0e266aec0acd0db80829ca305c3d771ed898538804c2036", size = 252365, upload-time = "2025-10-06T14:50:07.511Z" }, + { url = "https://files.pythonhosted.org/packages/41/88/d714b86ee2c17d6e09850c70c9d310abac3d808ab49dfa16b43aba9d53fd/multidict-6.7.0-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:040f393368e63fb0f3330e70c26bfd336656bed925e5cbe17c9da839a6ab13ec", size = 250062, upload-time = "2025-10-06T14:50:09.074Z" }, + { url = "https://files.pythonhosted.org/packages/15/fe/ad407bb9e818c2b31383f6131ca19ea7e35ce93cf1310fce69f12e89de75/multidict-6.7.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:b3bc26a951007b1057a1c543af845f1c7e3e71cc240ed1ace7bf4484aa99196e", size = 249683, upload-time = "2025-10-06T14:50:10.714Z" }, + { url = "https://files.pythonhosted.org/packages/8c/a4/a89abdb0229e533fb925e7c6e5c40201c2873efebc9abaf14046a4536ee6/multidict-6.7.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:7b022717c748dd1992a83e219587aabe45980d88969f01b316e78683e6285f64", size = 261254, upload-time = "2025-10-06T14:50:12.28Z" }, + { url = "https://files.pythonhosted.org/packages/8d/aa/0e2b27bd88b40a4fb8dc53dd74eecac70edaa4c1dd0707eb2164da3675b3/multidict-6.7.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:9600082733859f00d79dee64effc7aef1beb26adb297416a4ad2116fd61374bd", size = 257967, upload-time = "2025-10-06T14:50:14.16Z" }, + { url = "https://files.pythonhosted.org/packages/d0/8e/0c67b7120d5d5f6d874ed85a085f9dc770a7f9d8813e80f44a9fec820bb7/multidict-6.7.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:94218fcec4d72bc61df51c198d098ce2b378e0ccbac41ddbed5ef44092913288", size = 250085, upload-time = "2025-10-06T14:50:15.639Z" }, + { url = "https://files.pythonhosted.org/packages/ba/55/b73e1d624ea4b8fd4dd07a3bb70f6e4c7c6c5d9d640a41c6ffe5cdbd2a55/multidict-6.7.0-cp313-cp313-win32.whl", hash = "sha256:a37bd74c3fa9d00be2d7b8eca074dc56bd8077ddd2917a839bd989612671ed17", size = 41713, upload-time = "2025-10-06T14:50:17.066Z" }, + { url = "https://files.pythonhosted.org/packages/32/31/75c59e7d3b4205075b4c183fa4ca398a2daf2303ddf616b04ae6ef55cffe/multidict-6.7.0-cp313-cp313-win_amd64.whl", hash = "sha256:30d193c6cc6d559db42b6bcec8a5d395d34d60c9877a0b71ecd7c204fcf15390", size = 45915, upload-time = "2025-10-06T14:50:18.264Z" }, + { url = "https://files.pythonhosted.org/packages/31/2a/8987831e811f1184c22bc2e45844934385363ee61c0a2dcfa8f71b87e608/multidict-6.7.0-cp313-cp313-win_arm64.whl", hash = "sha256:ea3334cabe4d41b7ccd01e4d349828678794edbc2d3ae97fc162a3312095092e", size = 43077, upload-time = "2025-10-06T14:50:19.853Z" }, + { url = "https://files.pythonhosted.org/packages/e8/68/7b3a5170a382a340147337b300b9eb25a9ddb573bcdfff19c0fa3f31ffba/multidict-6.7.0-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:ad9ce259f50abd98a1ca0aa6e490b58c316a0fce0617f609723e40804add2c00", size = 83114, upload-time = "2025-10-06T14:50:21.223Z" }, + { url = "https://files.pythonhosted.org/packages/55/5c/3fa2d07c84df4e302060f555bbf539310980362236ad49f50eeb0a1c1eb9/multidict-6.7.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:07f5594ac6d084cbb5de2df218d78baf55ef150b91f0ff8a21cc7a2e3a5a58eb", size = 48442, upload-time = "2025-10-06T14:50:22.871Z" }, + { url = "https://files.pythonhosted.org/packages/fc/56/67212d33239797f9bd91962bb899d72bb0f4c35a8652dcdb8ed049bef878/multidict-6.7.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:0591b48acf279821a579282444814a2d8d0af624ae0bc600aa4d1b920b6e924b", size = 46885, upload-time = "2025-10-06T14:50:24.258Z" }, + { url = "https://files.pythonhosted.org/packages/46/d1/908f896224290350721597a61a69cd19b89ad8ee0ae1f38b3f5cd12ea2ac/multidict-6.7.0-cp313-cp313t-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:749a72584761531d2b9467cfbdfd29487ee21124c304c4b6cb760d8777b27f9c", size = 242588, upload-time = "2025-10-06T14:50:25.716Z" }, + { url = "https://files.pythonhosted.org/packages/ab/67/8604288bbd68680eee0ab568fdcb56171d8b23a01bcd5cb0c8fedf6e5d99/multidict-6.7.0-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:6b4c3d199f953acd5b446bf7c0de1fe25d94e09e79086f8dc2f48a11a129cdf1", size = 249966, upload-time = "2025-10-06T14:50:28.192Z" }, + { url = "https://files.pythonhosted.org/packages/20/33/9228d76339f1ba51e3efef7da3ebd91964d3006217aae13211653193c3ff/multidict-6.7.0-cp313-cp313t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:9fb0211dfc3b51efea2f349ec92c114d7754dd62c01f81c3e32b765b70c45c9b", size = 228618, upload-time = "2025-10-06T14:50:29.82Z" }, + { url = "https://files.pythonhosted.org/packages/f8/2d/25d9b566d10cab1c42b3b9e5b11ef79c9111eaf4463b8c257a3bd89e0ead/multidict-6.7.0-cp313-cp313t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:a027ec240fe73a8d6281872690b988eed307cd7d91b23998ff35ff577ca688b5", size = 257539, upload-time = "2025-10-06T14:50:31.731Z" }, + { url = "https://files.pythonhosted.org/packages/b6/b1/8d1a965e6637fc33de3c0d8f414485c2b7e4af00f42cab3d84e7b955c222/multidict-6.7.0-cp313-cp313t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d1d964afecdf3a8288789df2f5751dc0a8261138c3768d9af117ed384e538fad", size = 256345, upload-time = "2025-10-06T14:50:33.26Z" }, + { url = "https://files.pythonhosted.org/packages/ba/0c/06b5a8adbdeedada6f4fb8d8f193d44a347223b11939b42953eeb6530b6b/multidict-6.7.0-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:caf53b15b1b7df9fbd0709aa01409000a2b4dd03a5f6f5cc548183c7c8f8b63c", size = 247934, upload-time = "2025-10-06T14:50:34.808Z" }, + { url = "https://files.pythonhosted.org/packages/8f/31/b2491b5fe167ca044c6eb4b8f2c9f3b8a00b24c432c365358eadac5d7625/multidict-6.7.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:654030da3197d927f05a536a66186070e98765aa5142794c9904555d3a9d8fb5", size = 245243, upload-time = "2025-10-06T14:50:36.436Z" }, + { url = "https://files.pythonhosted.org/packages/61/1a/982913957cb90406c8c94f53001abd9eafc271cb3e70ff6371590bec478e/multidict-6.7.0-cp313-cp313t-musllinux_1_2_armv7l.whl", hash = "sha256:2090d3718829d1e484706a2f525e50c892237b2bf9b17a79b059cb98cddc2f10", size = 235878, upload-time = "2025-10-06T14:50:37.953Z" }, + { url = "https://files.pythonhosted.org/packages/be/c0/21435d804c1a1cf7a2608593f4d19bca5bcbd7a81a70b253fdd1c12af9c0/multidict-6.7.0-cp313-cp313t-musllinux_1_2_i686.whl", hash = "sha256:2d2cfeec3f6f45651b3d408c4acec0ebf3daa9bc8a112a084206f5db5d05b754", size = 243452, upload-time = "2025-10-06T14:50:39.574Z" }, + { url = "https://files.pythonhosted.org/packages/54/0a/4349d540d4a883863191be6eb9a928846d4ec0ea007d3dcd36323bb058ac/multidict-6.7.0-cp313-cp313t-musllinux_1_2_ppc64le.whl", hash = "sha256:4ef089f985b8c194d341eb2c24ae6e7408c9a0e2e5658699c92f497437d88c3c", size = 252312, upload-time = "2025-10-06T14:50:41.612Z" }, + { url = "https://files.pythonhosted.org/packages/26/64/d5416038dbda1488daf16b676e4dbfd9674dde10a0cc8f4fc2b502d8125d/multidict-6.7.0-cp313-cp313t-musllinux_1_2_s390x.whl", hash = "sha256:e93a0617cd16998784bf4414c7e40f17a35d2350e5c6f0bd900d3a8e02bd3762", size = 246935, upload-time = "2025-10-06T14:50:43.972Z" }, + { url = "https://files.pythonhosted.org/packages/9f/8c/8290c50d14e49f35e0bd4abc25e1bc7711149ca9588ab7d04f886cdf03d9/multidict-6.7.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:f0feece2ef8ebc42ed9e2e8c78fc4aa3cf455733b507c09ef7406364c94376c6", size = 243385, upload-time = "2025-10-06T14:50:45.648Z" }, + { url = "https://files.pythonhosted.org/packages/ef/a0/f83ae75e42d694b3fbad3e047670e511c138be747bc713cf1b10d5096416/multidict-6.7.0-cp313-cp313t-win32.whl", hash = "sha256:19a1d55338ec1be74ef62440ca9e04a2f001a04d0cc49a4983dc320ff0f3212d", size = 47777, upload-time = "2025-10-06T14:50:47.154Z" }, + { url = "https://files.pythonhosted.org/packages/dc/80/9b174a92814a3830b7357307a792300f42c9e94664b01dee8e457551fa66/multidict-6.7.0-cp313-cp313t-win_amd64.whl", hash = "sha256:3da4fb467498df97e986af166b12d01f05d2e04f978a9c1c680ea1988e0bc4b6", size = 53104, upload-time = "2025-10-06T14:50:48.851Z" }, + { url = "https://files.pythonhosted.org/packages/cc/28/04baeaf0428d95bb7a7bea0e691ba2f31394338ba424fb0679a9ed0f4c09/multidict-6.7.0-cp313-cp313t-win_arm64.whl", hash = "sha256:b4121773c49a0776461f4a904cdf6264c88e42218aaa8407e803ca8025872792", size = 45503, upload-time = "2025-10-06T14:50:50.16Z" }, + { url = "https://files.pythonhosted.org/packages/e2/b1/3da6934455dd4b261d4c72f897e3a5728eba81db59959f3a639245891baa/multidict-6.7.0-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:3bab1e4aff7adaa34410f93b1f8e57c4b36b9af0426a76003f441ee1d3c7e842", size = 75128, upload-time = "2025-10-06T14:50:51.92Z" }, + { url = "https://files.pythonhosted.org/packages/14/2c/f069cab5b51d175a1a2cb4ccdf7a2c2dabd58aa5bd933fa036a8d15e2404/multidict-6.7.0-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:b8512bac933afc3e45fb2b18da8e59b78d4f408399a960339598374d4ae3b56b", size = 44410, upload-time = "2025-10-06T14:50:53.275Z" }, + { url = "https://files.pythonhosted.org/packages/42/e2/64bb41266427af6642b6b128e8774ed84c11b80a90702c13ac0a86bb10cc/multidict-6.7.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:79dcf9e477bc65414ebfea98ffd013cb39552b5ecd62908752e0e413d6d06e38", size = 43205, upload-time = "2025-10-06T14:50:54.911Z" }, + { url = "https://files.pythonhosted.org/packages/02/68/6b086fef8a3f1a8541b9236c594f0c9245617c29841f2e0395d979485cde/multidict-6.7.0-cp314-cp314-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:31bae522710064b5cbeddaf2e9f32b1abab70ac6ac91d42572502299e9953128", size = 245084, upload-time = "2025-10-06T14:50:56.369Z" }, + { url = "https://files.pythonhosted.org/packages/15/ee/f524093232007cd7a75c1d132df70f235cfd590a7c9eaccd7ff422ef4ae8/multidict-6.7.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4a0df7ff02397bb63e2fd22af2c87dfa39e8c7f12947bc524dbdc528282c7e34", size = 252667, upload-time = "2025-10-06T14:50:57.991Z" }, + { url = "https://files.pythonhosted.org/packages/02/a5/eeb3f43ab45878f1895118c3ef157a480db58ede3f248e29b5354139c2c9/multidict-6.7.0-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:7a0222514e8e4c514660e182d5156a415c13ef0aabbd71682fc714e327b95e99", size = 233590, upload-time = "2025-10-06T14:50:59.589Z" }, + { url = "https://files.pythonhosted.org/packages/6a/1e/76d02f8270b97269d7e3dbd45644b1785bda457b474315f8cf999525a193/multidict-6.7.0-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:2397ab4daaf2698eb51a76721e98db21ce4f52339e535725de03ea962b5a3202", size = 264112, upload-time = "2025-10-06T14:51:01.183Z" }, + { url = "https://files.pythonhosted.org/packages/76/0b/c28a70ecb58963847c2a8efe334904cd254812b10e535aefb3bcce513918/multidict-6.7.0-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:8891681594162635948a636c9fe0ff21746aeb3dd5463f6e25d9bea3a8a39ca1", size = 261194, upload-time = "2025-10-06T14:51:02.794Z" }, + { url = "https://files.pythonhosted.org/packages/b4/63/2ab26e4209773223159b83aa32721b4021ffb08102f8ac7d689c943fded1/multidict-6.7.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:18706cc31dbf402a7945916dd5cddf160251b6dab8a2c5f3d6d5a55949f676b3", size = 248510, upload-time = "2025-10-06T14:51:04.724Z" }, + { url = "https://files.pythonhosted.org/packages/93/cd/06c1fa8282af1d1c46fd55c10a7930af652afdce43999501d4d68664170c/multidict-6.7.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:f844a1bbf1d207dd311a56f383f7eda2d0e134921d45751842d8235e7778965d", size = 248395, upload-time = "2025-10-06T14:51:06.306Z" }, + { url = "https://files.pythonhosted.org/packages/99/ac/82cb419dd6b04ccf9e7e61befc00c77614fc8134362488b553402ecd55ce/multidict-6.7.0-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:d4393e3581e84e5645506923816b9cc81f5609a778c7e7534054091acc64d1c6", size = 239520, upload-time = "2025-10-06T14:51:08.091Z" }, + { url = "https://files.pythonhosted.org/packages/fa/f3/a0f9bf09493421bd8716a362e0cd1d244f5a6550f5beffdd6b47e885b331/multidict-6.7.0-cp314-cp314-musllinux_1_2_i686.whl", hash = "sha256:fbd18dc82d7bf274b37aa48d664534330af744e03bccf696d6f4c6042e7d19e7", size = 245479, upload-time = "2025-10-06T14:51:10.365Z" }, + { url = "https://files.pythonhosted.org/packages/8d/01/476d38fc73a212843f43c852b0eee266b6971f0e28329c2184a8df90c376/multidict-6.7.0-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:b6234e14f9314731ec45c42fc4554b88133ad53a09092cc48a88e771c125dadb", size = 258903, upload-time = "2025-10-06T14:51:12.466Z" }, + { url = "https://files.pythonhosted.org/packages/49/6d/23faeb0868adba613b817d0e69c5f15531b24d462af8012c4f6de4fa8dc3/multidict-6.7.0-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:08d4379f9744d8f78d98c8673c06e202ffa88296f009c71bbafe8a6bf847d01f", size = 252333, upload-time = "2025-10-06T14:51:14.48Z" }, + { url = "https://files.pythonhosted.org/packages/1e/cc/48d02ac22b30fa247f7dad82866e4b1015431092f4ba6ebc7e77596e0b18/multidict-6.7.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:9fe04da3f79387f450fd0061d4dd2e45a72749d31bf634aecc9e27f24fdc4b3f", size = 243411, upload-time = "2025-10-06T14:51:16.072Z" }, + { url = "https://files.pythonhosted.org/packages/4a/03/29a8bf5a18abf1fe34535c88adbdfa88c9fb869b5a3b120692c64abe8284/multidict-6.7.0-cp314-cp314-win32.whl", hash = "sha256:fbafe31d191dfa7c4c51f7a6149c9fb7e914dcf9ffead27dcfd9f1ae382b3885", size = 40940, upload-time = "2025-10-06T14:51:17.544Z" }, + { url = "https://files.pythonhosted.org/packages/82/16/7ed27b680791b939de138f906d5cf2b4657b0d45ca6f5dd6236fdddafb1a/multidict-6.7.0-cp314-cp314-win_amd64.whl", hash = "sha256:2f67396ec0310764b9222a1728ced1ab638f61aadc6226f17a71dd9324f9a99c", size = 45087, upload-time = "2025-10-06T14:51:18.875Z" }, + { url = "https://files.pythonhosted.org/packages/cd/3c/e3e62eb35a1950292fe39315d3c89941e30a9d07d5d2df42965ab041da43/multidict-6.7.0-cp314-cp314-win_arm64.whl", hash = "sha256:ba672b26069957ee369cfa7fc180dde1fc6f176eaf1e6beaf61fbebbd3d9c000", size = 42368, upload-time = "2025-10-06T14:51:20.225Z" }, + { url = "https://files.pythonhosted.org/packages/8b/40/cd499bd0dbc5f1136726db3153042a735fffd0d77268e2ee20d5f33c010f/multidict-6.7.0-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:c1dcc7524066fa918c6a27d61444d4ee7900ec635779058571f70d042d86ed63", size = 82326, upload-time = "2025-10-06T14:51:21.588Z" }, + { url = "https://files.pythonhosted.org/packages/13/8a/18e031eca251c8df76daf0288e6790561806e439f5ce99a170b4af30676b/multidict-6.7.0-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:27e0b36c2d388dc7b6ced3406671b401e84ad7eb0656b8f3a2f46ed0ce483718", size = 48065, upload-time = "2025-10-06T14:51:22.93Z" }, + { url = "https://files.pythonhosted.org/packages/40/71/5e6701277470a87d234e433fb0a3a7deaf3bcd92566e421e7ae9776319de/multidict-6.7.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:2a7baa46a22e77f0988e3b23d4ede5513ebec1929e34ee9495be535662c0dfe2", size = 46475, upload-time = "2025-10-06T14:51:24.352Z" }, + { url = "https://files.pythonhosted.org/packages/fe/6a/bab00cbab6d9cfb57afe1663318f72ec28289ea03fd4e8236bb78429893a/multidict-6.7.0-cp314-cp314t-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:7bf77f54997a9166a2f5675d1201520586439424c2511723a7312bdb4bcc034e", size = 239324, upload-time = "2025-10-06T14:51:25.822Z" }, + { url = "https://files.pythonhosted.org/packages/2a/5f/8de95f629fc22a7769ade8b41028e3e5a822c1f8904f618d175945a81ad3/multidict-6.7.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e011555abada53f1578d63389610ac8a5400fc70ce71156b0aa30d326f1a5064", size = 246877, upload-time = "2025-10-06T14:51:27.604Z" }, + { url = "https://files.pythonhosted.org/packages/23/b4/38881a960458f25b89e9f4a4fdcb02ac101cfa710190db6e5528841e67de/multidict-6.7.0-cp314-cp314t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:28b37063541b897fd6a318007373930a75ca6d6ac7c940dbe14731ffdd8d498e", size = 225824, upload-time = "2025-10-06T14:51:29.664Z" }, + { url = "https://files.pythonhosted.org/packages/1e/39/6566210c83f8a261575f18e7144736059f0c460b362e96e9cf797a24b8e7/multidict-6.7.0-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:05047ada7a2fde2631a0ed706f1fd68b169a681dfe5e4cf0f8e4cb6618bbc2cd", size = 253558, upload-time = "2025-10-06T14:51:31.684Z" }, + { url = "https://files.pythonhosted.org/packages/00/a3/67f18315100f64c269f46e6c0319fa87ba68f0f64f2b8e7fd7c72b913a0b/multidict-6.7.0-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:716133f7d1d946a4e1b91b1756b23c088881e70ff180c24e864c26192ad7534a", size = 252339, upload-time = "2025-10-06T14:51:33.699Z" }, + { url = "https://files.pythonhosted.org/packages/c8/2a/1cb77266afee2458d82f50da41beba02159b1d6b1f7973afc9a1cad1499b/multidict-6.7.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d1bed1b467ef657f2a0ae62844a607909ef1c6889562de5e1d505f74457d0b96", size = 244895, upload-time = "2025-10-06T14:51:36.189Z" }, + { url = "https://files.pythonhosted.org/packages/dd/72/09fa7dd487f119b2eb9524946ddd36e2067c08510576d43ff68469563b3b/multidict-6.7.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:ca43bdfa5d37bd6aee89d85e1d0831fb86e25541be7e9d376ead1b28974f8e5e", size = 241862, upload-time = "2025-10-06T14:51:41.291Z" }, + { url = "https://files.pythonhosted.org/packages/65/92/bc1f8bd0853d8669300f732c801974dfc3702c3eeadae2f60cef54dc69d7/multidict-6.7.0-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:44b546bd3eb645fd26fb949e43c02a25a2e632e2ca21a35e2e132c8105dc8599", size = 232376, upload-time = "2025-10-06T14:51:43.55Z" }, + { url = "https://files.pythonhosted.org/packages/09/86/ac39399e5cb9d0c2ac8ef6e10a768e4d3bc933ac808d49c41f9dc23337eb/multidict-6.7.0-cp314-cp314t-musllinux_1_2_i686.whl", hash = "sha256:a6ef16328011d3f468e7ebc326f24c1445f001ca1dec335b2f8e66bed3006394", size = 240272, upload-time = "2025-10-06T14:51:45.265Z" }, + { url = "https://files.pythonhosted.org/packages/3d/b6/fed5ac6b8563ec72df6cb1ea8dac6d17f0a4a1f65045f66b6d3bf1497c02/multidict-6.7.0-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:5aa873cbc8e593d361ae65c68f85faadd755c3295ea2c12040ee146802f23b38", size = 248774, upload-time = "2025-10-06T14:51:46.836Z" }, + { url = "https://files.pythonhosted.org/packages/6b/8d/b954d8c0dc132b68f760aefd45870978deec6818897389dace00fcde32ff/multidict-6.7.0-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:3d7b6ccce016e29df4b7ca819659f516f0bc7a4b3efa3bb2012ba06431b044f9", size = 242731, upload-time = "2025-10-06T14:51:48.541Z" }, + { url = "https://files.pythonhosted.org/packages/16/9d/a2dac7009125d3540c2f54e194829ea18ac53716c61b655d8ed300120b0f/multidict-6.7.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:171b73bd4ee683d307599b66793ac80981b06f069b62eea1c9e29c9241aa66b0", size = 240193, upload-time = "2025-10-06T14:51:50.355Z" }, + { url = "https://files.pythonhosted.org/packages/39/ca/c05f144128ea232ae2178b008d5011d4e2cea86e4ee8c85c2631b1b94802/multidict-6.7.0-cp314-cp314t-win32.whl", hash = "sha256:b2d7f80c4e1fd010b07cb26820aae86b7e73b681ee4889684fb8d2d4537aab13", size = 48023, upload-time = "2025-10-06T14:51:51.883Z" }, + { url = "https://files.pythonhosted.org/packages/ba/8f/0a60e501584145588be1af5cc829265701ba3c35a64aec8e07cbb71d39bb/multidict-6.7.0-cp314-cp314t-win_amd64.whl", hash = "sha256:09929cab6fcb68122776d575e03c6cc64ee0b8fca48d17e135474b042ce515cd", size = 53507, upload-time = "2025-10-06T14:51:53.672Z" }, + { url = "https://files.pythonhosted.org/packages/7f/ae/3148b988a9c6239903e786eac19c889fab607c31d6efa7fb2147e5680f23/multidict-6.7.0-cp314-cp314t-win_arm64.whl", hash = "sha256:cc41db090ed742f32bd2d2c721861725e6109681eddf835d0a82bd3a5c382827", size = 44804, upload-time = "2025-10-06T14:51:55.415Z" }, + { url = "https://files.pythonhosted.org/packages/90/d7/4cf84257902265c4250769ac49f4eaab81c182ee9aff8bf59d2714dbb174/multidict-6.7.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:363eb68a0a59bd2303216d2346e6c441ba10d36d1f9969fcb6f1ba700de7bb5c", size = 77073, upload-time = "2025-10-06T14:51:57.386Z" }, + { url = "https://files.pythonhosted.org/packages/6d/51/194e999630a656e76c2965a1590d12faa5cd528170f2abaa04423e09fe8d/multidict-6.7.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d874eb056410ca05fed180b6642e680373688efafc7f077b2a2f61811e873a40", size = 44928, upload-time = "2025-10-06T14:51:58.791Z" }, + { url = "https://files.pythonhosted.org/packages/e5/6b/2a195373c33068c9158e0941d0b46cfcc9c1d894ca2eb137d1128081dff0/multidict-6.7.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:8b55d5497b51afdfde55925e04a022f1de14d4f4f25cdfd4f5d9b0aa96166851", size = 44581, upload-time = "2025-10-06T14:52:00.174Z" }, + { url = "https://files.pythonhosted.org/packages/69/7b/7f4f2e644b6978bf011a5fd9a5ebb7c21de3f38523b1f7897d36a1ac1311/multidict-6.7.0-cp39-cp39-manylinux1_i686.manylinux_2_28_i686.manylinux_2_5_i686.whl", hash = "sha256:f8e5c0031b90ca9ce555e2e8fd5c3b02a25f14989cbc310701823832c99eb687", size = 239901, upload-time = "2025-10-06T14:52:02.416Z" }, + { url = "https://files.pythonhosted.org/packages/3c/b5/952c72786710a031aa204a9adf7db66d7f97a2c6573889d58b9e60fe6702/multidict-6.7.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9cf41880c991716f3c7cec48e2f19ae4045fc9db5fc9cff27347ada24d710bb5", size = 240534, upload-time = "2025-10-06T14:52:04.105Z" }, + { url = "https://files.pythonhosted.org/packages/f3/ef/109fe1f2471e4c458c74242c7e4a833f2d9fc8a6813cd7ee345b0bad18f9/multidict-6.7.0-cp39-cp39-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:8cfc12a8630a29d601f48d47787bd7eb730e475e83edb5d6c5084317463373eb", size = 219545, upload-time = "2025-10-06T14:52:06.208Z" }, + { url = "https://files.pythonhosted.org/packages/42/bd/327d91288114967f9fe90dc53de70aa3fec1b9073e46aa32c4828f771a87/multidict-6.7.0-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:3996b50c3237c4aec17459217c1e7bbdead9a22a0fcd3c365564fbd16439dde6", size = 251187, upload-time = "2025-10-06T14:52:08.049Z" }, + { url = "https://files.pythonhosted.org/packages/f4/13/a8b078ebbaceb7819fd28cd004413c33b98f1b70d542a62e6a00b74fb09f/multidict-6.7.0-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:7f5170993a0dd3ab871c74f45c0a21a4e2c37a2f2b01b5f722a2ad9c6650469e", size = 249379, upload-time = "2025-10-06T14:52:09.831Z" }, + { url = "https://files.pythonhosted.org/packages/e3/6d/ab12e1246be4d65d1f55de1e6f6aaa9b8120eddcfdd1d290439c7833d5ce/multidict-6.7.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:ec81878ddf0e98817def1e77d4f50dae5ef5b0e4fe796fae3bd674304172416e", size = 239241, upload-time = "2025-10-06T14:52:11.561Z" }, + { url = "https://files.pythonhosted.org/packages/bb/d7/079a93625208c173b8fa756396814397c0fd9fee61ef87b75a748820b86e/multidict-6.7.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:9281bf5b34f59afbc6b1e477a372e9526b66ca446f4bf62592839c195a718b32", size = 237418, upload-time = "2025-10-06T14:52:13.671Z" }, + { url = "https://files.pythonhosted.org/packages/c9/29/03777c2212274aa9440918d604dc9d6af0e6b4558c611c32c3dcf1a13870/multidict-6.7.0-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:68af405971779d8b37198726f2b6fe3955db846fee42db7a4286fc542203934c", size = 232987, upload-time = "2025-10-06T14:52:15.708Z" }, + { url = "https://files.pythonhosted.org/packages/d9/00/11188b68d85a84e8050ee34724d6ded19ad03975caebe0c8dcb2829b37bf/multidict-6.7.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:3ba3ef510467abb0667421a286dc906e30eb08569365f5cdb131d7aff7c2dd84", size = 240985, upload-time = "2025-10-06T14:52:17.317Z" }, + { url = "https://files.pythonhosted.org/packages/df/0c/12eef6aeda21859c6cdf7d75bd5516d83be3efe3d8cc45fd1a3037f5b9dc/multidict-6.7.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:b61189b29081a20c7e4e0b49b44d5d44bb0dc92be3c6d06a11cc043f81bf9329", size = 246855, upload-time = "2025-10-06T14:52:19.096Z" }, + { url = "https://files.pythonhosted.org/packages/69/f6/076120fd8bb3975f09228e288e08bff6b9f1bfd5166397c7ba284f622ab2/multidict-6.7.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:fb287618b9c7aa3bf8d825f02d9201b2f13078a5ed3b293c8f4d953917d84d5e", size = 241804, upload-time = "2025-10-06T14:52:21.166Z" }, + { url = "https://files.pythonhosted.org/packages/5f/51/41bb950c81437b88a93e6ddfca1d8763569ae861e638442838c4375f7497/multidict-6.7.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:521f33e377ff64b96c4c556b81c55d0cfffb96a11c194fd0c3f1e56f3d8dd5a4", size = 235321, upload-time = "2025-10-06T14:52:23.208Z" }, + { url = "https://files.pythonhosted.org/packages/5a/cf/5bbd31f055199d56c1f6b04bbadad3ccb24e6d5d4db75db774fc6d6674b8/multidict-6.7.0-cp39-cp39-win32.whl", hash = "sha256:ce8fdc2dca699f8dbf055a61d73eaa10482569ad20ee3c36ef9641f69afa8c91", size = 41435, upload-time = "2025-10-06T14:52:24.735Z" }, + { url = "https://files.pythonhosted.org/packages/af/01/547ffe9c2faec91c26965c152f3fea6cff068b6037401f61d310cc861ff4/multidict-6.7.0-cp39-cp39-win_amd64.whl", hash = "sha256:7e73299c99939f089dd9b2120a04a516b95cdf8c1cd2b18c53ebf0de80b1f18f", size = 46193, upload-time = "2025-10-06T14:52:26.101Z" }, + { url = "https://files.pythonhosted.org/packages/27/77/cfa5461d1d2651d6fc24216c92b4a21d4e385a41c46e0d9f3b070675167b/multidict-6.7.0-cp39-cp39-win_arm64.whl", hash = "sha256:6bdce131e14b04fd34a809b6380dbfd826065c3e2fe8a50dbae659fa0c390546", size = 43118, upload-time = "2025-10-06T14:52:27.876Z" }, + { url = "https://files.pythonhosted.org/packages/b7/da/7d22601b625e241d4f23ef1ebff8acfc60da633c9e7e7922e24d10f592b3/multidict-6.7.0-py3-none-any.whl", hash = "sha256:394fc5c42a333c9ffc3e421a4c85e08580d990e08b99f6bf35b4132114c5dcb3", size = 12317, upload-time = "2025-10-06T14:52:29.272Z" }, +] + +[[package]] +name = "mypy-extensions" +version = "1.1.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a2/6e/371856a3fb9d31ca8dac321cda606860fa4548858c0cc45d9d1d4ca2628b/mypy_extensions-1.1.0.tar.gz", hash = "sha256:52e68efc3284861e772bbcd66823fde5ae21fd2fdb51c62a211403730b916558", size = 6343, upload-time = "2025-04-22T14:54:24.164Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/79/7b/2c79738432f5c924bef5071f933bcc9efd0473bac3b4aa584a6f7c1c8df8/mypy_extensions-1.1.0-py3-none-any.whl", hash = "sha256:1be4cccdb0f2482337c4743e60421de3a356cd97508abadd57d47403e94f5505", size = 4963, upload-time = "2025-04-22T14:54:22.983Z" }, +] + +[[package]] +name = "packaging" +version = "25.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a1/d4/1fc4078c65507b51b96ca8f8c3ba19e6a61c8253c72794544580a7b6c24d/packaging-25.0.tar.gz", hash = "sha256:d443872c98d677bf60f6a1f2f8c1cb748e8fe762d2bf9d3148b5599295b0fc4f", size = 165727, upload-time = "2025-04-19T11:48:59.673Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/20/12/38679034af332785aac8774540895e234f4d07f7545804097de4b666afd8/packaging-25.0-py3-none-any.whl", hash = "sha256:29572ef2b1f17581046b3a2227d5c611fb25ec70ca1ba8554b24b0e69331a484", size = 66469, upload-time = "2025-04-19T11:48:57.875Z" }, +] + +[[package]] +name = "pathspec" +version = "0.12.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ca/bc/f35b8446f4531a7cb215605d100cd88b7ac6f44ab3fc94870c120ab3adbf/pathspec-0.12.1.tar.gz", hash = "sha256:a482d51503a1ab33b1c67a6c3813a26953dbdc71c31dacaef9a838c4e29f5712", size = 51043, upload-time = "2023-12-10T22:30:45Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/cc/20/ff623b09d963f88bfde16306a54e12ee5ea43e9b597108672ff3a408aad6/pathspec-0.12.1-py3-none-any.whl", hash = "sha256:a0d503e138a4c123b27490a4f7beda6a01c6f288df0e4a8b79c7eb0dc7b4cc08", size = 31191, upload-time = "2023-12-10T22:30:43.14Z" }, +] + +[[package]] +name = "platformdirs" +version = "4.3.6" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/13/fc/128cc9cb8f03208bdbf93d3aa862e16d376844a14f9a0ce5cf4507372de4/platformdirs-4.3.6.tar.gz", hash = "sha256:357fb2acbc885b0419afd3ce3ed34564c13c9b95c89360cd9563f73aa5e2b907", size = 21302, upload-time = "2024-09-17T19:06:50.688Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/3c/a6/bc1012356d8ece4d66dd75c4b9fc6c1f6650ddd5991e421177d9f8f671be/platformdirs-4.3.6-py3-none-any.whl", hash = "sha256:73e575e1408ab8103900836b97580d5307456908a03e92031bab39e4554cc3fb", size = 18439, upload-time = "2024-09-17T19:06:49.212Z" }, +] + +[[package]] +name = "platformdirs" +version = "4.4.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/23/e8/21db9c9987b0e728855bd57bff6984f67952bea55d6f75e055c46b5383e8/platformdirs-4.4.0.tar.gz", hash = "sha256:ca753cf4d81dc309bc67b0ea38fd15dc97bc30ce419a7f58d13eb3bf14c4febf", size = 21634, upload-time = "2025-08-26T14:32:04.268Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/40/4b/2028861e724d3bd36227adfa20d3fd24c3fc6d52032f4a93c133be5d17ce/platformdirs-4.4.0-py3-none-any.whl", hash = "sha256:abd01743f24e5287cd7a5db3752faf1a2d65353f38ec26d98e25a6db65958c85", size = 18654, upload-time = "2025-08-26T14:32:02.735Z" }, +] + +[[package]] +name = "platformdirs" +version = "4.5.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", +] +sdist = { url = "https://files.pythonhosted.org/packages/61/33/9611380c2bdb1225fdef633e2a9610622310fed35ab11dac9620972ee088/platformdirs-4.5.0.tar.gz", hash = "sha256:70ddccdd7c99fc5942e9fc25636a8b34d04c24b335100223152c2803e4063312", size = 21632, upload-time = "2025-10-08T17:44:48.791Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/73/cb/ac7874b3e5d58441674fb70742e6c374b28b0c7cb988d37d991cde47166c/platformdirs-4.5.0-py3-none-any.whl", hash = "sha256:e578a81bb873cbb89a41fcc904c7ef523cc18284b7e3b3ccf06aca1403b7ebd3", size = 18651, upload-time = "2025-10-08T17:44:47.223Z" }, +] + +[[package]] +name = "pluggy" +version = "1.5.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/96/2d/02d4312c973c6050a18b314a5ad0b3210edb65a906f868e31c111dede4a6/pluggy-1.5.0.tar.gz", hash = "sha256:2cffa88e94fdc978c4c574f15f9e59b7f4201d439195c3715ca9e2486f1d0cf1", size = 67955, upload-time = "2024-04-20T21:34:42.531Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/88/5f/e351af9a41f866ac3f1fac4ca0613908d9a41741cfcf2228f4ad853b697d/pluggy-1.5.0-py3-none-any.whl", hash = "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669", size = 20556, upload-time = "2024-04-20T21:34:40.434Z" }, +] + +[[package]] +name = "pluggy" +version = "1.6.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/f9/e2/3e91f31a7d2b083fe6ef3fa267035b518369d9511ffab804f839851d2779/pluggy-1.6.0.tar.gz", hash = "sha256:7dcc130b76258d33b90f61b658791dede3486c3e6bfb003ee5c9bfb396dd22f3", size = 69412, upload-time = "2025-05-15T12:30:07.975Z" } +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 = "propcache" +version = "0.2.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/a9/4d/5e5a60b78dbc1d464f8a7bbaeb30957257afdc8512cbb9dfd5659304f5cd/propcache-0.2.0.tar.gz", hash = "sha256:df81779732feb9d01e5d513fad0122efb3d53bbc75f61b2a4f29a020bc985e70", size = 40951, upload-time = "2024-10-07T12:56:36.896Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/3a/08/1963dfb932b8d74d5b09098507b37e9b96c835ba89ab8aad35aa330f4ff3/propcache-0.2.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:c5869b8fd70b81835a6f187c5fdbe67917a04d7e52b6e7cc4e5fe39d55c39d58", size = 80712, upload-time = "2024-10-07T12:54:02.193Z" }, + { url = "https://files.pythonhosted.org/packages/e6/59/49072aba9bf8a8ed958e576182d46f038e595b17ff7408bc7e8807e721e1/propcache-0.2.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:952e0d9d07609d9c5be361f33b0d6d650cd2bae393aabb11d9b719364521984b", size = 46301, upload-time = "2024-10-07T12:54:03.576Z" }, + { url = "https://files.pythonhosted.org/packages/33/a2/6b1978c2e0d80a678e2c483f45e5443c15fe5d32c483902e92a073314ef1/propcache-0.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:33ac8f098df0585c0b53009f039dfd913b38c1d2edafed0cedcc0c32a05aa110", size = 45581, upload-time = "2024-10-07T12:54:05.415Z" }, + { url = "https://files.pythonhosted.org/packages/43/95/55acc9adff8f997c7572f23d41993042290dfb29e404cdadb07039a4386f/propcache-0.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:97e48e8875e6c13909c800fa344cd54cc4b2b0db1d5f911f840458a500fde2c2", size = 208659, upload-time = "2024-10-07T12:54:06.742Z" }, + { url = "https://files.pythonhosted.org/packages/bd/2c/ef7371ff715e6cd19ea03fdd5637ecefbaa0752fee5b0f2fe8ea8407ee01/propcache-0.2.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:388f3217649d6d59292b722d940d4d2e1e6a7003259eb835724092a1cca0203a", size = 222613, upload-time = "2024-10-07T12:54:08.204Z" }, + { url = "https://files.pythonhosted.org/packages/5e/1c/fef251f79fd4971a413fa4b1ae369ee07727b4cc2c71e2d90dfcde664fbb/propcache-0.2.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f571aea50ba5623c308aa146eb650eebf7dbe0fd8c5d946e28343cb3b5aad577", size = 221067, upload-time = "2024-10-07T12:54:10.449Z" }, + { url = "https://files.pythonhosted.org/packages/8d/e7/22e76ae6fc5a1708bdce92bdb49de5ebe89a173db87e4ef597d6bbe9145a/propcache-0.2.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3dfafb44f7bb35c0c06eda6b2ab4bfd58f02729e7c4045e179f9a861b07c9850", size = 208920, upload-time = "2024-10-07T12:54:11.903Z" }, + { url = "https://files.pythonhosted.org/packages/04/3e/f10aa562781bcd8a1e0b37683a23bef32bdbe501d9cc7e76969becaac30d/propcache-0.2.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a3ebe9a75be7ab0b7da2464a77bb27febcb4fab46a34f9288f39d74833db7f61", size = 200050, upload-time = "2024-10-07T12:54:13.292Z" }, + { url = "https://files.pythonhosted.org/packages/d0/98/8ac69f638358c5f2a0043809c917802f96f86026e86726b65006830f3dc6/propcache-0.2.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:d2f0d0f976985f85dfb5f3d685697ef769faa6b71993b46b295cdbbd6be8cc37", size = 202346, upload-time = "2024-10-07T12:54:14.644Z" }, + { url = "https://files.pythonhosted.org/packages/ee/78/4acfc5544a5075d8e660af4d4e468d60c418bba93203d1363848444511ad/propcache-0.2.0-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:a3dc1a4b165283bd865e8f8cb5f0c64c05001e0718ed06250d8cac9bec115b48", size = 199750, upload-time = "2024-10-07T12:54:16.286Z" }, + { url = "https://files.pythonhosted.org/packages/a2/8f/90ada38448ca2e9cf25adc2fe05d08358bda1b9446f54a606ea38f41798b/propcache-0.2.0-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:9e0f07b42d2a50c7dd2d8675d50f7343d998c64008f1da5fef888396b7f84630", size = 201279, upload-time = "2024-10-07T12:54:17.752Z" }, + { url = "https://files.pythonhosted.org/packages/08/31/0e299f650f73903da851f50f576ef09bfffc8e1519e6a2f1e5ed2d19c591/propcache-0.2.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:e63e3e1e0271f374ed489ff5ee73d4b6e7c60710e1f76af5f0e1a6117cd26394", size = 211035, upload-time = "2024-10-07T12:54:19.109Z" }, + { url = "https://files.pythonhosted.org/packages/85/3e/e356cc6b09064bff1c06d0b2413593e7c925726f0139bc7acef8a21e87a8/propcache-0.2.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:56bb5c98f058a41bb58eead194b4db8c05b088c93d94d5161728515bd52b052b", size = 215565, upload-time = "2024-10-07T12:54:20.578Z" }, + { url = "https://files.pythonhosted.org/packages/8b/54/4ef7236cd657e53098bd05aa59cbc3cbf7018fba37b40eaed112c3921e51/propcache-0.2.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:7665f04d0c7f26ff8bb534e1c65068409bf4687aa2534faf7104d7182debb336", size = 207604, upload-time = "2024-10-07T12:54:22.588Z" }, + { url = "https://files.pythonhosted.org/packages/1f/27/d01d7799c068443ee64002f0655d82fb067496897bf74b632e28ee6a32cf/propcache-0.2.0-cp310-cp310-win32.whl", hash = "sha256:7cf18abf9764746b9c8704774d8b06714bcb0a63641518a3a89c7f85cc02c2ad", size = 40526, upload-time = "2024-10-07T12:54:23.867Z" }, + { url = "https://files.pythonhosted.org/packages/bb/44/6c2add5eeafb7f31ff0d25fbc005d930bea040a1364cf0f5768750ddf4d1/propcache-0.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:cfac69017ef97db2438efb854edf24f5a29fd09a536ff3a992b75990720cdc99", size = 44958, upload-time = "2024-10-07T12:54:24.983Z" }, + { url = "https://files.pythonhosted.org/packages/e0/1c/71eec730e12aec6511e702ad0cd73c2872eccb7cad39de8ba3ba9de693ef/propcache-0.2.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:63f13bf09cc3336eb04a837490b8f332e0db41da66995c9fd1ba04552e516354", size = 80811, upload-time = "2024-10-07T12:54:26.165Z" }, + { url = "https://files.pythonhosted.org/packages/89/c3/7e94009f9a4934c48a371632197406a8860b9f08e3f7f7d922ab69e57a41/propcache-0.2.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:608cce1da6f2672a56b24a015b42db4ac612ee709f3d29f27a00c943d9e851de", size = 46365, upload-time = "2024-10-07T12:54:28.034Z" }, + { url = "https://files.pythonhosted.org/packages/c0/1d/c700d16d1d6903aeab28372fe9999762f074b80b96a0ccc953175b858743/propcache-0.2.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:466c219deee4536fbc83c08d09115249db301550625c7fef1c5563a584c9bc87", size = 45602, upload-time = "2024-10-07T12:54:29.148Z" }, + { url = "https://files.pythonhosted.org/packages/2e/5e/4a3e96380805bf742712e39a4534689f4cddf5fa2d3a93f22e9fd8001b23/propcache-0.2.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fc2db02409338bf36590aa985a461b2c96fce91f8e7e0f14c50c5fcc4f229016", size = 236161, upload-time = "2024-10-07T12:54:31.557Z" }, + { url = "https://files.pythonhosted.org/packages/a5/85/90132481183d1436dff6e29f4fa81b891afb6cb89a7306f32ac500a25932/propcache-0.2.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a6ed8db0a556343d566a5c124ee483ae113acc9a557a807d439bcecc44e7dfbb", size = 244938, upload-time = "2024-10-07T12:54:33.051Z" }, + { url = "https://files.pythonhosted.org/packages/4a/89/c893533cb45c79c970834274e2d0f6d64383ec740be631b6a0a1d2b4ddc0/propcache-0.2.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:91997d9cb4a325b60d4e3f20967f8eb08dfcb32b22554d5ef78e6fd1dda743a2", size = 243576, upload-time = "2024-10-07T12:54:34.497Z" }, + { url = "https://files.pythonhosted.org/packages/8c/56/98c2054c8526331a05f205bf45cbb2cda4e58e56df70e76d6a509e5d6ec6/propcache-0.2.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4c7dde9e533c0a49d802b4f3f218fa9ad0a1ce21f2c2eb80d5216565202acab4", size = 236011, upload-time = "2024-10-07T12:54:35.903Z" }, + { url = "https://files.pythonhosted.org/packages/2d/0c/8b8b9f8a6e1abd869c0fa79b907228e7abb966919047d294ef5df0d136cf/propcache-0.2.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffcad6c564fe6b9b8916c1aefbb37a362deebf9394bd2974e9d84232e3e08504", size = 224834, upload-time = "2024-10-07T12:54:37.238Z" }, + { url = "https://files.pythonhosted.org/packages/18/bb/397d05a7298b7711b90e13108db697732325cafdcd8484c894885c1bf109/propcache-0.2.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:97a58a28bcf63284e8b4d7b460cbee1edaab24634e82059c7b8c09e65284f178", size = 224946, upload-time = "2024-10-07T12:54:38.72Z" }, + { url = "https://files.pythonhosted.org/packages/25/19/4fc08dac19297ac58135c03770b42377be211622fd0147f015f78d47cd31/propcache-0.2.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:945db8ee295d3af9dbdbb698cce9bbc5c59b5c3fe328bbc4387f59a8a35f998d", size = 217280, upload-time = "2024-10-07T12:54:40.089Z" }, + { url = "https://files.pythonhosted.org/packages/7e/76/c79276a43df2096ce2aba07ce47576832b1174c0c480fe6b04bd70120e59/propcache-0.2.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:39e104da444a34830751715f45ef9fc537475ba21b7f1f5b0f4d71a3b60d7fe2", size = 220088, upload-time = "2024-10-07T12:54:41.726Z" }, + { url = "https://files.pythonhosted.org/packages/c3/9a/8a8cf428a91b1336b883f09c8b884e1734c87f724d74b917129a24fe2093/propcache-0.2.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:c5ecca8f9bab618340c8e848d340baf68bcd8ad90a8ecd7a4524a81c1764b3db", size = 233008, upload-time = "2024-10-07T12:54:43.742Z" }, + { url = "https://files.pythonhosted.org/packages/25/7b/768a8969abd447d5f0f3333df85c6a5d94982a1bc9a89c53c154bf7a8b11/propcache-0.2.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:c436130cc779806bdf5d5fae0d848713105472b8566b75ff70048c47d3961c5b", size = 237719, upload-time = "2024-10-07T12:54:45.065Z" }, + { url = "https://files.pythonhosted.org/packages/ed/0d/e5d68ccc7976ef8b57d80613ac07bbaf0614d43f4750cf953f0168ef114f/propcache-0.2.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:191db28dc6dcd29d1a3e063c3be0b40688ed76434622c53a284e5427565bbd9b", size = 227729, upload-time = "2024-10-07T12:54:46.405Z" }, + { url = "https://files.pythonhosted.org/packages/05/64/17eb2796e2d1c3d0c431dc5f40078d7282f4645af0bb4da9097fbb628c6c/propcache-0.2.0-cp311-cp311-win32.whl", hash = "sha256:5f2564ec89058ee7c7989a7b719115bdfe2a2fb8e7a4543b8d1c0cc4cf6478c1", size = 40473, upload-time = "2024-10-07T12:54:47.694Z" }, + { url = "https://files.pythonhosted.org/packages/83/c5/e89fc428ccdc897ade08cd7605f174c69390147526627a7650fb883e0cd0/propcache-0.2.0-cp311-cp311-win_amd64.whl", hash = "sha256:6e2e54267980349b723cff366d1e29b138b9a60fa376664a157a342689553f71", size = 44921, upload-time = "2024-10-07T12:54:48.935Z" }, + { url = "https://files.pythonhosted.org/packages/7c/46/a41ca1097769fc548fc9216ec4c1471b772cc39720eb47ed7e38ef0006a9/propcache-0.2.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:2ee7606193fb267be4b2e3b32714f2d58cad27217638db98a60f9efb5efeccc2", size = 80800, upload-time = "2024-10-07T12:54:50.409Z" }, + { url = "https://files.pythonhosted.org/packages/75/4f/93df46aab9cc473498ff56be39b5f6ee1e33529223d7a4d8c0a6101a9ba2/propcache-0.2.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:91ee8fc02ca52e24bcb77b234f22afc03288e1dafbb1f88fe24db308910c4ac7", size = 46443, upload-time = "2024-10-07T12:54:51.634Z" }, + { url = "https://files.pythonhosted.org/packages/0b/17/308acc6aee65d0f9a8375e36c4807ac6605d1f38074b1581bd4042b9fb37/propcache-0.2.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2e900bad2a8456d00a113cad8c13343f3b1f327534e3589acc2219729237a2e8", size = 45676, upload-time = "2024-10-07T12:54:53.454Z" }, + { url = "https://files.pythonhosted.org/packages/65/44/626599d2854d6c1d4530b9a05e7ff2ee22b790358334b475ed7c89f7d625/propcache-0.2.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f52a68c21363c45297aca15561812d542f8fc683c85201df0bebe209e349f793", size = 246191, upload-time = "2024-10-07T12:54:55.438Z" }, + { url = "https://files.pythonhosted.org/packages/f2/df/5d996d7cb18df076debae7d76ac3da085c0575a9f2be6b1f707fe227b54c/propcache-0.2.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1e41d67757ff4fbc8ef2af99b338bfb955010444b92929e9e55a6d4dcc3c4f09", size = 251791, upload-time = "2024-10-07T12:54:57.441Z" }, + { url = "https://files.pythonhosted.org/packages/2e/6d/9f91e5dde8b1f662f6dd4dff36098ed22a1ef4e08e1316f05f4758f1576c/propcache-0.2.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a64e32f8bd94c105cc27f42d3b658902b5bcc947ece3c8fe7bc1b05982f60e89", size = 253434, upload-time = "2024-10-07T12:54:58.857Z" }, + { url = "https://files.pythonhosted.org/packages/3c/e9/1b54b7e26f50b3e0497cd13d3483d781d284452c2c50dd2a615a92a087a3/propcache-0.2.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:55346705687dbd7ef0d77883ab4f6fabc48232f587925bdaf95219bae072491e", size = 248150, upload-time = "2024-10-07T12:55:00.19Z" }, + { url = "https://files.pythonhosted.org/packages/a7/ef/a35bf191c8038fe3ce9a414b907371c81d102384eda5dbafe6f4dce0cf9b/propcache-0.2.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:00181262b17e517df2cd85656fcd6b4e70946fe62cd625b9d74ac9977b64d8d9", size = 233568, upload-time = "2024-10-07T12:55:01.723Z" }, + { url = "https://files.pythonhosted.org/packages/97/d9/d00bb9277a9165a5e6d60f2142cd1a38a750045c9c12e47ae087f686d781/propcache-0.2.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6994984550eaf25dd7fc7bd1b700ff45c894149341725bb4edc67f0ffa94efa4", size = 229874, upload-time = "2024-10-07T12:55:03.962Z" }, + { url = "https://files.pythonhosted.org/packages/8e/78/c123cf22469bdc4b18efb78893e69c70a8b16de88e6160b69ca6bdd88b5d/propcache-0.2.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:56295eb1e5f3aecd516d91b00cfd8bf3a13991de5a479df9e27dd569ea23959c", size = 225857, upload-time = "2024-10-07T12:55:06.439Z" }, + { url = "https://files.pythonhosted.org/packages/31/1b/fd6b2f1f36d028820d35475be78859d8c89c8f091ad30e377ac49fd66359/propcache-0.2.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:439e76255daa0f8151d3cb325f6dd4a3e93043e6403e6491813bcaaaa8733887", size = 227604, upload-time = "2024-10-07T12:55:08.254Z" }, + { url = "https://files.pythonhosted.org/packages/99/36/b07be976edf77a07233ba712e53262937625af02154353171716894a86a6/propcache-0.2.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:f6475a1b2ecb310c98c28d271a30df74f9dd436ee46d09236a6b750a7599ce57", size = 238430, upload-time = "2024-10-07T12:55:09.766Z" }, + { url = "https://files.pythonhosted.org/packages/0d/64/5822f496c9010e3966e934a011ac08cac8734561842bc7c1f65586e0683c/propcache-0.2.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:3444cdba6628accf384e349014084b1cacd866fbb88433cd9d279d90a54e0b23", size = 244814, upload-time = "2024-10-07T12:55:11.145Z" }, + { url = "https://files.pythonhosted.org/packages/fd/bd/8657918a35d50b18a9e4d78a5df7b6c82a637a311ab20851eef4326305c1/propcache-0.2.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:4a9d9b4d0a9b38d1c391bb4ad24aa65f306c6f01b512e10a8a34a2dc5675d348", size = 235922, upload-time = "2024-10-07T12:55:12.508Z" }, + { url = "https://files.pythonhosted.org/packages/a8/6f/ec0095e1647b4727db945213a9f395b1103c442ef65e54c62e92a72a3f75/propcache-0.2.0-cp312-cp312-win32.whl", hash = "sha256:69d3a98eebae99a420d4b28756c8ce6ea5a29291baf2dc9ff9414b42676f61d5", size = 40177, upload-time = "2024-10-07T12:55:13.814Z" }, + { url = "https://files.pythonhosted.org/packages/20/a2/bd0896fdc4f4c1db46d9bc361c8c79a9bf08ccc08ba054a98e38e7ba1557/propcache-0.2.0-cp312-cp312-win_amd64.whl", hash = "sha256:ad9c9b99b05f163109466638bd30ada1722abb01bbb85c739c50b6dc11f92dc3", size = 44446, upload-time = "2024-10-07T12:55:14.972Z" }, + { url = "https://files.pythonhosted.org/packages/a8/a7/5f37b69197d4f558bfef5b4bceaff7c43cc9b51adf5bd75e9081d7ea80e4/propcache-0.2.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:ecddc221a077a8132cf7c747d5352a15ed763b674c0448d811f408bf803d9ad7", size = 78120, upload-time = "2024-10-07T12:55:16.179Z" }, + { url = "https://files.pythonhosted.org/packages/c8/cd/48ab2b30a6b353ecb95a244915f85756d74f815862eb2ecc7a518d565b48/propcache-0.2.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:0e53cb83fdd61cbd67202735e6a6687a7b491c8742dfc39c9e01e80354956763", size = 45127, upload-time = "2024-10-07T12:55:18.275Z" }, + { url = "https://files.pythonhosted.org/packages/a5/ba/0a1ef94a3412aab057bd996ed5f0ac7458be5bf469e85c70fa9ceb43290b/propcache-0.2.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:92fe151145a990c22cbccf9ae15cae8ae9eddabfc949a219c9f667877e40853d", size = 44419, upload-time = "2024-10-07T12:55:19.487Z" }, + { url = "https://files.pythonhosted.org/packages/b4/6c/ca70bee4f22fa99eacd04f4d2f1699be9d13538ccf22b3169a61c60a27fa/propcache-0.2.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d6a21ef516d36909931a2967621eecb256018aeb11fc48656e3257e73e2e247a", size = 229611, upload-time = "2024-10-07T12:55:21.377Z" }, + { url = "https://files.pythonhosted.org/packages/19/70/47b872a263e8511ca33718d96a10c17d3c853aefadeb86dc26e8421184b9/propcache-0.2.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f88a4095e913f98988f5b338c1d4d5d07dbb0b6bad19892fd447484e483ba6b", size = 234005, upload-time = "2024-10-07T12:55:22.898Z" }, + { url = "https://files.pythonhosted.org/packages/4f/be/3b0ab8c84a22e4a3224719099c1229ddfdd8a6a1558cf75cb55ee1e35c25/propcache-0.2.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5a5b3bb545ead161be780ee85a2b54fdf7092815995661947812dde94a40f6fb", size = 237270, upload-time = "2024-10-07T12:55:24.354Z" }, + { url = "https://files.pythonhosted.org/packages/04/d8/f071bb000d4b8f851d312c3c75701e586b3f643fe14a2e3409b1b9ab3936/propcache-0.2.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:67aeb72e0f482709991aa91345a831d0b707d16b0257e8ef88a2ad246a7280bf", size = 231877, upload-time = "2024-10-07T12:55:25.774Z" }, + { url = "https://files.pythonhosted.org/packages/93/e7/57a035a1359e542bbb0a7df95aad6b9871ebee6dce2840cb157a415bd1f3/propcache-0.2.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3c997f8c44ec9b9b0bcbf2d422cc00a1d9b9c681f56efa6ca149a941e5560da2", size = 217848, upload-time = "2024-10-07T12:55:27.148Z" }, + { url = "https://files.pythonhosted.org/packages/f0/93/d1dea40f112ec183398fb6c42fde340edd7bab202411c4aa1a8289f461b6/propcache-0.2.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:2a66df3d4992bc1d725b9aa803e8c5a66c010c65c741ad901e260ece77f58d2f", size = 216987, upload-time = "2024-10-07T12:55:29.294Z" }, + { url = "https://files.pythonhosted.org/packages/62/4c/877340871251145d3522c2b5d25c16a1690ad655fbab7bb9ece6b117e39f/propcache-0.2.0-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:3ebbcf2a07621f29638799828b8d8668c421bfb94c6cb04269130d8de4fb7136", size = 212451, upload-time = "2024-10-07T12:55:30.643Z" }, + { url = "https://files.pythonhosted.org/packages/7c/bb/a91b72efeeb42906ef58ccf0cdb87947b54d7475fee3c93425d732f16a61/propcache-0.2.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:1235c01ddaa80da8235741e80815ce381c5267f96cc49b1477fdcf8c047ef325", size = 212879, upload-time = "2024-10-07T12:55:32.024Z" }, + { url = "https://files.pythonhosted.org/packages/9b/7f/ee7fea8faac57b3ec5d91ff47470c6c5d40d7f15d0b1fccac806348fa59e/propcache-0.2.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:3947483a381259c06921612550867b37d22e1df6d6d7e8361264b6d037595f44", size = 222288, upload-time = "2024-10-07T12:55:33.401Z" }, + { url = "https://files.pythonhosted.org/packages/ff/d7/acd67901c43d2e6b20a7a973d9d5fd543c6e277af29b1eb0e1f7bd7ca7d2/propcache-0.2.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:d5bed7f9805cc29c780f3aee05de3262ee7ce1f47083cfe9f77471e9d6777e83", size = 228257, upload-time = "2024-10-07T12:55:35.381Z" }, + { url = "https://files.pythonhosted.org/packages/8d/6f/6272ecc7a8daad1d0754cfc6c8846076a8cb13f810005c79b15ce0ef0cf2/propcache-0.2.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:e4a91d44379f45f5e540971d41e4626dacd7f01004826a18cb048e7da7e96544", size = 221075, upload-time = "2024-10-07T12:55:36.789Z" }, + { url = "https://files.pythonhosted.org/packages/7c/bd/c7a6a719a6b3dd8b3aeadb3675b5783983529e4a3185946aa444d3e078f6/propcache-0.2.0-cp313-cp313-win32.whl", hash = "sha256:f902804113e032e2cdf8c71015651c97af6418363bea8d78dc0911d56c335032", size = 39654, upload-time = "2024-10-07T12:55:38.762Z" }, + { url = "https://files.pythonhosted.org/packages/88/e7/0eef39eff84fa3e001b44de0bd41c7c0e3432e7648ffd3d64955910f002d/propcache-0.2.0-cp313-cp313-win_amd64.whl", hash = "sha256:8f188cfcc64fb1266f4684206c9de0e80f54622c3f22a910cbd200478aeae61e", size = 43705, upload-time = "2024-10-07T12:55:39.921Z" }, + { url = "https://files.pythonhosted.org/packages/b4/94/2c3d64420fd58ed462e2b416386d48e72dec027cf7bb572066cf3866e939/propcache-0.2.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:53d1bd3f979ed529f0805dd35ddaca330f80a9a6d90bc0121d2ff398f8ed8861", size = 82315, upload-time = "2024-10-07T12:55:41.166Z" }, + { url = "https://files.pythonhosted.org/packages/73/b7/9e2a17d9a126f2012b22ddc5d0979c28ca75104e24945214790c1d787015/propcache-0.2.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:83928404adf8fb3d26793665633ea79b7361efa0287dfbd372a7e74311d51ee6", size = 47188, upload-time = "2024-10-07T12:55:42.316Z" }, + { url = "https://files.pythonhosted.org/packages/80/ef/18af27caaae5589c08bb5a461cfa136b83b7e7983be604f2140d91f92b97/propcache-0.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:77a86c261679ea5f3896ec060be9dc8e365788248cc1e049632a1be682442063", size = 46314, upload-time = "2024-10-07T12:55:43.544Z" }, + { url = "https://files.pythonhosted.org/packages/fa/df/8dbd3e472baf73251c0fbb571a3f0a4e3a40c52a1c8c2a6c46ab08736ff9/propcache-0.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:218db2a3c297a3768c11a34812e63b3ac1c3234c3a086def9c0fee50d35add1f", size = 212874, upload-time = "2024-10-07T12:55:44.823Z" }, + { url = "https://files.pythonhosted.org/packages/7c/57/5d4d783ac594bd56434679b8643673ae12de1ce758116fd8912a7f2313ec/propcache-0.2.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7735e82e3498c27bcb2d17cb65d62c14f1100b71723b68362872bca7d0913d90", size = 224578, upload-time = "2024-10-07T12:55:46.253Z" }, + { url = "https://files.pythonhosted.org/packages/66/27/072be8ad434c9a3aa1b561f527984ea0ed4ac072fd18dfaaa2aa2d6e6a2b/propcache-0.2.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:20a617c776f520c3875cf4511e0d1db847a076d720714ae35ffe0df3e440be68", size = 222636, upload-time = "2024-10-07T12:55:47.608Z" }, + { url = "https://files.pythonhosted.org/packages/c3/f1/69a30ff0928d07f50bdc6f0147fd9a08e80904fd3fdb711785e518de1021/propcache-0.2.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:67b69535c870670c9f9b14a75d28baa32221d06f6b6fa6f77a0a13c5a7b0a5b9", size = 213573, upload-time = "2024-10-07T12:55:49.82Z" }, + { url = "https://files.pythonhosted.org/packages/a8/2e/c16716ae113fe0a3219978df3665a6fea049d81d50bd28c4ae72a4c77567/propcache-0.2.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4569158070180c3855e9c0791c56be3ceeb192defa2cdf6a3f39e54319e56b89", size = 205438, upload-time = "2024-10-07T12:55:51.231Z" }, + { url = "https://files.pythonhosted.org/packages/e1/df/80e2c5cd5ed56a7bfb1aa58cedb79617a152ae43de7c0a7e800944a6b2e2/propcache-0.2.0-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:db47514ffdbd91ccdc7e6f8407aac4ee94cc871b15b577c1c324236b013ddd04", size = 202352, upload-time = "2024-10-07T12:55:52.596Z" }, + { url = "https://files.pythonhosted.org/packages/0f/4e/79f665fa04839f30ffb2903211c718b9660fbb938ac7a4df79525af5aeb3/propcache-0.2.0-cp38-cp38-musllinux_1_2_armv7l.whl", hash = "sha256:2a60ad3e2553a74168d275a0ef35e8c0a965448ffbc3b300ab3a5bb9956c2162", size = 200476, upload-time = "2024-10-07T12:55:54.016Z" }, + { url = "https://files.pythonhosted.org/packages/a9/39/b9ea7b011521dd7cfd2f89bb6b8b304f3c789ea6285445bc145bebc83094/propcache-0.2.0-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:662dd62358bdeaca0aee5761de8727cfd6861432e3bb828dc2a693aa0471a563", size = 201581, upload-time = "2024-10-07T12:55:56.246Z" }, + { url = "https://files.pythonhosted.org/packages/e4/81/e8e96c97aa0b675a14e37b12ca9c9713b15cfacf0869e64bf3ab389fabf1/propcache-0.2.0-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:25a1f88b471b3bc911d18b935ecb7115dff3a192b6fef46f0bfaf71ff4f12418", size = 225628, upload-time = "2024-10-07T12:55:57.686Z" }, + { url = "https://files.pythonhosted.org/packages/eb/99/15f998c502c214f6c7f51462937605d514a8943a9a6c1fa10f40d2710976/propcache-0.2.0-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:f60f0ac7005b9f5a6091009b09a419ace1610e163fa5deaba5ce3484341840e7", size = 229270, upload-time = "2024-10-07T12:55:59.065Z" }, + { url = "https://files.pythonhosted.org/packages/ff/3a/a9f1a0c0e5b994b8f1a1c71bea56bb3e9eeec821cb4dd61e14051c4ba00b/propcache-0.2.0-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:74acd6e291f885678631b7ebc85d2d4aec458dd849b8c841b57ef04047833bed", size = 207771, upload-time = "2024-10-07T12:56:00.393Z" }, + { url = "https://files.pythonhosted.org/packages/ff/3e/6103906a66d6713f32880cf6a5ba84a1406b4d66e1b9389bb9b8e1789f9e/propcache-0.2.0-cp38-cp38-win32.whl", hash = "sha256:d9b6ddac6408194e934002a69bcaadbc88c10b5f38fb9307779d1c629181815d", size = 41015, upload-time = "2024-10-07T12:56:01.953Z" }, + { url = "https://files.pythonhosted.org/packages/37/23/a30214b4c1f2bea24cc1197ef48d67824fbc41d5cf5472b17c37fef6002c/propcache-0.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:676135dcf3262c9c5081cc8f19ad55c8a64e3f7282a21266d05544450bffc3a5", size = 45749, upload-time = "2024-10-07T12:56:03.095Z" }, + { url = "https://files.pythonhosted.org/packages/38/05/797e6738c9f44ab5039e3ff329540c934eabbe8ad7e63c305c75844bc86f/propcache-0.2.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:25c8d773a62ce0451b020c7b29a35cfbc05de8b291163a7a0f3b7904f27253e6", size = 81903, upload-time = "2024-10-07T12:56:04.651Z" }, + { url = "https://files.pythonhosted.org/packages/9f/84/8d5edb9a73e1a56b24dd8f2adb6aac223109ff0e8002313d52e5518258ba/propcache-0.2.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:375a12d7556d462dc64d70475a9ee5982465fbb3d2b364f16b86ba9135793638", size = 46960, upload-time = "2024-10-07T12:56:06.38Z" }, + { url = "https://files.pythonhosted.org/packages/e7/77/388697bedda984af0d12d68e536b98129b167282da3401965c8450de510e/propcache-0.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1ec43d76b9677637a89d6ab86e1fef70d739217fefa208c65352ecf0282be957", size = 46133, upload-time = "2024-10-07T12:56:07.606Z" }, + { url = "https://files.pythonhosted.org/packages/e2/dc/60d444610bc5b1d7a758534f58362b1bcee736a785473f8a39c91f05aad1/propcache-0.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f45eec587dafd4b2d41ac189c2156461ebd0c1082d2fe7013571598abb8505d1", size = 211105, upload-time = "2024-10-07T12:56:08.826Z" }, + { url = "https://files.pythonhosted.org/packages/bc/c6/40eb0dd1de6f8e84f454615ab61f68eb4a58f9d63d6f6eaf04300ac0cc17/propcache-0.2.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bc092ba439d91df90aea38168e11f75c655880c12782facf5cf9c00f3d42b562", size = 226613, upload-time = "2024-10-07T12:56:11.184Z" }, + { url = "https://files.pythonhosted.org/packages/de/b6/e078b5e9de58e20db12135eb6a206b4b43cb26c6b62ee0fe36ac40763a64/propcache-0.2.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fa1076244f54bb76e65e22cb6910365779d5c3d71d1f18b275f1dfc7b0d71b4d", size = 225587, upload-time = "2024-10-07T12:56:15.294Z" }, + { url = "https://files.pythonhosted.org/packages/ce/4e/97059dd24494d1c93d1efb98bb24825e1930265b41858dd59c15cb37a975/propcache-0.2.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:682a7c79a2fbf40f5dbb1eb6bfe2cd865376deeac65acf9beb607505dced9e12", size = 211826, upload-time = "2024-10-07T12:56:16.997Z" }, + { url = "https://files.pythonhosted.org/packages/fc/23/4dbf726602a989d2280fe130a9b9dd71faa8d3bb8cd23d3261ff3c23f692/propcache-0.2.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:8e40876731f99b6f3c897b66b803c9e1c07a989b366c6b5b475fafd1f7ba3fb8", size = 203140, upload-time = "2024-10-07T12:56:18.368Z" }, + { url = "https://files.pythonhosted.org/packages/5b/ce/f3bff82c885dbd9ae9e43f134d5b02516c3daa52d46f7a50e4f52ef9121f/propcache-0.2.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:363ea8cd3c5cb6679f1c2f5f1f9669587361c062e4899fce56758efa928728f8", size = 208841, upload-time = "2024-10-07T12:56:19.859Z" }, + { url = "https://files.pythonhosted.org/packages/29/d7/19a4d3b4c7e95d08f216da97035d0b103d0c90411c6f739d47088d2da1f0/propcache-0.2.0-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:140fbf08ab3588b3468932974a9331aff43c0ab8a2ec2c608b6d7d1756dbb6cb", size = 203315, upload-time = "2024-10-07T12:56:21.256Z" }, + { url = "https://files.pythonhosted.org/packages/db/87/5748212a18beb8d4ab46315c55ade8960d1e2cdc190764985b2d229dd3f4/propcache-0.2.0-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:e70fac33e8b4ac63dfc4c956fd7d85a0b1139adcfc0d964ce288b7c527537fea", size = 204724, upload-time = "2024-10-07T12:56:23.644Z" }, + { url = "https://files.pythonhosted.org/packages/84/2a/c3d2f989fc571a5bad0fabcd970669ccb08c8f9b07b037ecddbdab16a040/propcache-0.2.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:b33d7a286c0dc1a15f5fc864cc48ae92a846df287ceac2dd499926c3801054a6", size = 215514, upload-time = "2024-10-07T12:56:25.733Z" }, + { url = "https://files.pythonhosted.org/packages/c9/1f/4c44c133b08bc5f776afcb8f0833889c2636b8a83e07ea1d9096c1e401b0/propcache-0.2.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:f6d5749fdd33d90e34c2efb174c7e236829147a2713334d708746e94c4bde40d", size = 220063, upload-time = "2024-10-07T12:56:28.497Z" }, + { url = "https://files.pythonhosted.org/packages/2e/25/280d0a3bdaee68db74c0acd9a472e59e64b516735b59cffd3a326ff9058a/propcache-0.2.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:22aa8f2272d81d9317ff5756bb108021a056805ce63dd3630e27d042c8092798", size = 211620, upload-time = "2024-10-07T12:56:29.891Z" }, + { url = "https://files.pythonhosted.org/packages/28/8c/266898981b7883c1563c35954f9ce9ced06019fdcc487a9520150c48dc91/propcache-0.2.0-cp39-cp39-win32.whl", hash = "sha256:73e4b40ea0eda421b115248d7e79b59214411109a5bc47d0d48e4c73e3b8fcf9", size = 41049, upload-time = "2024-10-07T12:56:31.246Z" }, + { url = "https://files.pythonhosted.org/packages/af/53/a3e5b937f58e757a940716b88105ec4c211c42790c1ea17052b46dc16f16/propcache-0.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:9517d5e9e0731957468c29dbfd0f976736a0e55afaea843726e887f36fe017df", size = 45587, upload-time = "2024-10-07T12:56:33.416Z" }, + { url = "https://files.pythonhosted.org/packages/3d/b6/e6d98278f2d49b22b4d033c9f792eda783b9ab2094b041f013fc69bcde87/propcache-0.2.0-py3-none-any.whl", hash = "sha256:2ccc28197af5313706511fab3a8b66dcd6da067a1331372c82ea1cb74285e036", size = 11603, upload-time = "2024-10-07T12:56:35.137Z" }, +] + +[[package]] +name = "propcache" +version = "0.4.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/9e/da/e9fc233cf63743258bff22b3dfa7ea5baef7b5bc324af47a0ad89b8ffc6f/propcache-0.4.1.tar.gz", hash = "sha256:f48107a8c637e80362555f37ecf49abe20370e557cc4ab374f04ec4423c97c3d", size = 46442, upload-time = "2025-10-08T19:49:02.291Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/3c/0e/934b541323035566a9af292dba85a195f7b78179114f2c6ebb24551118a9/propcache-0.4.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:7c2d1fa3201efaf55d730400d945b5b3ab6e672e100ba0f9a409d950ab25d7db", size = 79534, upload-time = "2025-10-08T19:46:02.083Z" }, + { url = "https://files.pythonhosted.org/packages/a1/6b/db0d03d96726d995dc7171286c6ba9d8d14251f37433890f88368951a44e/propcache-0.4.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:1eb2994229cc8ce7fe9b3db88f5465f5fd8651672840b2e426b88cdb1a30aac8", size = 45526, upload-time = "2025-10-08T19:46:03.884Z" }, + { url = "https://files.pythonhosted.org/packages/e4/c3/82728404aea669e1600f304f2609cde9e665c18df5a11cdd57ed73c1dceb/propcache-0.4.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:66c1f011f45a3b33d7bcb22daed4b29c0c9e2224758b6be00686731e1b46f925", size = 47263, upload-time = "2025-10-08T19:46:05.405Z" }, + { url = "https://files.pythonhosted.org/packages/df/1b/39313ddad2bf9187a1432654c38249bab4562ef535ef07f5eb6eb04d0b1b/propcache-0.4.1-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9a52009f2adffe195d0b605c25ec929d26b36ef986ba85244891dee3b294df21", size = 201012, upload-time = "2025-10-08T19:46:07.165Z" }, + { url = "https://files.pythonhosted.org/packages/5b/01/f1d0b57d136f294a142acf97f4ed58c8e5b974c21e543000968357115011/propcache-0.4.1-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:5d4e2366a9c7b837555cf02fb9be2e3167d333aff716332ef1b7c3a142ec40c5", size = 209491, upload-time = "2025-10-08T19:46:08.909Z" }, + { url = "https://files.pythonhosted.org/packages/a1/c8/038d909c61c5bb039070b3fb02ad5cccdb1dde0d714792e251cdb17c9c05/propcache-0.4.1-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:9d2b6caef873b4f09e26ea7e33d65f42b944837563a47a94719cc3544319a0db", size = 215319, upload-time = "2025-10-08T19:46:10.7Z" }, + { url = "https://files.pythonhosted.org/packages/08/57/8c87e93142b2c1fa2408e45695205a7ba05fb5db458c0bf5c06ba0e09ea6/propcache-0.4.1-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2b16ec437a8c8a965ecf95739448dd938b5c7f56e67ea009f4300d8df05f32b7", size = 196856, upload-time = "2025-10-08T19:46:12.003Z" }, + { url = "https://files.pythonhosted.org/packages/42/df/5615fec76aa561987a534759b3686008a288e73107faa49a8ae5795a9f7a/propcache-0.4.1-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:296f4c8ed03ca7476813fe666c9ea97869a8d7aec972618671b33a38a5182ef4", size = 193241, upload-time = "2025-10-08T19:46:13.495Z" }, + { url = "https://files.pythonhosted.org/packages/d5/21/62949eb3a7a54afe8327011c90aca7e03547787a88fb8bd9726806482fea/propcache-0.4.1-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:1f0978529a418ebd1f49dad413a2b68af33f85d5c5ca5c6ca2a3bed375a7ac60", size = 190552, upload-time = "2025-10-08T19:46:14.938Z" }, + { url = "https://files.pythonhosted.org/packages/30/ee/ab4d727dd70806e5b4de96a798ae7ac6e4d42516f030ee60522474b6b332/propcache-0.4.1-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:fd138803047fb4c062b1c1dd95462f5209456bfab55c734458f15d11da288f8f", size = 200113, upload-time = "2025-10-08T19:46:16.695Z" }, + { url = "https://files.pythonhosted.org/packages/8a/0b/38b46208e6711b016aa8966a3ac793eee0d05c7159d8342aa27fc0bc365e/propcache-0.4.1-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:8c9b3cbe4584636d72ff556d9036e0c9317fa27b3ac1f0f558e7e84d1c9c5900", size = 200778, upload-time = "2025-10-08T19:46:18.023Z" }, + { url = "https://files.pythonhosted.org/packages/cf/81/5abec54355ed344476bee711e9f04815d4b00a311ab0535599204eecc257/propcache-0.4.1-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:f93243fdc5657247533273ac4f86ae106cc6445a0efacb9a1bfe982fcfefd90c", size = 193047, upload-time = "2025-10-08T19:46:19.449Z" }, + { url = "https://files.pythonhosted.org/packages/ec/b6/1f237c04e32063cb034acd5f6ef34ef3a394f75502e72703545631ab1ef6/propcache-0.4.1-cp310-cp310-win32.whl", hash = "sha256:a0ee98db9c5f80785b266eb805016e36058ac72c51a064040f2bc43b61101cdb", size = 38093, upload-time = "2025-10-08T19:46:20.643Z" }, + { url = "https://files.pythonhosted.org/packages/a6/67/354aac4e0603a15f76439caf0427781bcd6797f370377f75a642133bc954/propcache-0.4.1-cp310-cp310-win_amd64.whl", hash = "sha256:1cdb7988c4e5ac7f6d175a28a9aa0c94cb6f2ebe52756a3c0cda98d2809a9e37", size = 41638, upload-time = "2025-10-08T19:46:21.935Z" }, + { url = "https://files.pythonhosted.org/packages/e0/e1/74e55b9fd1a4c209ff1a9a824bf6c8b3d1fc5a1ac3eabe23462637466785/propcache-0.4.1-cp310-cp310-win_arm64.whl", hash = "sha256:d82ad62b19645419fe79dd63b3f9253e15b30e955c0170e5cebc350c1844e581", size = 38229, upload-time = "2025-10-08T19:46:23.368Z" }, + { url = "https://files.pythonhosted.org/packages/8c/d4/4e2c9aaf7ac2242b9358f98dccd8f90f2605402f5afeff6c578682c2c491/propcache-0.4.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:60a8fda9644b7dfd5dece8c61d8a85e271cb958075bfc4e01083c148b61a7caf", size = 80208, upload-time = "2025-10-08T19:46:24.597Z" }, + { url = "https://files.pythonhosted.org/packages/c2/21/d7b68e911f9c8e18e4ae43bdbc1e1e9bbd971f8866eb81608947b6f585ff/propcache-0.4.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c30b53e7e6bda1d547cabb47c825f3843a0a1a42b0496087bb58d8fedf9f41b5", size = 45777, upload-time = "2025-10-08T19:46:25.733Z" }, + { url = "https://files.pythonhosted.org/packages/d3/1d/11605e99ac8ea9435651ee71ab4cb4bf03f0949586246476a25aadfec54a/propcache-0.4.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:6918ecbd897443087a3b7cd978d56546a812517dcaaca51b49526720571fa93e", size = 47647, upload-time = "2025-10-08T19:46:27.304Z" }, + { url = "https://files.pythonhosted.org/packages/58/1a/3c62c127a8466c9c843bccb503d40a273e5cc69838805f322e2826509e0d/propcache-0.4.1-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3d902a36df4e5989763425a8ab9e98cd8ad5c52c823b34ee7ef307fd50582566", size = 214929, upload-time = "2025-10-08T19:46:28.62Z" }, + { url = "https://files.pythonhosted.org/packages/56/b9/8fa98f850960b367c4b8fe0592e7fc341daa7a9462e925228f10a60cf74f/propcache-0.4.1-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:a9695397f85973bb40427dedddf70d8dc4a44b22f1650dd4af9eedf443d45165", size = 221778, upload-time = "2025-10-08T19:46:30.358Z" }, + { url = "https://files.pythonhosted.org/packages/46/a6/0ab4f660eb59649d14b3d3d65c439421cf2f87fe5dd68591cbe3c1e78a89/propcache-0.4.1-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:2bb07ffd7eaad486576430c89f9b215f9e4be68c4866a96e97db9e97fead85dc", size = 228144, upload-time = "2025-10-08T19:46:32.607Z" }, + { url = "https://files.pythonhosted.org/packages/52/6a/57f43e054fb3d3a56ac9fc532bc684fc6169a26c75c353e65425b3e56eef/propcache-0.4.1-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fd6f30fdcf9ae2a70abd34da54f18da086160e4d7d9251f81f3da0ff84fc5a48", size = 210030, upload-time = "2025-10-08T19:46:33.969Z" }, + { url = "https://files.pythonhosted.org/packages/40/e2/27e6feebb5f6b8408fa29f5efbb765cd54c153ac77314d27e457a3e993b7/propcache-0.4.1-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:fc38cba02d1acba4e2869eef1a57a43dfbd3d49a59bf90dda7444ec2be6a5570", size = 208252, upload-time = "2025-10-08T19:46:35.309Z" }, + { url = "https://files.pythonhosted.org/packages/9e/f8/91c27b22ccda1dbc7967f921c42825564fa5336a01ecd72eb78a9f4f53c2/propcache-0.4.1-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:67fad6162281e80e882fb3ec355398cf72864a54069d060321f6cd0ade95fe85", size = 202064, upload-time = "2025-10-08T19:46:36.993Z" }, + { url = "https://files.pythonhosted.org/packages/f2/26/7f00bd6bd1adba5aafe5f4a66390f243acab58eab24ff1a08bebb2ef9d40/propcache-0.4.1-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:f10207adf04d08bec185bae14d9606a1444715bc99180f9331c9c02093e1959e", size = 212429, upload-time = "2025-10-08T19:46:38.398Z" }, + { url = "https://files.pythonhosted.org/packages/84/89/fd108ba7815c1117ddca79c228f3f8a15fc82a73bca8b142eb5de13b2785/propcache-0.4.1-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:e9b0d8d0845bbc4cfcdcbcdbf5086886bc8157aa963c31c777ceff7846c77757", size = 216727, upload-time = "2025-10-08T19:46:39.732Z" }, + { url = "https://files.pythonhosted.org/packages/79/37/3ec3f7e3173e73f1d600495d8b545b53802cbf35506e5732dd8578db3724/propcache-0.4.1-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:981333cb2f4c1896a12f4ab92a9cc8f09ea664e9b7dbdc4eff74627af3a11c0f", size = 205097, upload-time = "2025-10-08T19:46:41.025Z" }, + { url = "https://files.pythonhosted.org/packages/61/b0/b2631c19793f869d35f47d5a3a56fb19e9160d3c119f15ac7344fc3ccae7/propcache-0.4.1-cp311-cp311-win32.whl", hash = "sha256:f1d2f90aeec838a52f1c1a32fe9a619fefd5e411721a9117fbf82aea638fe8a1", size = 38084, upload-time = "2025-10-08T19:46:42.693Z" }, + { url = "https://files.pythonhosted.org/packages/f4/78/6cce448e2098e9f3bfc91bb877f06aa24b6ccace872e39c53b2f707c4648/propcache-0.4.1-cp311-cp311-win_amd64.whl", hash = "sha256:364426a62660f3f699949ac8c621aad6977be7126c5807ce48c0aeb8e7333ea6", size = 41637, upload-time = "2025-10-08T19:46:43.778Z" }, + { url = "https://files.pythonhosted.org/packages/9c/e9/754f180cccd7f51a39913782c74717c581b9cc8177ad0e949f4d51812383/propcache-0.4.1-cp311-cp311-win_arm64.whl", hash = "sha256:e53f3a38d3510c11953f3e6a33f205c6d1b001129f972805ca9b42fc308bc239", size = 38064, upload-time = "2025-10-08T19:46:44.872Z" }, + { url = "https://files.pythonhosted.org/packages/a2/0f/f17b1b2b221d5ca28b4b876e8bb046ac40466513960646bda8e1853cdfa2/propcache-0.4.1-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:e153e9cd40cc8945138822807139367f256f89c6810c2634a4f6902b52d3b4e2", size = 80061, upload-time = "2025-10-08T19:46:46.075Z" }, + { url = "https://files.pythonhosted.org/packages/76/47/8ccf75935f51448ba9a16a71b783eb7ef6b9ee60f5d14c7f8a8a79fbeed7/propcache-0.4.1-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:cd547953428f7abb73c5ad82cbb32109566204260d98e41e5dfdc682eb7f8403", size = 46037, upload-time = "2025-10-08T19:46:47.23Z" }, + { url = "https://files.pythonhosted.org/packages/0a/b6/5c9a0e42df4d00bfb4a3cbbe5cf9f54260300c88a0e9af1f47ca5ce17ac0/propcache-0.4.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f048da1b4f243fc44f205dfd320933a951b8d89e0afd4c7cacc762a8b9165207", size = 47324, upload-time = "2025-10-08T19:46:48.384Z" }, + { url = "https://files.pythonhosted.org/packages/9e/d3/6c7ee328b39a81ee877c962469f1e795f9db87f925251efeb0545e0020d0/propcache-0.4.1-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:ec17c65562a827bba85e3872ead335f95405ea1674860d96483a02f5c698fa72", size = 225505, upload-time = "2025-10-08T19:46:50.055Z" }, + { url = "https://files.pythonhosted.org/packages/01/5d/1c53f4563490b1d06a684742cc6076ef944bc6457df6051b7d1a877c057b/propcache-0.4.1-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:405aac25c6394ef275dee4c709be43745d36674b223ba4eb7144bf4d691b7367", size = 230242, upload-time = "2025-10-08T19:46:51.815Z" }, + { url = "https://files.pythonhosted.org/packages/20/e1/ce4620633b0e2422207c3cb774a0ee61cac13abc6217763a7b9e2e3f4a12/propcache-0.4.1-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:0013cb6f8dde4b2a2f66903b8ba740bdfe378c943c4377a200551ceb27f379e4", size = 238474, upload-time = "2025-10-08T19:46:53.208Z" }, + { url = "https://files.pythonhosted.org/packages/46/4b/3aae6835b8e5f44ea6a68348ad90f78134047b503765087be2f9912140ea/propcache-0.4.1-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:15932ab57837c3368b024473a525e25d316d8353016e7cc0e5ba9eb343fbb1cf", size = 221575, upload-time = "2025-10-08T19:46:54.511Z" }, + { url = "https://files.pythonhosted.org/packages/6e/a5/8a5e8678bcc9d3a1a15b9a29165640d64762d424a16af543f00629c87338/propcache-0.4.1-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:031dce78b9dc099f4c29785d9cf5577a3faf9ebf74ecbd3c856a7b92768c3df3", size = 216736, upload-time = "2025-10-08T19:46:56.212Z" }, + { url = "https://files.pythonhosted.org/packages/f1/63/b7b215eddeac83ca1c6b934f89d09a625aa9ee4ba158338854c87210cc36/propcache-0.4.1-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:ab08df6c9a035bee56e31af99be621526bd237bea9f32def431c656b29e41778", size = 213019, upload-time = "2025-10-08T19:46:57.595Z" }, + { url = "https://files.pythonhosted.org/packages/57/74/f580099a58c8af587cac7ba19ee7cb418506342fbbe2d4a4401661cca886/propcache-0.4.1-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:4d7af63f9f93fe593afbf104c21b3b15868efb2c21d07d8732c0c4287e66b6a6", size = 220376, upload-time = "2025-10-08T19:46:59.067Z" }, + { url = "https://files.pythonhosted.org/packages/c4/ee/542f1313aff7eaf19c2bb758c5d0560d2683dac001a1c96d0774af799843/propcache-0.4.1-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:cfc27c945f422e8b5071b6e93169679e4eb5bf73bbcbf1ba3ae3a83d2f78ebd9", size = 226988, upload-time = "2025-10-08T19:47:00.544Z" }, + { url = "https://files.pythonhosted.org/packages/8f/18/9c6b015dd9c6930f6ce2229e1f02fb35298b847f2087ea2b436a5bfa7287/propcache-0.4.1-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:35c3277624a080cc6ec6f847cbbbb5b49affa3598c4535a0a4682a697aaa5c75", size = 215615, upload-time = "2025-10-08T19:47:01.968Z" }, + { url = "https://files.pythonhosted.org/packages/80/9e/e7b85720b98c45a45e1fca6a177024934dc9bc5f4d5dd04207f216fc33ed/propcache-0.4.1-cp312-cp312-win32.whl", hash = "sha256:671538c2262dadb5ba6395e26c1731e1d52534bfe9ae56d0b5573ce539266aa8", size = 38066, upload-time = "2025-10-08T19:47:03.503Z" }, + { url = "https://files.pythonhosted.org/packages/54/09/d19cff2a5aaac632ec8fc03737b223597b1e347416934c1b3a7df079784c/propcache-0.4.1-cp312-cp312-win_amd64.whl", hash = "sha256:cb2d222e72399fcf5890d1d5cc1060857b9b236adff2792ff48ca2dfd46c81db", size = 41655, upload-time = "2025-10-08T19:47:04.973Z" }, + { url = "https://files.pythonhosted.org/packages/68/ab/6b5c191bb5de08036a8c697b265d4ca76148efb10fa162f14af14fb5f076/propcache-0.4.1-cp312-cp312-win_arm64.whl", hash = "sha256:204483131fb222bdaaeeea9f9e6c6ed0cac32731f75dfc1d4a567fc1926477c1", size = 37789, upload-time = "2025-10-08T19:47:06.077Z" }, + { url = "https://files.pythonhosted.org/packages/bf/df/6d9c1b6ac12b003837dde8a10231a7344512186e87b36e855bef32241942/propcache-0.4.1-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:43eedf29202c08550aac1d14e0ee619b0430aaef78f85864c1a892294fbc28cf", size = 77750, upload-time = "2025-10-08T19:47:07.648Z" }, + { url = "https://files.pythonhosted.org/packages/8b/e8/677a0025e8a2acf07d3418a2e7ba529c9c33caf09d3c1f25513023c1db56/propcache-0.4.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:d62cdfcfd89ccb8de04e0eda998535c406bf5e060ffd56be6c586cbcc05b3311", size = 44780, upload-time = "2025-10-08T19:47:08.851Z" }, + { url = "https://files.pythonhosted.org/packages/89/a4/92380f7ca60f99ebae761936bc48a72a639e8a47b29050615eef757cb2a7/propcache-0.4.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:cae65ad55793da34db5f54e4029b89d3b9b9490d8abe1b4c7ab5d4b8ec7ebf74", size = 46308, upload-time = "2025-10-08T19:47:09.982Z" }, + { url = "https://files.pythonhosted.org/packages/2d/48/c5ac64dee5262044348d1d78a5f85dd1a57464a60d30daee946699963eb3/propcache-0.4.1-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:333ddb9031d2704a301ee3e506dc46b1fe5f294ec198ed6435ad5b6a085facfe", size = 208182, upload-time = "2025-10-08T19:47:11.319Z" }, + { url = "https://files.pythonhosted.org/packages/c6/0c/cd762dd011a9287389a6a3eb43aa30207bde253610cca06824aeabfe9653/propcache-0.4.1-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:fd0858c20f078a32cf55f7e81473d96dcf3b93fd2ccdb3d40fdf54b8573df3af", size = 211215, upload-time = "2025-10-08T19:47:13.146Z" }, + { url = "https://files.pythonhosted.org/packages/30/3e/49861e90233ba36890ae0ca4c660e95df565b2cd15d4a68556ab5865974e/propcache-0.4.1-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:678ae89ebc632c5c204c794f8dab2837c5f159aeb59e6ed0539500400577298c", size = 218112, upload-time = "2025-10-08T19:47:14.913Z" }, + { url = "https://files.pythonhosted.org/packages/f1/8b/544bc867e24e1bd48f3118cecd3b05c694e160a168478fa28770f22fd094/propcache-0.4.1-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:d472aeb4fbf9865e0c6d622d7f4d54a4e101a89715d8904282bb5f9a2f476c3f", size = 204442, upload-time = "2025-10-08T19:47:16.277Z" }, + { url = "https://files.pythonhosted.org/packages/50/a6/4282772fd016a76d3e5c0df58380a5ea64900afd836cec2c2f662d1b9bb3/propcache-0.4.1-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:4d3df5fa7e36b3225954fba85589da77a0fe6a53e3976de39caf04a0db4c36f1", size = 199398, upload-time = "2025-10-08T19:47:17.962Z" }, + { url = "https://files.pythonhosted.org/packages/3e/ec/d8a7cd406ee1ddb705db2139f8a10a8a427100347bd698e7014351c7af09/propcache-0.4.1-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:ee17f18d2498f2673e432faaa71698032b0127ebf23ae5974eeaf806c279df24", size = 196920, upload-time = "2025-10-08T19:47:19.355Z" }, + { url = "https://files.pythonhosted.org/packages/f6/6c/f38ab64af3764f431e359f8baf9e0a21013e24329e8b85d2da32e8ed07ca/propcache-0.4.1-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:580e97762b950f993ae618e167e7be9256b8353c2dcd8b99ec100eb50f5286aa", size = 203748, upload-time = "2025-10-08T19:47:21.338Z" }, + { url = "https://files.pythonhosted.org/packages/d6/e3/fa846bd70f6534d647886621388f0a265254d30e3ce47e5c8e6e27dbf153/propcache-0.4.1-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:501d20b891688eb8e7aa903021f0b72d5a55db40ffaab27edefd1027caaafa61", size = 205877, upload-time = "2025-10-08T19:47:23.059Z" }, + { url = "https://files.pythonhosted.org/packages/e2/39/8163fc6f3133fea7b5f2827e8eba2029a0277ab2c5beee6c1db7b10fc23d/propcache-0.4.1-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:9a0bd56e5b100aef69bd8562b74b46254e7c8812918d3baa700c8a8009b0af66", size = 199437, upload-time = "2025-10-08T19:47:24.445Z" }, + { url = "https://files.pythonhosted.org/packages/93/89/caa9089970ca49c7c01662bd0eeedfe85494e863e8043565aeb6472ce8fe/propcache-0.4.1-cp313-cp313-win32.whl", hash = "sha256:bcc9aaa5d80322bc2fb24bb7accb4a30f81e90ab8d6ba187aec0744bc302ad81", size = 37586, upload-time = "2025-10-08T19:47:25.736Z" }, + { url = "https://files.pythonhosted.org/packages/f5/ab/f76ec3c3627c883215b5c8080debb4394ef5a7a29be811f786415fc1e6fd/propcache-0.4.1-cp313-cp313-win_amd64.whl", hash = "sha256:381914df18634f5494334d201e98245c0596067504b9372d8cf93f4bb23e025e", size = 40790, upload-time = "2025-10-08T19:47:26.847Z" }, + { url = "https://files.pythonhosted.org/packages/59/1b/e71ae98235f8e2ba5004d8cb19765a74877abf189bc53fc0c80d799e56c3/propcache-0.4.1-cp313-cp313-win_arm64.whl", hash = "sha256:8873eb4460fd55333ea49b7d189749ecf6e55bf85080f11b1c4530ed3034cba1", size = 37158, upload-time = "2025-10-08T19:47:27.961Z" }, + { url = "https://files.pythonhosted.org/packages/83/ce/a31bbdfc24ee0dcbba458c8175ed26089cf109a55bbe7b7640ed2470cfe9/propcache-0.4.1-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:92d1935ee1f8d7442da9c0c4fa7ac20d07e94064184811b685f5c4fada64553b", size = 81451, upload-time = "2025-10-08T19:47:29.445Z" }, + { url = "https://files.pythonhosted.org/packages/25/9c/442a45a470a68456e710d96cacd3573ef26a1d0a60067e6a7d5e655621ed/propcache-0.4.1-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:473c61b39e1460d386479b9b2f337da492042447c9b685f28be4f74d3529e566", size = 46374, upload-time = "2025-10-08T19:47:30.579Z" }, + { url = "https://files.pythonhosted.org/packages/f4/bf/b1d5e21dbc3b2e889ea4327044fb16312a736d97640fb8b6aa3f9c7b3b65/propcache-0.4.1-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:c0ef0aaafc66fbd87842a3fe3902fd889825646bc21149eafe47be6072725835", size = 48396, upload-time = "2025-10-08T19:47:31.79Z" }, + { url = "https://files.pythonhosted.org/packages/f4/04/5b4c54a103d480e978d3c8a76073502b18db0c4bc17ab91b3cb5092ad949/propcache-0.4.1-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f95393b4d66bfae908c3ca8d169d5f79cd65636ae15b5e7a4f6e67af675adb0e", size = 275950, upload-time = "2025-10-08T19:47:33.481Z" }, + { url = "https://files.pythonhosted.org/packages/b4/c1/86f846827fb969c4b78b0af79bba1d1ea2156492e1b83dea8b8a6ae27395/propcache-0.4.1-cp313-cp313t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c07fda85708bc48578467e85099645167a955ba093be0a2dcba962195676e859", size = 273856, upload-time = "2025-10-08T19:47:34.906Z" }, + { url = "https://files.pythonhosted.org/packages/36/1d/fc272a63c8d3bbad6878c336c7a7dea15e8f2d23a544bda43205dfa83ada/propcache-0.4.1-cp313-cp313t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:af223b406d6d000830c6f65f1e6431783fc3f713ba3e6cc8c024d5ee96170a4b", size = 280420, upload-time = "2025-10-08T19:47:36.338Z" }, + { url = "https://files.pythonhosted.org/packages/07/0c/01f2219d39f7e53d52e5173bcb09c976609ba30209912a0680adfb8c593a/propcache-0.4.1-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a78372c932c90ee474559c5ddfffd718238e8673c340dc21fe45c5b8b54559a0", size = 263254, upload-time = "2025-10-08T19:47:37.692Z" }, + { url = "https://files.pythonhosted.org/packages/2d/18/cd28081658ce597898f0c4d174d4d0f3c5b6d4dc27ffafeef835c95eb359/propcache-0.4.1-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:564d9f0d4d9509e1a870c920a89b2fec951b44bf5ba7d537a9e7c1ccec2c18af", size = 261205, upload-time = "2025-10-08T19:47:39.659Z" }, + { url = "https://files.pythonhosted.org/packages/7a/71/1f9e22eb8b8316701c2a19fa1f388c8a3185082607da8e406a803c9b954e/propcache-0.4.1-cp313-cp313t-musllinux_1_2_armv7l.whl", hash = "sha256:17612831fda0138059cc5546f4d12a2aacfb9e47068c06af35c400ba58ba7393", size = 247873, upload-time = "2025-10-08T19:47:41.084Z" }, + { url = "https://files.pythonhosted.org/packages/4a/65/3d4b61f36af2b4eddba9def857959f1016a51066b4f1ce348e0cf7881f58/propcache-0.4.1-cp313-cp313t-musllinux_1_2_ppc64le.whl", hash = "sha256:41a89040cb10bd345b3c1a873b2bf36413d48da1def52f268a055f7398514874", size = 262739, upload-time = "2025-10-08T19:47:42.51Z" }, + { url = "https://files.pythonhosted.org/packages/2a/42/26746ab087faa77c1c68079b228810436ccd9a5ce9ac85e2b7307195fd06/propcache-0.4.1-cp313-cp313t-musllinux_1_2_s390x.whl", hash = "sha256:e35b88984e7fa64aacecea39236cee32dd9bd8c55f57ba8a75cf2399553f9bd7", size = 263514, upload-time = "2025-10-08T19:47:43.927Z" }, + { url = "https://files.pythonhosted.org/packages/94/13/630690fe201f5502d2403dd3cfd451ed8858fe3c738ee88d095ad2ff407b/propcache-0.4.1-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:6f8b465489f927b0df505cbe26ffbeed4d6d8a2bbc61ce90eb074ff129ef0ab1", size = 257781, upload-time = "2025-10-08T19:47:45.448Z" }, + { url = "https://files.pythonhosted.org/packages/92/f7/1d4ec5841505f423469efbfc381d64b7b467438cd5a4bbcbb063f3b73d27/propcache-0.4.1-cp313-cp313t-win32.whl", hash = "sha256:2ad890caa1d928c7c2965b48f3a3815c853180831d0e5503d35cf00c472f4717", size = 41396, upload-time = "2025-10-08T19:47:47.202Z" }, + { url = "https://files.pythonhosted.org/packages/48/f0/615c30622316496d2cbbc29f5985f7777d3ada70f23370608c1d3e081c1f/propcache-0.4.1-cp313-cp313t-win_amd64.whl", hash = "sha256:f7ee0e597f495cf415bcbd3da3caa3bd7e816b74d0d52b8145954c5e6fd3ff37", size = 44897, upload-time = "2025-10-08T19:47:48.336Z" }, + { url = "https://files.pythonhosted.org/packages/fd/ca/6002e46eccbe0e33dcd4069ef32f7f1c9e243736e07adca37ae8c4830ec3/propcache-0.4.1-cp313-cp313t-win_arm64.whl", hash = "sha256:929d7cbe1f01bb7baffb33dc14eb5691c95831450a26354cd210a8155170c93a", size = 39789, upload-time = "2025-10-08T19:47:49.876Z" }, + { url = "https://files.pythonhosted.org/packages/8e/5c/bca52d654a896f831b8256683457ceddd490ec18d9ec50e97dfd8fc726a8/propcache-0.4.1-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:3f7124c9d820ba5548d431afb4632301acf965db49e666aa21c305cbe8c6de12", size = 78152, upload-time = "2025-10-08T19:47:51.051Z" }, + { url = "https://files.pythonhosted.org/packages/65/9b/03b04e7d82a5f54fb16113d839f5ea1ede58a61e90edf515f6577c66fa8f/propcache-0.4.1-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:c0d4b719b7da33599dfe3b22d3db1ef789210a0597bc650b7cee9c77c2be8c5c", size = 44869, upload-time = "2025-10-08T19:47:52.594Z" }, + { url = "https://files.pythonhosted.org/packages/b2/fa/89a8ef0468d5833a23fff277b143d0573897cf75bd56670a6d28126c7d68/propcache-0.4.1-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:9f302f4783709a78240ebc311b793f123328716a60911d667e0c036bc5dcbded", size = 46596, upload-time = "2025-10-08T19:47:54.073Z" }, + { url = "https://files.pythonhosted.org/packages/86/bd/47816020d337f4a746edc42fe8d53669965138f39ee117414c7d7a340cfe/propcache-0.4.1-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c80ee5802e3fb9ea37938e7eecc307fb984837091d5fd262bb37238b1ae97641", size = 206981, upload-time = "2025-10-08T19:47:55.715Z" }, + { url = "https://files.pythonhosted.org/packages/df/f6/c5fa1357cc9748510ee55f37173eb31bfde6d94e98ccd9e6f033f2fc06e1/propcache-0.4.1-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ed5a841e8bb29a55fb8159ed526b26adc5bdd7e8bd7bf793ce647cb08656cdf4", size = 211490, upload-time = "2025-10-08T19:47:57.499Z" }, + { url = "https://files.pythonhosted.org/packages/80/1e/e5889652a7c4a3846683401a48f0f2e5083ce0ec1a8a5221d8058fbd1adf/propcache-0.4.1-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:55c72fd6ea2da4c318e74ffdf93c4fe4e926051133657459131a95c846d16d44", size = 215371, upload-time = "2025-10-08T19:47:59.317Z" }, + { url = "https://files.pythonhosted.org/packages/b2/f2/889ad4b2408f72fe1a4f6a19491177b30ea7bf1a0fd5f17050ca08cfc882/propcache-0.4.1-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:8326e144341460402713f91df60ade3c999d601e7eb5ff8f6f7862d54de0610d", size = 201424, upload-time = "2025-10-08T19:48:00.67Z" }, + { url = "https://files.pythonhosted.org/packages/27/73/033d63069b57b0812c8bd19f311faebeceb6ba31b8f32b73432d12a0b826/propcache-0.4.1-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:060b16ae65bc098da7f6d25bf359f1f31f688384858204fe5d652979e0015e5b", size = 197566, upload-time = "2025-10-08T19:48:02.604Z" }, + { url = "https://files.pythonhosted.org/packages/dc/89/ce24f3dc182630b4e07aa6d15f0ff4b14ed4b9955fae95a0b54c58d66c05/propcache-0.4.1-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:89eb3fa9524f7bec9de6e83cf3faed9d79bffa560672c118a96a171a6f55831e", size = 193130, upload-time = "2025-10-08T19:48:04.499Z" }, + { url = "https://files.pythonhosted.org/packages/a9/24/ef0d5fd1a811fb5c609278d0209c9f10c35f20581fcc16f818da959fc5b4/propcache-0.4.1-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:dee69d7015dc235f526fe80a9c90d65eb0039103fe565776250881731f06349f", size = 202625, upload-time = "2025-10-08T19:48:06.213Z" }, + { url = "https://files.pythonhosted.org/packages/f5/02/98ec20ff5546f68d673df2f7a69e8c0d076b5abd05ca882dc7ee3a83653d/propcache-0.4.1-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:5558992a00dfd54ccbc64a32726a3357ec93825a418a401f5cc67df0ac5d9e49", size = 204209, upload-time = "2025-10-08T19:48:08.432Z" }, + { url = "https://files.pythonhosted.org/packages/a0/87/492694f76759b15f0467a2a93ab68d32859672b646aa8a04ce4864e7932d/propcache-0.4.1-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:c9b822a577f560fbd9554812526831712c1436d2c046cedee4c3796d3543b144", size = 197797, upload-time = "2025-10-08T19:48:09.968Z" }, + { url = "https://files.pythonhosted.org/packages/ee/36/66367de3575db1d2d3f3d177432bd14ee577a39d3f5d1b3d5df8afe3b6e2/propcache-0.4.1-cp314-cp314-win32.whl", hash = "sha256:ab4c29b49d560fe48b696cdcb127dd36e0bc2472548f3bf56cc5cb3da2b2984f", size = 38140, upload-time = "2025-10-08T19:48:11.232Z" }, + { url = "https://files.pythonhosted.org/packages/0c/2a/a758b47de253636e1b8aef181c0b4f4f204bf0dd964914fb2af90a95b49b/propcache-0.4.1-cp314-cp314-win_amd64.whl", hash = "sha256:5a103c3eb905fcea0ab98be99c3a9a5ab2de60228aa5aceedc614c0281cf6153", size = 41257, upload-time = "2025-10-08T19:48:12.707Z" }, + { url = "https://files.pythonhosted.org/packages/34/5e/63bd5896c3fec12edcbd6f12508d4890d23c265df28c74b175e1ef9f4f3b/propcache-0.4.1-cp314-cp314-win_arm64.whl", hash = "sha256:74c1fb26515153e482e00177a1ad654721bf9207da8a494a0c05e797ad27b992", size = 38097, upload-time = "2025-10-08T19:48:13.923Z" }, + { url = "https://files.pythonhosted.org/packages/99/85/9ff785d787ccf9bbb3f3106f79884a130951436f58392000231b4c737c80/propcache-0.4.1-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:824e908bce90fb2743bd6b59db36eb4f45cd350a39637c9f73b1c1ea66f5b75f", size = 81455, upload-time = "2025-10-08T19:48:15.16Z" }, + { url = "https://files.pythonhosted.org/packages/90/85/2431c10c8e7ddb1445c1f7c4b54d886e8ad20e3c6307e7218f05922cad67/propcache-0.4.1-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:c2b5e7db5328427c57c8e8831abda175421b709672f6cfc3d630c3b7e2146393", size = 46372, upload-time = "2025-10-08T19:48:16.424Z" }, + { url = "https://files.pythonhosted.org/packages/01/20/b0972d902472da9bcb683fa595099911f4d2e86e5683bcc45de60dd05dc3/propcache-0.4.1-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:6f6ff873ed40292cd4969ef5310179afd5db59fdf055897e282485043fc80ad0", size = 48411, upload-time = "2025-10-08T19:48:17.577Z" }, + { url = "https://files.pythonhosted.org/packages/e2/e3/7dc89f4f21e8f99bad3d5ddb3a3389afcf9da4ac69e3deb2dcdc96e74169/propcache-0.4.1-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:49a2dc67c154db2c1463013594c458881a069fcf98940e61a0569016a583020a", size = 275712, upload-time = "2025-10-08T19:48:18.901Z" }, + { url = "https://files.pythonhosted.org/packages/20/67/89800c8352489b21a8047c773067644e3897f02ecbbd610f4d46b7f08612/propcache-0.4.1-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:005f08e6a0529984491e37d8dbc3dd86f84bd78a8ceb5fa9a021f4c48d4984be", size = 273557, upload-time = "2025-10-08T19:48:20.762Z" }, + { url = "https://files.pythonhosted.org/packages/e2/a1/b52b055c766a54ce6d9c16d9aca0cad8059acd9637cdf8aa0222f4a026ef/propcache-0.4.1-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:5c3310452e0d31390da9035c348633b43d7e7feb2e37be252be6da45abd1abcc", size = 280015, upload-time = "2025-10-08T19:48:22.592Z" }, + { url = "https://files.pythonhosted.org/packages/48/c8/33cee30bd890672c63743049f3c9e4be087e6780906bfc3ec58528be59c1/propcache-0.4.1-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4c3c70630930447f9ef1caac7728c8ad1c56bc5015338b20fed0d08ea2480b3a", size = 262880, upload-time = "2025-10-08T19:48:23.947Z" }, + { url = "https://files.pythonhosted.org/packages/0c/b1/8f08a143b204b418285c88b83d00edbd61afbc2c6415ffafc8905da7038b/propcache-0.4.1-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:8e57061305815dfc910a3634dcf584f08168a8836e6999983569f51a8544cd89", size = 260938, upload-time = "2025-10-08T19:48:25.656Z" }, + { url = "https://files.pythonhosted.org/packages/cf/12/96e4664c82ca2f31e1c8dff86afb867348979eb78d3cb8546a680287a1e9/propcache-0.4.1-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:521a463429ef54143092c11a77e04056dd00636f72e8c45b70aaa3140d639726", size = 247641, upload-time = "2025-10-08T19:48:27.207Z" }, + { url = "https://files.pythonhosted.org/packages/18/ed/e7a9cfca28133386ba52278136d42209d3125db08d0a6395f0cba0c0285c/propcache-0.4.1-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:120c964da3fdc75e3731aa392527136d4ad35868cc556fd09bb6d09172d9a367", size = 262510, upload-time = "2025-10-08T19:48:28.65Z" }, + { url = "https://files.pythonhosted.org/packages/f5/76/16d8bf65e8845dd62b4e2b57444ab81f07f40caa5652b8969b87ddcf2ef6/propcache-0.4.1-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:d8f353eb14ee3441ee844ade4277d560cdd68288838673273b978e3d6d2c8f36", size = 263161, upload-time = "2025-10-08T19:48:30.133Z" }, + { url = "https://files.pythonhosted.org/packages/e7/70/c99e9edb5d91d5ad8a49fa3c1e8285ba64f1476782fed10ab251ff413ba1/propcache-0.4.1-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:ab2943be7c652f09638800905ee1bab2c544e537edb57d527997a24c13dc1455", size = 257393, upload-time = "2025-10-08T19:48:31.567Z" }, + { url = "https://files.pythonhosted.org/packages/08/02/87b25304249a35c0915d236575bc3574a323f60b47939a2262b77632a3ee/propcache-0.4.1-cp314-cp314t-win32.whl", hash = "sha256:05674a162469f31358c30bcaa8883cb7829fa3110bf9c0991fe27d7896c42d85", size = 42546, upload-time = "2025-10-08T19:48:32.872Z" }, + { url = "https://files.pythonhosted.org/packages/cb/ef/3c6ecf8b317aa982f309835e8f96987466123c6e596646d4e6a1dfcd080f/propcache-0.4.1-cp314-cp314t-win_amd64.whl", hash = "sha256:990f6b3e2a27d683cb7602ed6c86f15ee6b43b1194736f9baaeb93d0016633b1", size = 46259, upload-time = "2025-10-08T19:48:34.226Z" }, + { url = "https://files.pythonhosted.org/packages/c4/2d/346e946d4951f37eca1e4f55be0f0174c52cd70720f84029b02f296f4a38/propcache-0.4.1-cp314-cp314t-win_arm64.whl", hash = "sha256:ecef2343af4cc68e05131e45024ba34f6095821988a9d0a02aa7c73fcc448aa9", size = 40428, upload-time = "2025-10-08T19:48:35.441Z" }, + { url = "https://files.pythonhosted.org/packages/9b/01/0ebaec9003f5d619a7475165961f8e3083cf8644d704b60395df3601632d/propcache-0.4.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:3d233076ccf9e450c8b3bc6720af226b898ef5d051a2d145f7d765e6e9f9bcff", size = 80277, upload-time = "2025-10-08T19:48:36.647Z" }, + { url = "https://files.pythonhosted.org/packages/34/58/04af97ac586b4ef6b9026c3fd36ee7798b737a832f5d3440a4280dcebd3a/propcache-0.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:357f5bb5c377a82e105e44bd3d52ba22b616f7b9773714bff93573988ef0a5fb", size = 45865, upload-time = "2025-10-08T19:48:37.859Z" }, + { url = "https://files.pythonhosted.org/packages/7c/19/b65d98ae21384518b291d9939e24a8aeac4fdb5101b732576f8f7540e834/propcache-0.4.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:cbc3b6dfc728105b2a57c06791eb07a94229202ea75c59db644d7d496b698cac", size = 47636, upload-time = "2025-10-08T19:48:39.038Z" }, + { url = "https://files.pythonhosted.org/packages/b3/0f/317048c6d91c356c7154dca5af019e6effeb7ee15fa6a6db327cc19e12b4/propcache-0.4.1-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:182b51b421f0501952d938dc0b0eb45246a5b5153c50d42b495ad5fb7517c888", size = 201126, upload-time = "2025-10-08T19:48:40.774Z" }, + { url = "https://files.pythonhosted.org/packages/71/69/0b2a7a5a6ee83292b4b997dbd80549d8ce7d40b6397c1646c0d9495f5a85/propcache-0.4.1-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:4b536b39c5199b96fc6245eb5fb796c497381d3942f169e44e8e392b29c9ebcc", size = 209837, upload-time = "2025-10-08T19:48:42.167Z" }, + { url = "https://files.pythonhosted.org/packages/a5/92/c699ac495a6698df6e497fc2de27af4b6ace10d8e76528357ce153722e45/propcache-0.4.1-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:db65d2af507bbfbdcedb254a11149f894169d90488dd3e7190f7cdcb2d6cd57a", size = 215578, upload-time = "2025-10-08T19:48:43.56Z" }, + { url = "https://files.pythonhosted.org/packages/b3/ee/14de81c5eb02c0ee4f500b4e39c4e1bd0677c06e72379e6ab18923c773fc/propcache-0.4.1-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:fd2dbc472da1f772a4dae4fa24be938a6c544671a912e30529984dd80400cd88", size = 197187, upload-time = "2025-10-08T19:48:45.309Z" }, + { url = "https://files.pythonhosted.org/packages/1d/94/48dce9aaa6d8dd5a0859bad75158ec522546d4ac23f8e2f05fac469477dd/propcache-0.4.1-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:daede9cd44e0f8bdd9e6cc9a607fc81feb80fae7a5fc6cecaff0e0bb32e42d00", size = 193478, upload-time = "2025-10-08T19:48:47.743Z" }, + { url = "https://files.pythonhosted.org/packages/60/b5/0516b563e801e1ace212afde869a0596a0d7115eec0b12d296d75633fb29/propcache-0.4.1-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:71b749281b816793678ae7f3d0d84bd36e694953822eaad408d682efc5ca18e0", size = 190650, upload-time = "2025-10-08T19:48:49.373Z" }, + { url = "https://files.pythonhosted.org/packages/24/89/e0f7d4a5978cd56f8cd67735f74052f257dc471ec901694e430f0d1572fe/propcache-0.4.1-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:0002004213ee1f36cfb3f9a42b5066100c44276b9b72b4e1504cddd3d692e86e", size = 200251, upload-time = "2025-10-08T19:48:51.4Z" }, + { url = "https://files.pythonhosted.org/packages/06/7d/a1fac863d473876ed4406c914f2e14aa82d2f10dd207c9e16fc383cc5a24/propcache-0.4.1-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:fe49d0a85038f36ba9e3ffafa1103e61170b28e95b16622e11be0a0ea07c6781", size = 200919, upload-time = "2025-10-08T19:48:53.227Z" }, + { url = "https://files.pythonhosted.org/packages/c3/4e/f86a256ff24944cf5743e4e6c6994e3526f6acfcfb55e21694c2424f758c/propcache-0.4.1-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:99d43339c83aaf4d32bda60928231848eee470c6bda8d02599cc4cebe872d183", size = 193211, upload-time = "2025-10-08T19:48:55.027Z" }, + { url = "https://files.pythonhosted.org/packages/6e/3f/3fbad5f4356b068f1b047d300a6ff2c66614d7030f078cd50be3fec04228/propcache-0.4.1-cp39-cp39-win32.whl", hash = "sha256:a129e76735bc792794d5177069691c3217898b9f5cee2b2661471e52ffe13f19", size = 38314, upload-time = "2025-10-08T19:48:56.792Z" }, + { url = "https://files.pythonhosted.org/packages/a4/45/d78d136c3a3d215677abb886785aae744da2c3005bcb99e58640c56529b1/propcache-0.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:948dab269721ae9a87fd16c514a0a2c2a1bdb23a9a61b969b0f9d9ee2968546f", size = 41912, upload-time = "2025-10-08T19:48:57.995Z" }, + { url = "https://files.pythonhosted.org/packages/fc/2a/b0632941f25139f4e58450b307242951f7c2717a5704977c6d5323a800af/propcache-0.4.1-cp39-cp39-win_arm64.whl", hash = "sha256:5fd37c406dd6dc85aa743e214cef35dc54bbdd1419baac4f6ae5e5b1a2976938", size = 38450, upload-time = "2025-10-08T19:48:59.349Z" }, + { 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 = "protobuf" +version = "5.29.5" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/43/29/d09e70352e4e88c9c7a198d5645d7277811448d76c23b00345670f7c8a38/protobuf-5.29.5.tar.gz", hash = "sha256:bc1463bafd4b0929216c35f437a8e28731a2b7fe3d98bb77a600efced5a15c84", size = 425226, upload-time = "2025-05-28T23:51:59.82Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/5f/11/6e40e9fc5bba02988a214c07cf324595789ca7820160bfd1f8be96e48539/protobuf-5.29.5-cp310-abi3-win32.whl", hash = "sha256:3f1c6468a2cfd102ff4703976138844f78ebd1fb45f49011afc5139e9e283079", size = 422963, upload-time = "2025-05-28T23:51:41.204Z" }, + { url = "https://files.pythonhosted.org/packages/81/7f/73cefb093e1a2a7c3ffd839e6f9fcafb7a427d300c7f8aef9c64405d8ac6/protobuf-5.29.5-cp310-abi3-win_amd64.whl", hash = "sha256:3f76e3a3675b4a4d867b52e4a5f5b78a2ef9565549d4037e06cf7b0942b1d3fc", size = 434818, upload-time = "2025-05-28T23:51:44.297Z" }, + { url = "https://files.pythonhosted.org/packages/dd/73/10e1661c21f139f2c6ad9b23040ff36fee624310dc28fba20d33fdae124c/protobuf-5.29.5-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:e38c5add5a311f2a6eb0340716ef9b039c1dfa428b28f25a7838ac329204a671", size = 418091, upload-time = "2025-05-28T23:51:45.907Z" }, + { url = "https://files.pythonhosted.org/packages/6c/04/98f6f8cf5b07ab1294c13f34b4e69b3722bb609c5b701d6c169828f9f8aa/protobuf-5.29.5-cp38-abi3-manylinux2014_aarch64.whl", hash = "sha256:fa18533a299d7ab6c55a238bf8629311439995f2e7eca5caaff08663606e9015", size = 319824, upload-time = "2025-05-28T23:51:47.545Z" }, + { url = "https://files.pythonhosted.org/packages/85/e4/07c80521879c2d15f321465ac24c70efe2381378c00bf5e56a0f4fbac8cd/protobuf-5.29.5-cp38-abi3-manylinux2014_x86_64.whl", hash = "sha256:63848923da3325e1bf7e9003d680ce6e14b07e55d0473253a690c3a8b8fd6e61", size = 319942, upload-time = "2025-05-28T23:51:49.11Z" }, + { url = "https://files.pythonhosted.org/packages/b1/24/dc0e489944b3cba0403f26223c0cc552a222af91a42257cbd25ef968b0c0/protobuf-5.29.5-cp38-cp38-win32.whl", hash = "sha256:ef91363ad4faba7b25d844ef1ada59ff1604184c0bcd8b39b8a6bef15e1af238", size = 422886, upload-time = "2025-05-28T23:51:50.688Z" }, + { url = "https://files.pythonhosted.org/packages/a2/bb/9a6e6ec4a99bccd35e326e8e5eab30ca42f3df987c13ca6522b8b4fbd5ff/protobuf-5.29.5-cp38-cp38-win_amd64.whl", hash = "sha256:7318608d56b6402d2ea7704ff1e1e4597bee46d760e7e4dd42a3d45e24b87f2e", size = 434837, upload-time = "2025-05-28T23:51:52.298Z" }, + { url = "https://files.pythonhosted.org/packages/e5/59/ca89678bb0352f094fc92f2b358daa40e3acc91a93aa8f922b24762bf841/protobuf-5.29.5-cp39-cp39-win32.whl", hash = "sha256:6f642dc9a61782fa72b90878af134c5afe1917c89a568cd3476d758d3c3a0736", size = 423025, upload-time = "2025-05-28T23:51:54.003Z" }, + { url = "https://files.pythonhosted.org/packages/96/8b/2c62731fe3e92ddbbeca0174f78f0f8739197cdeb7c75ceb5aad3706963b/protobuf-5.29.5-cp39-cp39-win_amd64.whl", hash = "sha256:470f3af547ef17847a28e1f47200a1cbf0ba3ff57b7de50d22776607cd2ea353", size = 434906, upload-time = "2025-05-28T23:51:55.782Z" }, + { url = "https://files.pythonhosted.org/packages/7e/cc/7e77861000a0691aeea8f4566e5d3aa716f2b1dece4a24439437e41d3d25/protobuf-5.29.5-py3-none-any.whl", hash = "sha256:6cf42630262c59b2d8de33954443d94b746c952b01434fc58a417fdbd2e84bd5", size = 172823, upload-time = "2025-05-28T23:51:58.157Z" }, +] + +[[package]] +name = "protobuf" +version = "6.33.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/19/ff/64a6c8f420818bb873713988ca5492cba3a7946be57e027ac63495157d97/protobuf-6.33.0.tar.gz", hash = "sha256:140303d5c8d2037730c548f8c7b93b20bb1dc301be280c378b82b8894589c954", size = 443463, upload-time = "2025-10-15T20:39:52.159Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/7e/ee/52b3fa8feb6db4a833dfea4943e175ce645144532e8a90f72571ad85df4e/protobuf-6.33.0-cp310-abi3-win32.whl", hash = "sha256:d6101ded078042a8f17959eccd9236fb7a9ca20d3b0098bbcb91533a5680d035", size = 425593, upload-time = "2025-10-15T20:39:40.29Z" }, + { url = "https://files.pythonhosted.org/packages/7b/c6/7a465f1825872c55e0341ff4a80198743f73b69ce5d43ab18043699d1d81/protobuf-6.33.0-cp310-abi3-win_amd64.whl", hash = "sha256:9a031d10f703f03768f2743a1c403af050b6ae1f3480e9c140f39c45f81b13ee", size = 436882, upload-time = "2025-10-15T20:39:42.841Z" }, + { url = "https://files.pythonhosted.org/packages/e1/a9/b6eee662a6951b9c3640e8e452ab3e09f117d99fc10baa32d1581a0d4099/protobuf-6.33.0-cp39-abi3-macosx_10_9_universal2.whl", hash = "sha256:905b07a65f1a4b72412314082c7dbfae91a9e8b68a0cc1577515f8df58ecf455", size = 427521, upload-time = "2025-10-15T20:39:43.803Z" }, + { url = "https://files.pythonhosted.org/packages/10/35/16d31e0f92c6d2f0e77c2a3ba93185130ea13053dd16200a57434c882f2b/protobuf-6.33.0-cp39-abi3-manylinux2014_aarch64.whl", hash = "sha256:e0697ece353e6239b90ee43a9231318302ad8353c70e6e45499fa52396debf90", size = 324445, upload-time = "2025-10-15T20:39:44.932Z" }, + { url = "https://files.pythonhosted.org/packages/e6/eb/2a981a13e35cda8b75b5585aaffae2eb904f8f351bdd3870769692acbd8a/protobuf-6.33.0-cp39-abi3-manylinux2014_s390x.whl", hash = "sha256:e0a1715e4f27355afd9570f3ea369735afc853a6c3951a6afe1f80d8569ad298", size = 339159, upload-time = "2025-10-15T20:39:46.186Z" }, + { url = "https://files.pythonhosted.org/packages/21/51/0b1cbad62074439b867b4e04cc09b93f6699d78fd191bed2bbb44562e077/protobuf-6.33.0-cp39-abi3-manylinux2014_x86_64.whl", hash = "sha256:35be49fd3f4fefa4e6e2aacc35e8b837d6703c37a2168a55ac21e9b1bc7559ef", size = 323172, upload-time = "2025-10-15T20:39:47.465Z" }, + { url = "https://files.pythonhosted.org/packages/57/33/fbe61bbe91a656619f107b9dfd84b16e1438766bd62157b8d1c1214491fd/protobuf-6.33.0-cp39-cp39-win32.whl", hash = "sha256:cd33a8e38ea3e39df66e1bbc462b076d6e5ba3a4ebbde58219d777223a7873d3", size = 425690, upload-time = "2025-10-15T20:39:48.909Z" }, + { url = "https://files.pythonhosted.org/packages/2c/e4/ccc4814ad9d12fa404f7e5ce1983a2403644b0ed2588678c762b7a26ed92/protobuf-6.33.0-cp39-cp39-win_amd64.whl", hash = "sha256:c963e86c3655af3a917962c9619e1a6b9670540351d7af9439d06064e3317cc9", size = 436876, upload-time = "2025-10-15T20:39:50.009Z" }, + { url = "https://files.pythonhosted.org/packages/07/d1/0a28c21707807c6aacd5dc9c3704b2aa1effbf37adebd8caeaf68b17a636/protobuf-6.33.0-py3-none-any.whl", hash = "sha256:25c9e1963c6734448ea2d308cfa610e692b801304ba0908d7bfa564ac5132995", size = 170477, upload-time = "2025-10-15T20:39:51.311Z" }, +] + +[[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 = "pygments" +version = "2.19.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/b0/77/a5b8c569bf593b0140bde72ea885a803b82086995367bf2037de0159d924/pygments-2.19.2.tar.gz", hash = "sha256:636cb2477cec7f8952536970bc533bc43743542f70392ae026374600add5b887", size = 4968631, upload-time = "2025-06-21T13:39:12.283Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217, upload-time = "2025-06-21T13:39:07.939Z" }, +] + +[[package]] +name = "pytest" +version = "8.3.5" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "colorama", marker = "python_full_version < '3.9' and sys_platform == 'win32'" }, + { name = "exceptiongroup", marker = "python_full_version < '3.9'" }, + { name = "iniconfig", version = "2.1.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "packaging", marker = "python_full_version < '3.9'" }, + { name = "pluggy", version = "1.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "tomli", marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ae/3c/c9d525a414d506893f0cd8a8d0de7706446213181570cdbd766691164e40/pytest-8.3.5.tar.gz", hash = "sha256:f4efe70cc14e511565ac476b57c279e12a855b11f48f212af1080ef2263d3845", size = 1450891, upload-time = "2025-03-02T12:54:54.503Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/30/3d/64ad57c803f1fa1e963a7946b6e0fea4a70df53c1a7fed304586539c2bac/pytest-8.3.5-py3-none-any.whl", hash = "sha256:c69214aa47deac29fad6c2a4f590b9c4a9fdb16a403176fe154b79c0b4d4d820", size = 343634, upload-time = "2025-03-02T12:54:52.069Z" }, +] + +[[package]] +name = "pytest" +version = "8.4.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "colorama", marker = "python_full_version >= '3.9' and sys_platform == 'win32'" }, + { name = "exceptiongroup", marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, + { name = "iniconfig", version = "2.1.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version == '3.9.*'" }, + { name = "iniconfig", version = "2.3.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.10'" }, + { name = "packaging", marker = "python_full_version >= '3.9'" }, + { name = "pluggy", version = "1.6.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "pygments", marker = "python_full_version >= '3.9'" }, + { name = "tomli", marker = "python_full_version >= '3.9' and python_full_version < '3.11'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/a3/5c/00a0e072241553e1a7496d638deababa67c5058571567b92a7eaa258397c/pytest-8.4.2.tar.gz", hash = "sha256:86c0d0b93306b961d58d62a4db4879f27fe25513d4b969df351abdddb3c30e01", size = 1519618, upload-time = "2025-09-04T14:34:22.711Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a8/a4/20da314d277121d6534b3a980b29035dcd51e6744bd79075a6ce8fa4eb8d/pytest-8.4.2-py3-none-any.whl", hash = "sha256:872f880de3fc3a5bdc88a11b39c9710c3497a547cfa9320bc3c5e62fbf272e79", size = 365750, upload-time = "2025-09-04T14:34:20.226Z" }, +] + +[[package]] +name = "python-dotenv" +version = "1.0.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/bc/57/e84d88dfe0aec03b7a2d4327012c1627ab5f03652216c63d49846d7a6c58/python-dotenv-1.0.1.tar.gz", hash = "sha256:e324ee90a023d808f1959c46bcbc04446a10ced277783dc6ee09987c37ec10ca", size = 39115, upload-time = "2024-01-23T06:33:00.505Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6a/3e/b68c118422ec867fa7ab88444e1274aa40681c606d59ac27de5a5588f082/python_dotenv-1.0.1-py3-none-any.whl", hash = "sha256:f7b63ef50f1b690dddf550d03497b66d609393b40b564ed0d674909a68ebf16a", size = 19863, upload-time = "2024-01-23T06:32:58.246Z" }, +] + +[[package]] +name = "python-dotenv" +version = "1.1.1" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/f6/b0/4bc07ccd3572a2f9df7e6782f52b0c6c90dcbb803ac4a167702d7d0dfe1e/python_dotenv-1.1.1.tar.gz", hash = "sha256:a8a6399716257f45be6a007360200409fce5cda2661e3dec71d23dc15f6189ab", size = 41978, upload-time = "2025-06-24T04:21:07.341Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/5f/ed/539768cf28c661b5b068d66d96a2f155c4971a5d55684a514c1a0e0dec2f/python_dotenv-1.1.1-py3-none-any.whl", hash = "sha256:31f23644fe2602f88ff55e1f5c79ba497e01224ee7737937930c448e4d0e24dc", size = 20556, upload-time = "2025-06-24T04:21:06.073Z" }, +] + +[[package]] +name = "pytokens" +version = "0.2.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/d4/c2/dbadcdddb412a267585459142bfd7cc241e6276db69339353ae6e241ab2b/pytokens-0.2.0.tar.gz", hash = "sha256:532d6421364e5869ea57a9523bf385f02586d4662acbcc0342afd69511b4dd43", size = 15368, upload-time = "2025-10-15T08:02:42.738Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/89/5a/c269ea6b348b6f2c32686635df89f32dbe05df1088dd4579302a6f8f99af/pytokens-0.2.0-py3-none-any.whl", hash = "sha256:74d4b318c67f4295c13782ddd9abcb7e297ec5630ad060eb90abf7ebbefe59f8", size = 12038, upload-time = "2025-10-15T08:02:41.694Z" }, +] + +[[package]] +name = "requests" +version = "2.32.4" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "certifi", marker = "python_full_version < '3.9'" }, + { name = "charset-normalizer", marker = "python_full_version < '3.9'" }, + { name = "idna", marker = "python_full_version < '3.9'" }, + { name = "urllib3", version = "2.2.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/e1/0a/929373653770d8a0d7ea76c37de6e41f11eb07559b103b1c02cafb3f7cf8/requests-2.32.4.tar.gz", hash = "sha256:27d0316682c8a29834d3264820024b62a36942083d52caf2f14c0591336d3422", size = 135258, upload-time = "2025-06-09T16:43:07.34Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/7c/e4/56027c4a6b4ae70ca9de302488c5ca95ad4a39e190093d6c1a8ace08341b/requests-2.32.4-py3-none-any.whl", hash = "sha256:27babd3cda2a6d50b30443204ee89830707d396671944c998b5975b031ac2b2c", size = 64847, upload-time = "2025-06-09T16:43:05.728Z" }, +] + +[[package]] +name = "requests" +version = "2.32.5" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "certifi", marker = "python_full_version >= '3.9'" }, + { name = "charset-normalizer", marker = "python_full_version >= '3.9'" }, + { name = "idna", marker = "python_full_version >= '3.9'" }, + { name = "urllib3", version = "2.5.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c9/74/b3ff8e6c8446842c3f5c837e9c3dfcfe2018ea6ecef224c710c85ef728f4/requests-2.32.5.tar.gz", hash = "sha256:dbba0bac56e100853db0ea71b82b4dfd5fe2bf6d3754a8893c3af500cec7d7cf", size = 134517, upload-time = "2025-08-18T20:46:02.573Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/db/4254e3eabe8020b458f1a747140d32277ec7a271daf1d235b70dc0b4e6e3/requests-2.32.5-py3-none-any.whl", hash = "sha256:2462f94637a34fd532264295e186976db0f5d453d1cdd31473c85a6a161affb6", size = 64738, upload-time = "2025-08-18T20:46:00.542Z" }, +] + +[[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 = "ruff" +version = "0.14.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ee/34/8218a19b2055b80601e8fd201ec723c74c7fe1ca06d525a43ed07b6d8e85/ruff-0.14.2.tar.gz", hash = "sha256:98da787668f239313d9c902ca7c523fe11b8ec3f39345553a51b25abc4629c96", size = 5539663, upload-time = "2025-10-23T19:37:00.956Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/16/dd/23eb2db5ad9acae7c845700493b72d3ae214dce0b226f27df89216110f2b/ruff-0.14.2-py3-none-linux_armv6l.whl", hash = "sha256:7cbe4e593505bdec5884c2d0a4d791a90301bc23e49a6b1eb642dd85ef9c64f1", size = 12533390, upload-time = "2025-10-23T19:36:18.044Z" }, + { url = "https://files.pythonhosted.org/packages/5a/8c/5f9acff43ddcf3f85130d0146d0477e28ccecc495f9f684f8f7119b74c0d/ruff-0.14.2-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:8d54b561729cee92f8d89c316ad7a3f9705533f5903b042399b6ae0ddfc62e11", size = 12887187, upload-time = "2025-10-23T19:36:22.664Z" }, + { url = "https://files.pythonhosted.org/packages/99/fa/047646491479074029665022e9f3dc6f0515797f40a4b6014ea8474c539d/ruff-0.14.2-py3-none-macosx_11_0_arm64.whl", hash = "sha256:5c8753dfa44ebb2cde10ce5b4d2ef55a41fb9d9b16732a2c5df64620dbda44a3", size = 11925177, upload-time = "2025-10-23T19:36:24.778Z" }, + { url = "https://files.pythonhosted.org/packages/15/8b/c44cf7fe6e59ab24a9d939493a11030b503bdc2a16622cede8b7b1df0114/ruff-0.14.2-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3d0bbeffb8d9f4fccf7b5198d566d0bad99a9cb622f1fc3467af96cb8773c9e3", size = 12358285, upload-time = "2025-10-23T19:36:26.979Z" }, + { url = "https://files.pythonhosted.org/packages/45/01/47701b26254267ef40369aea3acb62a7b23e921c27372d127e0f3af48092/ruff-0.14.2-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:7047f0c5a713a401e43a88d36843d9c83a19c584e63d664474675620aaa634a8", size = 12303832, upload-time = "2025-10-23T19:36:29.192Z" }, + { url = "https://files.pythonhosted.org/packages/2d/5c/ae7244ca4fbdf2bee9d6405dcd5bc6ae51ee1df66eb7a9884b77b8af856d/ruff-0.14.2-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3bf8d2f9aa1602599217d82e8e0af7fd33e5878c4d98f37906b7c93f46f9a839", size = 13036995, upload-time = "2025-10-23T19:36:31.861Z" }, + { url = "https://files.pythonhosted.org/packages/27/4c/0860a79ce6fd4c709ac01173f76f929d53f59748d0dcdd662519835dae43/ruff-0.14.2-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:1c505b389e19c57a317cf4b42db824e2fca96ffb3d86766c1c9f8b96d32048a7", size = 14512649, upload-time = "2025-10-23T19:36:33.915Z" }, + { url = "https://files.pythonhosted.org/packages/7f/7f/d365de998069720a3abfc250ddd876fc4b81a403a766c74ff9bde15b5378/ruff-0.14.2-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a307fc45ebd887b3f26b36d9326bb70bf69b01561950cdcc6c0bdf7bb8e0f7cc", size = 14088182, upload-time = "2025-10-23T19:36:36.983Z" }, + { url = "https://files.pythonhosted.org/packages/6c/ea/d8e3e6b209162000a7be1faa41b0a0c16a133010311edc3329753cc6596a/ruff-0.14.2-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:61ae91a32c853172f832c2f40bd05fd69f491db7289fb85a9b941ebdd549781a", size = 13599516, upload-time = "2025-10-23T19:36:39.208Z" }, + { url = "https://files.pythonhosted.org/packages/fa/ea/c7810322086db68989fb20a8d5221dd3b79e49e396b01badca07b433ab45/ruff-0.14.2-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1967e40286f63ee23c615e8e7e98098dedc7301568bd88991f6e544d8ae096", size = 13272690, upload-time = "2025-10-23T19:36:41.453Z" }, + { url = "https://files.pythonhosted.org/packages/a9/39/10b05acf8c45786ef501d454e00937e1b97964f846bf28883d1f9619928a/ruff-0.14.2-py3-none-manylinux_2_31_riscv64.whl", hash = "sha256:2877f02119cdebf52a632d743a2e302dea422bfae152ebe2f193d3285a3a65df", size = 13496497, upload-time = "2025-10-23T19:36:43.61Z" }, + { url = "https://files.pythonhosted.org/packages/59/a1/1f25f8301e13751c30895092485fada29076e5e14264bdacc37202e85d24/ruff-0.14.2-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:e681c5bc777de5af898decdcb6ba3321d0d466f4cb43c3e7cc2c3b4e7b843a05", size = 12266116, upload-time = "2025-10-23T19:36:45.625Z" }, + { url = "https://files.pythonhosted.org/packages/5c/fa/0029bfc9ce16ae78164e6923ef392e5f173b793b26cc39aa1d8b366cf9dc/ruff-0.14.2-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:e21be42d72e224736f0c992cdb9959a2fa53c7e943b97ef5d081e13170e3ffc5", size = 12281345, upload-time = "2025-10-23T19:36:47.618Z" }, + { url = "https://files.pythonhosted.org/packages/a5/ab/ece7baa3c0f29b7683be868c024f0838770c16607bea6852e46b202f1ff6/ruff-0.14.2-py3-none-musllinux_1_2_i686.whl", hash = "sha256:b8264016f6f209fac16262882dbebf3f8be1629777cf0f37e7aff071b3e9b92e", size = 12629296, upload-time = "2025-10-23T19:36:49.789Z" }, + { url = "https://files.pythonhosted.org/packages/a4/7f/638f54b43f3d4e48c6a68062794e5b367ddac778051806b9e235dfb7aa81/ruff-0.14.2-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:5ca36b4cb4db3067a3b24444463ceea5565ea78b95fe9a07ca7cb7fd16948770", size = 13371610, upload-time = "2025-10-23T19:36:51.882Z" }, + { url = "https://files.pythonhosted.org/packages/8d/35/3654a973ebe5b32e1fd4a08ed2d46755af7267da7ac710d97420d7b8657d/ruff-0.14.2-py3-none-win32.whl", hash = "sha256:41775927d287685e08f48d8eb3f765625ab0b7042cc9377e20e64f4eb0056ee9", size = 12415318, upload-time = "2025-10-23T19:36:53.961Z" }, + { url = "https://files.pythonhosted.org/packages/71/30/3758bcf9e0b6a4193a6f51abf84254aba00887dfa8c20aba18aa366c5f57/ruff-0.14.2-py3-none-win_amd64.whl", hash = "sha256:0df3424aa5c3c08b34ed8ce099df1021e3adaca6e90229273496b839e5a7e1af", size = 13565279, upload-time = "2025-10-23T19:36:56.578Z" }, + { url = "https://files.pythonhosted.org/packages/2e/5d/aa883766f8ef9ffbe6aa24f7192fb71632f31a30e77eb39aa2b0dc4290ac/ruff-0.14.2-py3-none-win_arm64.whl", hash = "sha256:ea9d635e83ba21569fbacda7e78afbfeb94911c9434aff06192d9bc23fd5495a", size = 12554956, upload-time = "2025-10-23T19:36:58.714Z" }, +] + +[[package]] +name = "slack-bolt" +version = "1.26.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "slack-sdk" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/c3/14/0f490731fbfc95b5711e8124b30bb6e2a4be5edad22256891adad66f8b79/slack_bolt-1.26.0.tar.gz", hash = "sha256:b0b806b9dcf009ee50172830c1d170e231cd873c5b819703bbcdc59a0fe5ff3e", size = 129915, upload-time = "2025-10-06T23:41:51.708Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/fa/77/57aff95f88f2f1a959088ff29c45ceaf8dcad540e9966b647d6942a007f0/slack_bolt-1.26.0-py2.py3-none-any.whl", hash = "sha256:d8386ecb27aaa487c1a5e4b43a4125f532100fc3a26e49dd2a66f5837ff2e3be", size = 230084, upload-time = "2025-10-06T23:41:50.118Z" }, +] + +[[package]] +name = "slack-sdk" +version = "3.37.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/8e/c2/0a174a155623d7dc3ed4d1360cdf755590acdc2c3fc9ce0d2340f468909f/slack_sdk-3.37.0.tar.gz", hash = "sha256:242d6cffbd9e843af807487ff04853189b812081aeaa22f90a8f159f20220ed9", size = 241612, upload-time = "2025-10-06T23:07:20.856Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/07/fd/a502ee24d8c7d12a8f749878ae0949b8eeb50aeac22dc5a613d417a256d0/slack_sdk-3.37.0-py2.py3-none-any.whl", hash = "sha256:e108a0836eafda74d8a95e76c12c2bcb010e645d504d8497451e4c7ebb229c87", size = 302751, upload-time = "2025-10-06T23:07:19.542Z" }, +] + +[[package]] +name = "sniffio" +version = "1.3.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372, upload-time = "2024-02-25T23:20:04.057Z" } +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 = "tomli" +version = "2.3.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/52/ed/3f73f72945444548f33eba9a87fc7a6e969915e7b1acc8260b30e1f76a2f/tomli-2.3.0.tar.gz", hash = "sha256:64be704a875d2a59753d80ee8a533c3fe183e3f06807ff7dc2232938ccb01549", size = 17392, upload-time = "2025-10-08T22:01:47.119Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b3/2e/299f62b401438d5fe1624119c723f5d877acc86a4c2492da405626665f12/tomli-2.3.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:88bd15eb972f3664f5ed4b57c1634a97153b4bac4479dcb6a495f41921eb7f45", size = 153236, upload-time = "2025-10-08T22:01:00.137Z" }, + { url = "https://files.pythonhosted.org/packages/86/7f/d8fffe6a7aefdb61bced88fcb5e280cfd71e08939da5894161bd71bea022/tomli-2.3.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:883b1c0d6398a6a9d29b508c331fa56adbcdff647f6ace4dfca0f50e90dfd0ba", size = 148084, upload-time = "2025-10-08T22:01:01.63Z" }, + { url = "https://files.pythonhosted.org/packages/47/5c/24935fb6a2ee63e86d80e4d3b58b222dafaf438c416752c8b58537c8b89a/tomli-2.3.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:d1381caf13ab9f300e30dd8feadb3de072aeb86f1d34a8569453ff32a7dea4bf", size = 234832, upload-time = "2025-10-08T22:01:02.543Z" }, + { url = "https://files.pythonhosted.org/packages/89/da/75dfd804fc11e6612846758a23f13271b76d577e299592b4371a4ca4cd09/tomli-2.3.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a0e285d2649b78c0d9027570d4da3425bdb49830a6156121360b3f8511ea3441", size = 242052, upload-time = "2025-10-08T22:01:03.836Z" }, + { url = "https://files.pythonhosted.org/packages/70/8c/f48ac899f7b3ca7eb13af73bacbc93aec37f9c954df3c08ad96991c8c373/tomli-2.3.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:0a154a9ae14bfcf5d8917a59b51ffd5a3ac1fd149b71b47a3a104ca4edcfa845", size = 239555, upload-time = "2025-10-08T22:01:04.834Z" }, + { url = "https://files.pythonhosted.org/packages/ba/28/72f8afd73f1d0e7829bfc093f4cb98ce0a40ffc0cc997009ee1ed94ba705/tomli-2.3.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:74bf8464ff93e413514fefd2be591c3b0b23231a77f901db1eb30d6f712fc42c", size = 245128, upload-time = "2025-10-08T22:01:05.84Z" }, + { url = "https://files.pythonhosted.org/packages/b6/eb/a7679c8ac85208706d27436e8d421dfa39d4c914dcf5fa8083a9305f58d9/tomli-2.3.0-cp311-cp311-win32.whl", hash = "sha256:00b5f5d95bbfc7d12f91ad8c593a1659b6387b43f054104cda404be6bda62456", size = 96445, upload-time = "2025-10-08T22:01:06.896Z" }, + { url = "https://files.pythonhosted.org/packages/0a/fe/3d3420c4cb1ad9cb462fb52967080575f15898da97e21cb6f1361d505383/tomli-2.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:4dc4ce8483a5d429ab602f111a93a6ab1ed425eae3122032db7e9acf449451be", size = 107165, upload-time = "2025-10-08T22:01:08.107Z" }, + { url = "https://files.pythonhosted.org/packages/ff/b7/40f36368fcabc518bb11c8f06379a0fd631985046c038aca08c6d6a43c6e/tomli-2.3.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:d7d86942e56ded512a594786a5ba0a5e521d02529b3826e7761a05138341a2ac", size = 154891, upload-time = "2025-10-08T22:01:09.082Z" }, + { url = "https://files.pythonhosted.org/packages/f9/3f/d9dd692199e3b3aab2e4e4dd948abd0f790d9ded8cd10cbaae276a898434/tomli-2.3.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:73ee0b47d4dad1c5e996e3cd33b8a76a50167ae5f96a2607cbe8cc773506ab22", size = 148796, upload-time = "2025-10-08T22:01:10.266Z" }, + { url = "https://files.pythonhosted.org/packages/60/83/59bff4996c2cf9f9387a0f5a3394629c7efa5ef16142076a23a90f1955fa/tomli-2.3.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:792262b94d5d0a466afb5bc63c7daa9d75520110971ee269152083270998316f", size = 242121, upload-time = "2025-10-08T22:01:11.332Z" }, + { url = "https://files.pythonhosted.org/packages/45/e5/7c5119ff39de8693d6baab6c0b6dcb556d192c165596e9fc231ea1052041/tomli-2.3.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4f195fe57ecceac95a66a75ac24d9d5fbc98ef0962e09b2eddec5d39375aae52", size = 250070, upload-time = "2025-10-08T22:01:12.498Z" }, + { url = "https://files.pythonhosted.org/packages/45/12/ad5126d3a278f27e6701abde51d342aa78d06e27ce2bb596a01f7709a5a2/tomli-2.3.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:e31d432427dcbf4d86958c184b9bfd1e96b5b71f8eb17e6d02531f434fd335b8", size = 245859, upload-time = "2025-10-08T22:01:13.551Z" }, + { url = "https://files.pythonhosted.org/packages/fb/a1/4d6865da6a71c603cfe6ad0e6556c73c76548557a8d658f9e3b142df245f/tomli-2.3.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:7b0882799624980785240ab732537fcfc372601015c00f7fc367c55308c186f6", size = 250296, upload-time = "2025-10-08T22:01:14.614Z" }, + { url = "https://files.pythonhosted.org/packages/a0/b7/a7a7042715d55c9ba6e8b196d65d2cb662578b4d8cd17d882d45322b0d78/tomli-2.3.0-cp312-cp312-win32.whl", hash = "sha256:ff72b71b5d10d22ecb084d345fc26f42b5143c5533db5e2eaba7d2d335358876", size = 97124, upload-time = "2025-10-08T22:01:15.629Z" }, + { url = "https://files.pythonhosted.org/packages/06/1e/f22f100db15a68b520664eb3328fb0ae4e90530887928558112c8d1f4515/tomli-2.3.0-cp312-cp312-win_amd64.whl", hash = "sha256:1cb4ed918939151a03f33d4242ccd0aa5f11b3547d0cf30f7c74a408a5b99878", size = 107698, upload-time = "2025-10-08T22:01:16.51Z" }, + { url = "https://files.pythonhosted.org/packages/89/48/06ee6eabe4fdd9ecd48bf488f4ac783844fd777f547b8d1b61c11939974e/tomli-2.3.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:5192f562738228945d7b13d4930baffda67b69425a7f0da96d360b0a3888136b", size = 154819, upload-time = "2025-10-08T22:01:17.964Z" }, + { url = "https://files.pythonhosted.org/packages/f1/01/88793757d54d8937015c75dcdfb673c65471945f6be98e6a0410fba167ed/tomli-2.3.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:be71c93a63d738597996be9528f4abe628d1adf5e6eb11607bc8fe1a510b5dae", size = 148766, upload-time = "2025-10-08T22:01:18.959Z" }, + { url = "https://files.pythonhosted.org/packages/42/17/5e2c956f0144b812e7e107f94f1cc54af734eb17b5191c0bbfb72de5e93e/tomli-2.3.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c4665508bcbac83a31ff8ab08f424b665200c0e1e645d2bd9ab3d3e557b6185b", size = 240771, upload-time = "2025-10-08T22:01:20.106Z" }, + { url = "https://files.pythonhosted.org/packages/d5/f4/0fbd014909748706c01d16824eadb0307115f9562a15cbb012cd9b3512c5/tomli-2.3.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4021923f97266babc6ccab9f5068642a0095faa0a51a246a6a02fccbb3514eaf", size = 248586, upload-time = "2025-10-08T22:01:21.164Z" }, + { url = "https://files.pythonhosted.org/packages/30/77/fed85e114bde5e81ecf9bc5da0cc69f2914b38f4708c80ae67d0c10180c5/tomli-2.3.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:a4ea38c40145a357d513bffad0ed869f13c1773716cf71ccaa83b0fa0cc4e42f", size = 244792, upload-time = "2025-10-08T22:01:22.417Z" }, + { url = "https://files.pythonhosted.org/packages/55/92/afed3d497f7c186dc71e6ee6d4fcb0acfa5f7d0a1a2878f8beae379ae0cc/tomli-2.3.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:ad805ea85eda330dbad64c7ea7a4556259665bdf9d2672f5dccc740eb9d3ca05", size = 248909, upload-time = "2025-10-08T22:01:23.859Z" }, + { url = "https://files.pythonhosted.org/packages/f8/84/ef50c51b5a9472e7265ce1ffc7f24cd4023d289e109f669bdb1553f6a7c2/tomli-2.3.0-cp313-cp313-win32.whl", hash = "sha256:97d5eec30149fd3294270e889b4234023f2c69747e555a27bd708828353ab606", size = 96946, upload-time = "2025-10-08T22:01:24.893Z" }, + { url = "https://files.pythonhosted.org/packages/b2/b7/718cd1da0884f281f95ccfa3a6cc572d30053cba64603f79d431d3c9b61b/tomli-2.3.0-cp313-cp313-win_amd64.whl", hash = "sha256:0c95ca56fbe89e065c6ead5b593ee64b84a26fca063b5d71a1122bf26e533999", size = 107705, upload-time = "2025-10-08T22:01:26.153Z" }, + { url = "https://files.pythonhosted.org/packages/19/94/aeafa14a52e16163008060506fcb6aa1949d13548d13752171a755c65611/tomli-2.3.0-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:cebc6fe843e0733ee827a282aca4999b596241195f43b4cc371d64fc6639da9e", size = 154244, upload-time = "2025-10-08T22:01:27.06Z" }, + { url = "https://files.pythonhosted.org/packages/db/e4/1e58409aa78eefa47ccd19779fc6f36787edbe7d4cd330eeeedb33a4515b/tomli-2.3.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:4c2ef0244c75aba9355561272009d934953817c49f47d768070c3c94355c2aa3", size = 148637, upload-time = "2025-10-08T22:01:28.059Z" }, + { url = "https://files.pythonhosted.org/packages/26/b6/d1eccb62f665e44359226811064596dd6a366ea1f985839c566cd61525ae/tomli-2.3.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:c22a8bf253bacc0cf11f35ad9808b6cb75ada2631c2d97c971122583b129afbc", size = 241925, upload-time = "2025-10-08T22:01:29.066Z" }, + { url = "https://files.pythonhosted.org/packages/70/91/7cdab9a03e6d3d2bb11beae108da5bdc1c34bdeb06e21163482544ddcc90/tomli-2.3.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0eea8cc5c5e9f89c9b90c4896a8deefc74f518db5927d0e0e8d4a80953d774d0", size = 249045, upload-time = "2025-10-08T22:01:31.98Z" }, + { url = "https://files.pythonhosted.org/packages/15/1b/8c26874ed1f6e4f1fcfeb868db8a794cbe9f227299402db58cfcc858766c/tomli-2.3.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:b74a0e59ec5d15127acdabd75ea17726ac4c5178ae51b85bfe39c4f8a278e879", size = 245835, upload-time = "2025-10-08T22:01:32.989Z" }, + { url = "https://files.pythonhosted.org/packages/fd/42/8e3c6a9a4b1a1360c1a2a39f0b972cef2cc9ebd56025168c4137192a9321/tomli-2.3.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:b5870b50c9db823c595983571d1296a6ff3e1b88f734a4c8f6fc6188397de005", size = 253109, upload-time = "2025-10-08T22:01:34.052Z" }, + { url = "https://files.pythonhosted.org/packages/22/0c/b4da635000a71b5f80130937eeac12e686eefb376b8dee113b4a582bba42/tomli-2.3.0-cp314-cp314-win32.whl", hash = "sha256:feb0dacc61170ed7ab602d3d972a58f14ee3ee60494292d384649a3dc38ef463", size = 97930, upload-time = "2025-10-08T22:01:35.082Z" }, + { url = "https://files.pythonhosted.org/packages/b9/74/cb1abc870a418ae99cd5c9547d6bce30701a954e0e721821df483ef7223c/tomli-2.3.0-cp314-cp314-win_amd64.whl", hash = "sha256:b273fcbd7fc64dc3600c098e39136522650c49bca95df2d11cf3b626422392c8", size = 107964, upload-time = "2025-10-08T22:01:36.057Z" }, + { url = "https://files.pythonhosted.org/packages/54/78/5c46fff6432a712af9f792944f4fcd7067d8823157949f4e40c56b8b3c83/tomli-2.3.0-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:940d56ee0410fa17ee1f12b817b37a4d4e4dc4d27340863cc67236c74f582e77", size = 163065, upload-time = "2025-10-08T22:01:37.27Z" }, + { url = "https://files.pythonhosted.org/packages/39/67/f85d9bd23182f45eca8939cd2bc7050e1f90c41f4a2ecbbd5963a1d1c486/tomli-2.3.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:f85209946d1fe94416debbb88d00eb92ce9cd5266775424ff81bc959e001acaf", size = 159088, upload-time = "2025-10-08T22:01:38.235Z" }, + { url = "https://files.pythonhosted.org/packages/26/5a/4b546a0405b9cc0659b399f12b6adb750757baf04250b148d3c5059fc4eb/tomli-2.3.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:a56212bdcce682e56b0aaf79e869ba5d15a6163f88d5451cbde388d48b13f530", size = 268193, upload-time = "2025-10-08T22:01:39.712Z" }, + { url = "https://files.pythonhosted.org/packages/42/4f/2c12a72ae22cf7b59a7fe75b3465b7aba40ea9145d026ba41cb382075b0e/tomli-2.3.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:c5f3ffd1e098dfc032d4d3af5c0ac64f6d286d98bc148698356847b80fa4de1b", size = 275488, upload-time = "2025-10-08T22:01:40.773Z" }, + { url = "https://files.pythonhosted.org/packages/92/04/a038d65dbe160c3aa5a624e93ad98111090f6804027d474ba9c37c8ae186/tomli-2.3.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:5e01decd096b1530d97d5d85cb4dff4af2d8347bd35686654a004f8dea20fc67", size = 272669, upload-time = "2025-10-08T22:01:41.824Z" }, + { url = "https://files.pythonhosted.org/packages/be/2f/8b7c60a9d1612a7cbc39ffcca4f21a73bf368a80fc25bccf8253e2563267/tomli-2.3.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:8a35dd0e643bb2610f156cca8db95d213a90015c11fee76c946aa62b7ae7e02f", size = 279709, upload-time = "2025-10-08T22:01:43.177Z" }, + { url = "https://files.pythonhosted.org/packages/7e/46/cc36c679f09f27ded940281c38607716c86cf8ba4a518d524e349c8b4874/tomli-2.3.0-cp314-cp314t-win32.whl", hash = "sha256:a1f7f282fe248311650081faafa5f4732bdbfef5d45fe3f2e702fbc6f2d496e0", size = 107563, upload-time = "2025-10-08T22:01:44.233Z" }, + { url = "https://files.pythonhosted.org/packages/84/ff/426ca8683cf7b753614480484f6437f568fd2fda2edbdf57a2d3d8b27a0b/tomli-2.3.0-cp314-cp314t-win_amd64.whl", hash = "sha256:70a251f8d4ba2d9ac2542eecf008b3c8a9fc5c3f9f02c56a9d7952612be2fdba", size = 119756, upload-time = "2025-10-08T22:01:45.234Z" }, + { url = "https://files.pythonhosted.org/packages/77/b8/0135fadc89e73be292b473cb820b4f5a08197779206b33191e801feeae40/tomli-2.3.0-py3-none-any.whl", hash = "sha256:e95b1af3c5b07d9e643909b5abbec77cd9f1217e6d0bca72b0234736b9fb1f1b", size = 14408, upload-time = "2025-10-08T22:01:46.04Z" }, +] + +[[package]] +name = "typing-extensions" +version = "4.13.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/f6/37/23083fcd6e35492953e8d2aaaa68b860eb422b34627b13f2ce3eb6106061/typing_extensions-4.13.2.tar.gz", hash = "sha256:e6c81219bd689f51865d9e372991c540bda33a0379d5573cddb9a3a23f7caaef", size = 106967, upload-time = "2025-04-10T14:19:05.416Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/8b/54/b1ae86c0973cc6f0210b53d508ca3641fb6d0c56823f288d108bc7ab3cc8/typing_extensions-4.13.2-py3-none-any.whl", hash = "sha256:a439e7c04b49fec3e5d3e2beaa21755cadbbdc391694e28ccdd36ca4a1408f8c", size = 45806, upload-time = "2025-04-10T14:19:03.967Z" }, +] + +[[package]] +name = "typing-extensions" +version = "4.15.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/72/94/1a15dd82efb362ac84269196e94cf00f187f7ed21c242792a923cdb1c61f/typing_extensions-4.15.0.tar.gz", hash = "sha256:0cea48d173cc12fa28ecabc3b837ea3cf6f38c6d1136f85cbaaf598984861466", size = 109391, upload-time = "2025-08-25T13:49:26.313Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/18/67/36e9267722cc04a6b9f15c7f3441c2363321a3ea07da7ae0c0707beb2a9c/typing_extensions-4.15.0-py3-none-any.whl", hash = "sha256:f0fa19c6845758ab08074a0cfa8b7aecb71c999ca73d62883bc25cc018c4e548", size = 44614, upload-time = "2025-08-25T13:49:24.86Z" }, +] + +[[package]] +name = "urllib3" +version = "2.2.3" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/ed/63/22ba4ebfe7430b76388e7cd448d5478814d3032121827c12a2cc287e2260/urllib3-2.2.3.tar.gz", hash = "sha256:e7d814a81dad81e6caf2ec9fdedb284ecc9c73076b62654547cc64ccdcae26e9", size = 300677, upload-time = "2024-09-12T10:52:18.401Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ce/d9/5f4c13cecde62396b0d3fe530a50ccea91e7dfc1ccf0e09c228841bb5ba8/urllib3-2.2.3-py3-none-any.whl", hash = "sha256:ca899ca043dcb1bafa3e262d73aa25c465bfb49e0bd9dd5d59f1d0acba2f8fac", size = 126338, upload-time = "2024-09-12T10:52:16.589Z" }, +] + +[[package]] +name = "urllib3" +version = "2.5.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/15/22/9ee70a2574a4f4599c47dd506532914ce044817c7752a79b6a51286319bc/urllib3-2.5.0.tar.gz", hash = "sha256:3fc47733c7e419d4bc3f6b3dc2b4f890bb743906a30d56ba4a5bfa4bbff92760", size = 393185, upload-time = "2025-06-18T14:07:41.644Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a7/c2/fe1e52489ae3122415c51f387e221dd0773709bad6c6cdaa599e8a2c5185/urllib3-2.5.0-py3-none-any.whl", hash = "sha256:e6b01673c0fa6a13e374b50871808eb3bf7046c4b125b216f6bf1cc604cff0dc", size = 129795, upload-time = "2025-06-18T14:07:40.39Z" }, +] + +[[package]] +name = "werkzeug" +version = "3.0.6" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "markupsafe", version = "2.1.5", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/d4/f9/0ba83eaa0df9b9e9d1efeb2ea351d0677c37d41ee5d0f91e98423c7281c9/werkzeug-3.0.6.tar.gz", hash = "sha256:a8dd59d4de28ca70471a34cba79bed5f7ef2e036a76b3ab0835474246eb41f8d", size = 805170, upload-time = "2024-10-25T18:52:31.688Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6c/69/05837f91dfe42109203ffa3e488214ff86a6d68b2ed6c167da6cdc42349b/werkzeug-3.0.6-py3-none-any.whl", hash = "sha256:1bc0c2310d2fbb07b1dd1105eba2f7af72f322e1e455f2f93c993bee8c8a5f17", size = 227979, upload-time = "2024-10-25T18:52:30.129Z" }, +] + +[[package]] +name = "werkzeug" +version = "3.1.3" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "markupsafe", version = "3.0.3", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/9f/69/83029f1f6300c5fb2471d621ab06f6ec6b3324685a2ce0f9777fd4a8b71e/werkzeug-3.1.3.tar.gz", hash = "sha256:60723ce945c19328679790e3282cc758aa4a6040e4bb330f53d30fa546d44746", size = 806925, upload-time = "2024-11-08T15:52:18.093Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/52/24/ab44c871b0f07f491e5d2ad12c9bd7358e527510618cb1b803a88e986db1/werkzeug-3.1.3-py3-none-any.whl", hash = "sha256:54b78bf3716d19a65be4fceccc0d1d7b89e608834989dfae50ea87564639213e", size = 224498, upload-time = "2024-11-08T15:52:16.132Z" }, +] + +[[package]] +name = "yarl" +version = "1.15.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +dependencies = [ + { name = "idna", marker = "python_full_version < '3.9'" }, + { name = "multidict", version = "6.1.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, + { name = "propcache", version = "0.2.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version < '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/06/e1/d5427a061819c9f885f58bb0467d02a523f1aec19f9e5f9c82ce950d90d3/yarl-1.15.2.tar.gz", hash = "sha256:a39c36f4218a5bb668b4f06874d676d35a035ee668e6e7e3538835c703634b84", size = 169318, upload-time = "2024-10-13T18:48:04.311Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/61/f8/6b1bbc6f597d8937ad8661c042aa6bdbbe46a3a6e38e2c04214b9c82e804/yarl-1.15.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:e4ee8b8639070ff246ad3649294336b06db37a94bdea0d09ea491603e0be73b8", size = 136479, upload-time = "2024-10-13T18:44:32.077Z" }, + { url = "https://files.pythonhosted.org/packages/61/e0/973c0d16b1cb710d318b55bd5d019a1ecd161d28670b07d8d9df9a83f51f/yarl-1.15.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:a7cf963a357c5f00cb55b1955df8bbe68d2f2f65de065160a1c26b85a1e44172", size = 88671, upload-time = "2024-10-13T18:44:35.334Z" }, + { url = "https://files.pythonhosted.org/packages/16/df/241cfa1cf33b96da2c8773b76fe3ee58e04cb09ecfe794986ec436ae97dc/yarl-1.15.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:43ebdcc120e2ca679dba01a779333a8ea76b50547b55e812b8b92818d604662c", size = 86578, upload-time = "2024-10-13T18:44:37.58Z" }, + { url = "https://files.pythonhosted.org/packages/02/a4/ee2941d1f93600d921954a0850e20581159772304e7de49f60588e9128a2/yarl-1.15.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3433da95b51a75692dcf6cc8117a31410447c75a9a8187888f02ad45c0a86c50", size = 307212, upload-time = "2024-10-13T18:44:39.932Z" }, + { url = "https://files.pythonhosted.org/packages/08/64/2e6561af430b092b21c7a867ae3079f62e1532d3e51fee765fd7a74cef6c/yarl-1.15.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:38d0124fa992dbacd0c48b1b755d3ee0a9f924f427f95b0ef376556a24debf01", size = 321589, upload-time = "2024-10-13T18:44:42.527Z" }, + { url = "https://files.pythonhosted.org/packages/f8/af/056ab318a7117fa70f6ab502ff880e47af973948d1d123aff397cd68499c/yarl-1.15.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ded1b1803151dd0f20a8945508786d57c2f97a50289b16f2629f85433e546d47", size = 319443, upload-time = "2024-10-13T18:44:45.03Z" }, + { url = "https://files.pythonhosted.org/packages/99/d1/051b0bc2c90c9a2618bab10a9a9a61a96ddb28c7c54161a5c97f9e625205/yarl-1.15.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ace4cad790f3bf872c082366c9edd7f8f8f77afe3992b134cfc810332206884f", size = 310324, upload-time = "2024-10-13T18:44:47.675Z" }, + { url = "https://files.pythonhosted.org/packages/23/1b/16df55016f9ac18457afda165031086bce240d8bcf494501fb1164368617/yarl-1.15.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c77494a2f2282d9bbbbcab7c227a4d1b4bb829875c96251f66fb5f3bae4fb053", size = 300428, upload-time = "2024-10-13T18:44:49.431Z" }, + { url = "https://files.pythonhosted.org/packages/83/a5/5188d1c575139a8dfd90d463d56f831a018f41f833cdf39da6bd8a72ee08/yarl-1.15.2-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:b7f227ca6db5a9fda0a2b935a2ea34a7267589ffc63c8045f0e4edb8d8dcf956", size = 307079, upload-time = "2024-10-13T18:44:51.96Z" }, + { url = "https://files.pythonhosted.org/packages/ba/4e/2497f8f2b34d1a261bebdbe00066242eacc9a7dccd4f02ddf0995014290a/yarl-1.15.2-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:31561a5b4d8dbef1559b3600b045607cf804bae040f64b5f5bca77da38084a8a", size = 305835, upload-time = "2024-10-13T18:44:53.83Z" }, + { url = "https://files.pythonhosted.org/packages/91/db/40a347e1f8086e287a53c72dc333198816885bc770e3ecafcf5eaeb59311/yarl-1.15.2-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:3e52474256a7db9dcf3c5f4ca0b300fdea6c21cca0148c8891d03a025649d935", size = 311033, upload-time = "2024-10-13T18:44:56.464Z" }, + { url = "https://files.pythonhosted.org/packages/2f/a6/1500e1e694616c25eed6bf8c1aacc0943f124696d2421a07ae5e9ee101a5/yarl-1.15.2-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:0e1af74a9529a1137c67c887ed9cde62cff53aa4d84a3adbec329f9ec47a3936", size = 326317, upload-time = "2024-10-13T18:44:59.015Z" }, + { url = "https://files.pythonhosted.org/packages/37/db/868d4b59cc76932ce880cc9946cd0ae4ab111a718494a94cb50dd5b67d82/yarl-1.15.2-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:15c87339490100c63472a76d87fe7097a0835c705eb5ae79fd96e343473629ed", size = 324196, upload-time = "2024-10-13T18:45:00.772Z" }, + { url = "https://files.pythonhosted.org/packages/bd/41/b6c917c2fde2601ee0b45c82a0c502dc93e746dea469d3a6d1d0a24749e8/yarl-1.15.2-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:74abb8709ea54cc483c4fb57fb17bb66f8e0f04438cff6ded322074dbd17c7ec", size = 317023, upload-time = "2024-10-13T18:45:03.427Z" }, + { url = "https://files.pythonhosted.org/packages/b0/85/2cde6b656fd83c474f19606af3f7a3e94add8988760c87a101ee603e7b8f/yarl-1.15.2-cp310-cp310-win32.whl", hash = "sha256:ffd591e22b22f9cb48e472529db6a47203c41c2c5911ff0a52e85723196c0d75", size = 78136, upload-time = "2024-10-13T18:45:05.173Z" }, + { url = "https://files.pythonhosted.org/packages/ef/3c/4414901b0588427870002b21d790bd1fad142a9a992a22e5037506d0ed9d/yarl-1.15.2-cp310-cp310-win_amd64.whl", hash = "sha256:1695497bb2a02a6de60064c9f077a4ae9c25c73624e0d43e3aa9d16d983073c2", size = 84231, upload-time = "2024-10-13T18:45:07.622Z" }, + { url = "https://files.pythonhosted.org/packages/4a/59/3ae125c97a2a8571ea16fdf59fcbd288bc169e0005d1af9946a90ea831d9/yarl-1.15.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:9fcda20b2de7042cc35cf911702fa3d8311bd40055a14446c1e62403684afdc5", size = 136492, upload-time = "2024-10-13T18:45:09.962Z" }, + { url = "https://files.pythonhosted.org/packages/f9/2b/efa58f36b582db45b94c15e87803b775eb8a4ca0db558121a272e67f3564/yarl-1.15.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0545de8c688fbbf3088f9e8b801157923be4bf8e7b03e97c2ecd4dfa39e48e0e", size = 88614, upload-time = "2024-10-13T18:45:12.329Z" }, + { url = "https://files.pythonhosted.org/packages/82/69/eb73c0453a2ff53194df485dc7427d54e6cb8d1180fcef53251a8e24d069/yarl-1.15.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:fbda058a9a68bec347962595f50546a8a4a34fd7b0654a7b9697917dc2bf810d", size = 86607, upload-time = "2024-10-13T18:45:13.88Z" }, + { url = "https://files.pythonhosted.org/packages/48/4e/89beaee3a4da0d1c6af1176d738cff415ff2ad3737785ee25382409fe3e3/yarl-1.15.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d1ac2bc069f4a458634c26b101c2341b18da85cb96afe0015990507efec2e417", size = 334077, upload-time = "2024-10-13T18:45:16.217Z" }, + { url = "https://files.pythonhosted.org/packages/da/e8/8fcaa7552093f94c3f327783e2171da0eaa71db0c267510898a575066b0f/yarl-1.15.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:cd126498171f752dd85737ab1544329a4520c53eed3997f9b08aefbafb1cc53b", size = 347365, upload-time = "2024-10-13T18:45:18.812Z" }, + { url = "https://files.pythonhosted.org/packages/be/fa/dc2002f82a89feab13a783d3e6b915a3a2e0e83314d9e3f6d845ee31bfcc/yarl-1.15.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3db817b4e95eb05c362e3b45dafe7144b18603e1211f4a5b36eb9522ecc62bcf", size = 344823, upload-time = "2024-10-13T18:45:20.644Z" }, + { url = "https://files.pythonhosted.org/packages/ae/c8/c4a00fe7f2aa6970c2651df332a14c88f8baaedb2e32d6c3b8c8a003ea74/yarl-1.15.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:076b1ed2ac819933895b1a000904f62d615fe4533a5cf3e052ff9a1da560575c", size = 337132, upload-time = "2024-10-13T18:45:22.487Z" }, + { url = "https://files.pythonhosted.org/packages/07/bf/84125f85f44bf2af03f3cf64e87214b42cd59dcc8a04960d610a9825f4d4/yarl-1.15.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f8cfd847e6b9ecf9f2f2531c8427035f291ec286c0a4944b0a9fce58c6446046", size = 326258, upload-time = "2024-10-13T18:45:25.049Z" }, + { url = "https://files.pythonhosted.org/packages/00/19/73ad8122b2fa73fe22e32c24b82a6c053cf6c73e2f649b73f7ef97bee8d0/yarl-1.15.2-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:32b66be100ac5739065496c74c4b7f3015cef792c3174982809274d7e51b3e04", size = 336212, upload-time = "2024-10-13T18:45:26.808Z" }, + { url = "https://files.pythonhosted.org/packages/39/1d/2fa4337d11f6587e9b7565f84eba549f2921494bc8b10bfe811079acaa70/yarl-1.15.2-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:34a2d76a1984cac04ff8b1bfc939ec9dc0914821264d4a9c8fd0ed6aa8d4cfd2", size = 330397, upload-time = "2024-10-13T18:45:29.112Z" }, + { url = "https://files.pythonhosted.org/packages/39/ab/dce75e06806bcb4305966471ead03ce639d8230f4f52c32bd614d820c044/yarl-1.15.2-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:0afad2cd484908f472c8fe2e8ef499facee54a0a6978be0e0cff67b1254fd747", size = 334985, upload-time = "2024-10-13T18:45:31.709Z" }, + { url = "https://files.pythonhosted.org/packages/c1/98/3f679149347a5e34c952bf8f71a387bc96b3488fae81399a49f8b1a01134/yarl-1.15.2-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:c68e820879ff39992c7f148113b46efcd6ec765a4865581f2902b3c43a5f4bbb", size = 356033, upload-time = "2024-10-13T18:45:34.325Z" }, + { url = "https://files.pythonhosted.org/packages/f7/8c/96546061c19852d0a4b1b07084a58c2e8911db6bcf7838972cff542e09fb/yarl-1.15.2-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:98f68df80ec6ca3015186b2677c208c096d646ef37bbf8b49764ab4a38183931", size = 357710, upload-time = "2024-10-13T18:45:36.216Z" }, + { url = "https://files.pythonhosted.org/packages/01/45/ade6fb3daf689816ebaddb3175c962731edf300425c3254c559b6d0dcc27/yarl-1.15.2-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:3c56ec1eacd0a5d35b8a29f468659c47f4fe61b2cab948ca756c39b7617f0aa5", size = 345532, upload-time = "2024-10-13T18:45:38.123Z" }, + { url = "https://files.pythonhosted.org/packages/e7/d7/8de800d3aecda0e64c43e8fc844f7effc8731a6099fa0c055738a2247504/yarl-1.15.2-cp311-cp311-win32.whl", hash = "sha256:eedc3f247ee7b3808ea07205f3e7d7879bc19ad3e6222195cd5fbf9988853e4d", size = 78250, upload-time = "2024-10-13T18:45:39.908Z" }, + { url = "https://files.pythonhosted.org/packages/3a/6c/69058bbcfb0164f221aa30e0cd1a250f6babb01221e27c95058c51c498ca/yarl-1.15.2-cp311-cp311-win_amd64.whl", hash = "sha256:0ccaa1bc98751fbfcf53dc8dfdb90d96e98838010fc254180dd6707a6e8bb179", size = 84492, upload-time = "2024-10-13T18:45:42.286Z" }, + { url = "https://files.pythonhosted.org/packages/e0/d1/17ff90e7e5b1a0b4ddad847f9ec6a214b87905e3a59d01bff9207ce2253b/yarl-1.15.2-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:82d5161e8cb8f36ec778fd7ac4d740415d84030f5b9ef8fe4da54784a1f46c94", size = 136721, upload-time = "2024-10-13T18:45:43.876Z" }, + { url = "https://files.pythonhosted.org/packages/44/50/a64ca0577aeb9507f4b672f9c833d46cf8f1e042ce2e80c11753b936457d/yarl-1.15.2-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:fa2bea05ff0a8fb4d8124498e00e02398f06d23cdadd0fe027d84a3f7afde31e", size = 88954, upload-time = "2024-10-13T18:45:46.305Z" }, + { url = "https://files.pythonhosted.org/packages/c9/0a/a30d0b02046d4088c1fd32d85d025bd70ceb55f441213dee14d503694f41/yarl-1.15.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:99e12d2bf587b44deb74e0d6170fec37adb489964dbca656ec41a7cd8f2ff178", size = 86692, upload-time = "2024-10-13T18:45:47.992Z" }, + { url = "https://files.pythonhosted.org/packages/06/0b/7613decb8baa26cba840d7ea2074bd3c5e27684cbcb6d06e7840d6c5226c/yarl-1.15.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:243fbbbf003754fe41b5bdf10ce1e7f80bcc70732b5b54222c124d6b4c2ab31c", size = 325762, upload-time = "2024-10-13T18:45:49.69Z" }, + { url = "https://files.pythonhosted.org/packages/97/f5/b8c389a58d1eb08f89341fc1bbcc23a0341f7372185a0a0704dbdadba53a/yarl-1.15.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:856b7f1a7b98a8c31823285786bd566cf06226ac4f38b3ef462f593c608a9bd6", size = 335037, upload-time = "2024-10-13T18:45:51.932Z" }, + { url = "https://files.pythonhosted.org/packages/cb/f9/d89b93a7bb8b66e01bf722dcc6fec15e11946e649e71414fd532b05c4d5d/yarl-1.15.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:553dad9af802a9ad1a6525e7528152a015b85fb8dbf764ebfc755c695f488367", size = 334221, upload-time = "2024-10-13T18:45:54.548Z" }, + { url = "https://files.pythonhosted.org/packages/10/77/1db077601998e0831a540a690dcb0f450c31f64c492e993e2eaadfbc7d31/yarl-1.15.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:30c3ff305f6e06650a761c4393666f77384f1cc6c5c0251965d6bfa5fbc88f7f", size = 330167, upload-time = "2024-10-13T18:45:56.675Z" }, + { url = "https://files.pythonhosted.org/packages/3b/c2/e5b7121662fd758656784fffcff2e411c593ec46dc9ec68e0859a2ffaee3/yarl-1.15.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:353665775be69bbfc6d54c8d134bfc533e332149faeddd631b0bc79df0897f46", size = 317472, upload-time = "2024-10-13T18:45:58.815Z" }, + { url = "https://files.pythonhosted.org/packages/c6/f3/41e366c17e50782651b192ba06a71d53500cc351547816bf1928fb043c4f/yarl-1.15.2-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:f4fe99ce44128c71233d0d72152db31ca119711dfc5f2c82385ad611d8d7f897", size = 330896, upload-time = "2024-10-13T18:46:01.126Z" }, + { url = "https://files.pythonhosted.org/packages/79/a2/d72e501bc1e33e68a5a31f584fe4556ab71a50a27bfd607d023f097cc9bb/yarl-1.15.2-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:9c1e3ff4b89cdd2e1a24c214f141e848b9e0451f08d7d4963cb4108d4d798f1f", size = 328787, upload-time = "2024-10-13T18:46:02.991Z" }, + { url = "https://files.pythonhosted.org/packages/9d/ba/890f7e1ea17f3c247748548eee876528ceb939e44566fa7d53baee57e5aa/yarl-1.15.2-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:711bdfae4e699a6d4f371137cbe9e740dc958530cb920eb6f43ff9551e17cfbc", size = 332631, upload-time = "2024-10-13T18:46:04.939Z" }, + { url = "https://files.pythonhosted.org/packages/48/c7/27b34206fd5dfe76b2caa08bf22f9212b2d665d5bb2df8a6dd3af498dcf4/yarl-1.15.2-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:4388c72174868884f76affcdd3656544c426407e0043c89b684d22fb265e04a5", size = 344023, upload-time = "2024-10-13T18:46:06.809Z" }, + { url = "https://files.pythonhosted.org/packages/88/e7/730b130f4f02bd8b00479baf9a57fdea1dc927436ed1d6ba08fa5c36c68e/yarl-1.15.2-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:f0e1844ad47c7bd5d6fa784f1d4accc5f4168b48999303a868fe0f8597bde715", size = 352290, upload-time = "2024-10-13T18:46:08.676Z" }, + { url = "https://files.pythonhosted.org/packages/84/9b/e8dda28f91a0af67098cddd455e6b540d3f682dda4c0de224215a57dee4a/yarl-1.15.2-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a5cafb02cf097a82d74403f7e0b6b9df3ffbfe8edf9415ea816314711764a27b", size = 343742, upload-time = "2024-10-13T18:46:10.583Z" }, + { url = "https://files.pythonhosted.org/packages/66/47/b1c6bb85f2b66decbe189e27fcc956ab74670a068655df30ef9a2e15c379/yarl-1.15.2-cp312-cp312-win32.whl", hash = "sha256:156ececdf636143f508770bf8a3a0498de64da5abd890c7dbb42ca9e3b6c05b8", size = 78051, upload-time = "2024-10-13T18:46:12.671Z" }, + { url = "https://files.pythonhosted.org/packages/7d/9e/1a897e5248ec53e96e9f15b3e6928efd5e75d322c6cf666f55c1c063e5c9/yarl-1.15.2-cp312-cp312-win_amd64.whl", hash = "sha256:435aca062444a7f0c884861d2e3ea79883bd1cd19d0a381928b69ae1b85bc51d", size = 84313, upload-time = "2024-10-13T18:46:15.237Z" }, + { url = "https://files.pythonhosted.org/packages/46/ab/be3229898d7eb1149e6ba7fe44f873cf054d275a00b326f2a858c9ff7175/yarl-1.15.2-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:416f2e3beaeae81e2f7a45dc711258be5bdc79c940a9a270b266c0bec038fb84", size = 135006, upload-time = "2024-10-13T18:46:16.909Z" }, + { url = "https://files.pythonhosted.org/packages/10/10/b91c186b1b0e63951f80481b3e6879bb9f7179d471fe7c4440c9e900e2a3/yarl-1.15.2-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:173563f3696124372831007e3d4b9821746964a95968628f7075d9231ac6bb33", size = 88121, upload-time = "2024-10-13T18:46:18.702Z" }, + { url = "https://files.pythonhosted.org/packages/bf/1d/4ceaccf836b9591abfde775e84249b847ac4c6c14ee2dd8d15b5b3cede44/yarl-1.15.2-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:9ce2e0f6123a60bd1a7f5ae3b2c49b240c12c132847f17aa990b841a417598a2", size = 85967, upload-time = "2024-10-13T18:46:20.354Z" }, + { url = "https://files.pythonhosted.org/packages/93/bd/c924f22bdb2c5d0ca03a9e64ecc5e041aace138c2a91afff7e2f01edc3a1/yarl-1.15.2-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eaea112aed589131f73d50d570a6864728bd7c0c66ef6c9154ed7b59f24da611", size = 325615, upload-time = "2024-10-13T18:46:22.057Z" }, + { url = "https://files.pythonhosted.org/packages/59/a5/6226accd5c01cafd57af0d249c7cf9dd12569cd9c78fbd93e8198e7a9d84/yarl-1.15.2-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e4ca3b9f370f218cc2a0309542cab8d0acdfd66667e7c37d04d617012485f904", size = 334945, upload-time = "2024-10-13T18:46:24.184Z" }, + { url = "https://files.pythonhosted.org/packages/4c/c1/cc6ccdd2bcd0ff7291602d5831754595260f8d2754642dfd34fef1791059/yarl-1.15.2-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:23ec1d3c31882b2a8a69c801ef58ebf7bae2553211ebbddf04235be275a38548", size = 336701, upload-time = "2024-10-13T18:46:27.038Z" }, + { url = "https://files.pythonhosted.org/packages/ef/ff/39a767ee249444e4b26ea998a526838238f8994c8f274befc1f94dacfb43/yarl-1.15.2-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:75119badf45f7183e10e348edff5a76a94dc19ba9287d94001ff05e81475967b", size = 330977, upload-time = "2024-10-13T18:46:28.921Z" }, + { url = "https://files.pythonhosted.org/packages/dd/ba/b1fed73f9d39e3e7be8f6786be5a2ab4399c21504c9168c3cadf6e441c2e/yarl-1.15.2-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:78e6fdc976ec966b99e4daa3812fac0274cc28cd2b24b0d92462e2e5ef90d368", size = 317402, upload-time = "2024-10-13T18:46:30.86Z" }, + { url = "https://files.pythonhosted.org/packages/82/e8/03e3ebb7f558374f29c04868b20ca484d7997f80a0a191490790a8c28058/yarl-1.15.2-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:8657d3f37f781d987037f9cc20bbc8b40425fa14380c87da0cb8dfce7c92d0fb", size = 331776, upload-time = "2024-10-13T18:46:33.037Z" }, + { url = "https://files.pythonhosted.org/packages/1f/83/90b0f4fd1ecf2602ba4ac50ad0bbc463122208f52dd13f152bbc0d8417dd/yarl-1.15.2-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:93bed8a8084544c6efe8856c362af08a23e959340c87a95687fdbe9c9f280c8b", size = 331585, upload-time = "2024-10-13T18:46:35.275Z" }, + { url = "https://files.pythonhosted.org/packages/c7/f6/1ed7e7f270ae5f9f1174c1f8597b29658f552fee101c26de8b2eb4ca147a/yarl-1.15.2-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:69d5856d526802cbda768d3e6246cd0d77450fa2a4bc2ea0ea14f0d972c2894b", size = 336395, upload-time = "2024-10-13T18:46:38.003Z" }, + { url = "https://files.pythonhosted.org/packages/e0/3a/4354ed8812909d9ec54a92716a53259b09e6b664209231f2ec5e75f4820d/yarl-1.15.2-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:ccad2800dfdff34392448c4bf834be124f10a5bc102f254521d931c1c53c455a", size = 342810, upload-time = "2024-10-13T18:46:39.952Z" }, + { url = "https://files.pythonhosted.org/packages/de/cc/39e55e16b1415a87f6d300064965d6cfb2ac8571e11339ccb7dada2444d9/yarl-1.15.2-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:a880372e2e5dbb9258a4e8ff43f13888039abb9dd6d515f28611c54361bc5644", size = 351441, upload-time = "2024-10-13T18:46:41.867Z" }, + { url = "https://files.pythonhosted.org/packages/fb/19/5cd4757079dc9d9f3de3e3831719b695f709a8ce029e70b33350c9d082a7/yarl-1.15.2-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:c998d0558805860503bc3a595994895ca0f7835e00668dadc673bbf7f5fbfcbe", size = 345875, upload-time = "2024-10-13T18:46:43.824Z" }, + { url = "https://files.pythonhosted.org/packages/83/a0/ef09b54634f73417f1ea4a746456a4372c1b044f07b26e16fa241bd2d94e/yarl-1.15.2-cp313-cp313-win32.whl", hash = "sha256:533a28754e7f7439f217550a497bb026c54072dbe16402b183fdbca2431935a9", size = 302609, upload-time = "2024-10-13T18:46:45.828Z" }, + { url = "https://files.pythonhosted.org/packages/20/9f/f39c37c17929d3975da84c737b96b606b68c495cc4ee86408f10523a1635/yarl-1.15.2-cp313-cp313-win_amd64.whl", hash = "sha256:5838f2b79dc8f96fdc44077c9e4e2e33d7089b10788464609df788eb97d03aad", size = 308252, upload-time = "2024-10-13T18:46:48.042Z" }, + { url = "https://files.pythonhosted.org/packages/7b/1f/544439ce6b7a498327d57ff40f0cd4f24bf4b1c1daf76c8c962dca022e71/yarl-1.15.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:fbbb63bed5fcd70cd3dd23a087cd78e4675fb5a2963b8af53f945cbbca79ae16", size = 138555, upload-time = "2024-10-13T18:46:50.448Z" }, + { url = "https://files.pythonhosted.org/packages/e8/b7/d6f33e7a42832f1e8476d0aabe089be0586a9110b5dfc2cef93444dc7c21/yarl-1.15.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:e2e93b88ecc8f74074012e18d679fb2e9c746f2a56f79cd5e2b1afcf2a8a786b", size = 89844, upload-time = "2024-10-13T18:46:52.297Z" }, + { url = "https://files.pythonhosted.org/packages/93/34/ede8d8ed7350b4b21e33fc4eff71e08de31da697034969b41190132d421f/yarl-1.15.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:af8ff8d7dc07ce873f643de6dfbcd45dc3db2c87462e5c387267197f59e6d776", size = 87671, upload-time = "2024-10-13T18:46:54.104Z" }, + { url = "https://files.pythonhosted.org/packages/fa/51/6d71e92bc54b5788b18f3dc29806f9ce37e12b7c610e8073357717f34b78/yarl-1.15.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:66f629632220a4e7858b58e4857927dd01a850a4cef2fb4044c8662787165cf7", size = 314558, upload-time = "2024-10-13T18:46:55.885Z" }, + { url = "https://files.pythonhosted.org/packages/76/0a/f9ffe503b4ef77cd77c9eefd37717c092e26f2c2dbbdd45700f864831292/yarl-1.15.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:833547179c31f9bec39b49601d282d6f0ea1633620701288934c5f66d88c3e50", size = 327622, upload-time = "2024-10-13T18:46:58.173Z" }, + { url = "https://files.pythonhosted.org/packages/8b/38/8eb602eeb153de0189d572dce4ed81b9b14f71de7c027d330b601b4fdcdc/yarl-1.15.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2aa738e0282be54eede1e3f36b81f1e46aee7ec7602aa563e81e0e8d7b67963f", size = 324447, upload-time = "2024-10-13T18:47:00.263Z" }, + { url = "https://files.pythonhosted.org/packages/c2/1e/1c78c695a4c7b957b5665e46a89ea35df48511dbed301a05c0a8beed0cc3/yarl-1.15.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9a13a07532e8e1c4a5a3afff0ca4553da23409fad65def1b71186fb867eeae8d", size = 319009, upload-time = "2024-10-13T18:47:02.417Z" }, + { url = "https://files.pythonhosted.org/packages/06/a0/7ea93de4ca1991e7f92a8901dcd1585165f547d342f7c6f36f1ea58b75de/yarl-1.15.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c45817e3e6972109d1a2c65091504a537e257bc3c885b4e78a95baa96df6a3f8", size = 307760, upload-time = "2024-10-13T18:47:04.553Z" }, + { url = "https://files.pythonhosted.org/packages/f4/b4/ceaa1f35cfb37fe06af3f7404438abf9a1262dc5df74dba37c90b0615e06/yarl-1.15.2-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:670eb11325ed3a6209339974b276811867defe52f4188fe18dc49855774fa9cf", size = 315038, upload-time = "2024-10-13T18:47:06.482Z" }, + { url = "https://files.pythonhosted.org/packages/da/45/a2ca2b547c56550eefc39e45d61e4b42ae6dbb3e913810b5a0eb53e86412/yarl-1.15.2-cp38-cp38-musllinux_1_2_armv7l.whl", hash = "sha256:d417a4f6943112fae3924bae2af7112562285848d9bcee737fc4ff7cbd450e6c", size = 312898, upload-time = "2024-10-13T18:47:09.291Z" }, + { url = "https://files.pythonhosted.org/packages/ea/e0/f692ba36dedc5b0b22084bba558a7ede053841e247b7dd2adbb9d40450be/yarl-1.15.2-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:bc8936d06cd53fddd4892677d65e98af514c8d78c79864f418bbf78a4a2edde4", size = 319370, upload-time = "2024-10-13T18:47:11.647Z" }, + { url = "https://files.pythonhosted.org/packages/b1/3f/0e382caf39958be6ae61d4bb0c82a68a3c45a494fc8cdc6f55c29757970e/yarl-1.15.2-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:954dde77c404084c2544e572f342aef384240b3e434e06cecc71597e95fd1ce7", size = 332429, upload-time = "2024-10-13T18:47:13.88Z" }, + { url = "https://files.pythonhosted.org/packages/21/6b/c824a4a1c45d67b15b431d4ab83b63462bfcbc710065902e10fa5c2ffd9e/yarl-1.15.2-cp38-cp38-musllinux_1_2_s390x.whl", hash = "sha256:5bc0df728e4def5e15a754521e8882ba5a5121bd6b5a3a0ff7efda5d6558ab3d", size = 333143, upload-time = "2024-10-13T18:47:16.141Z" }, + { url = "https://files.pythonhosted.org/packages/20/76/8af2a1d93fe95b04e284b5d55daaad33aae6e2f6254a1bcdb40e2752af6c/yarl-1.15.2-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:b71862a652f50babab4a43a487f157d26b464b1dedbcc0afda02fd64f3809d04", size = 326687, upload-time = "2024-10-13T18:47:18.179Z" }, + { url = "https://files.pythonhosted.org/packages/1c/53/490830773f907ef8a311cc5d82e5830f75f7692c1adacbdb731d3f1246fd/yarl-1.15.2-cp38-cp38-win32.whl", hash = "sha256:63eab904f8630aed5a68f2d0aeab565dcfc595dc1bf0b91b71d9ddd43dea3aea", size = 78705, upload-time = "2024-10-13T18:47:20.876Z" }, + { url = "https://files.pythonhosted.org/packages/9c/9d/d944e897abf37f50f4fa2d8d6f5fd0ed9413bc8327d3b4cc25ba9694e1ba/yarl-1.15.2-cp38-cp38-win_amd64.whl", hash = "sha256:2cf441c4b6e538ba0d2591574f95d3fdd33f1efafa864faa077d9636ecc0c4e9", size = 84998, upload-time = "2024-10-13T18:47:23.301Z" }, + { url = "https://files.pythonhosted.org/packages/91/1c/1c9d08c29b10499348eedc038cf61b6d96d5ba0e0d69438975845939ed3c/yarl-1.15.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:a32d58f4b521bb98b2c0aa9da407f8bd57ca81f34362bcb090e4a79e9924fefc", size = 138011, upload-time = "2024-10-13T18:47:25.002Z" }, + { url = "https://files.pythonhosted.org/packages/d4/33/2d4a1418bae6d7883c1fcc493be7b6d6fe015919835adc9e8eeba472e9f7/yarl-1.15.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:766dcc00b943c089349d4060b935c76281f6be225e39994c2ccec3a2a36ad627", size = 89618, upload-time = "2024-10-13T18:47:27.587Z" }, + { url = "https://files.pythonhosted.org/packages/78/2e/0024c674a376cfdc722a167a8f308f5779aca615cb7a28d67fbeabf3f697/yarl-1.15.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:bed1b5dbf90bad3bfc19439258c97873eab453c71d8b6869c136346acfe497e7", size = 87347, upload-time = "2024-10-13T18:47:29.671Z" }, + { url = "https://files.pythonhosted.org/packages/c5/08/a01874dabd4ddf475c5c2adc86f7ac329f83a361ee513a97841720ab7b24/yarl-1.15.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ed20a4bdc635f36cb19e630bfc644181dd075839b6fc84cac51c0f381ac472e2", size = 310438, upload-time = "2024-10-13T18:47:31.577Z" }, + { url = "https://files.pythonhosted.org/packages/09/95/691bc6de2c1b0e9c8bbaa5f8f38118d16896ba1a069a09d1fb073d41a093/yarl-1.15.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d538df442c0d9665664ab6dd5fccd0110fa3b364914f9c85b3ef9b7b2e157980", size = 325384, upload-time = "2024-10-13T18:47:33.587Z" }, + { url = "https://files.pythonhosted.org/packages/95/fd/fee11eb3337f48c62d39c5676e6a0e4e318e318900a901b609a3c45394df/yarl-1.15.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c6cf1d92edf936ceedc7afa61b07e9d78a27b15244aa46bbcd534c7458ee1b", size = 321820, upload-time = "2024-10-13T18:47:35.633Z" }, + { url = "https://files.pythonhosted.org/packages/7a/ad/4a2c9bbebaefdce4a69899132f4bf086abbddb738dc6e794a31193bc0854/yarl-1.15.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ce44217ad99ffad8027d2fde0269ae368c86db66ea0571c62a000798d69401fb", size = 314150, upload-time = "2024-10-13T18:47:37.693Z" }, + { url = "https://files.pythonhosted.org/packages/38/7d/552c37bc6c4ae8ea900e44b6c05cb16d50dca72d3782ccd66f53e27e353f/yarl-1.15.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b47a6000a7e833ebfe5886b56a31cb2ff12120b1efd4578a6fcc38df16cc77bd", size = 304202, upload-time = "2024-10-13T18:47:40.411Z" }, + { url = "https://files.pythonhosted.org/packages/2e/f8/c22a158f3337f49775775ecef43fc097a98b20cdce37425b68b9c45a6f94/yarl-1.15.2-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:e52f77a0cd246086afde8815039f3e16f8d2be51786c0a39b57104c563c5cbb0", size = 310311, upload-time = "2024-10-13T18:47:43.236Z" }, + { url = "https://files.pythonhosted.org/packages/ce/e4/ebce06afa25c2a6c8e6c9a5915cbbc7940a37f3ec38e950e8f346ca908da/yarl-1.15.2-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:f9ca0e6ce7774dc7830dc0cc4bb6b3eec769db667f230e7c770a628c1aa5681b", size = 310645, upload-time = "2024-10-13T18:47:45.24Z" }, + { url = "https://files.pythonhosted.org/packages/0a/34/5504cc8fbd1be959ec0a1e9e9f471fd438c37cb877b0178ce09085b36b51/yarl-1.15.2-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:136f9db0f53c0206db38b8cd0c985c78ded5fd596c9a86ce5c0b92afb91c3a19", size = 313328, upload-time = "2024-10-13T18:47:47.546Z" }, + { url = "https://files.pythonhosted.org/packages/cf/e4/fb3f91a539c6505e347d7d75bc675d291228960ffd6481ced76a15412924/yarl-1.15.2-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:173866d9f7409c0fb514cf6e78952e65816600cb888c68b37b41147349fe0057", size = 330135, upload-time = "2024-10-13T18:47:50.279Z" }, + { url = "https://files.pythonhosted.org/packages/e1/08/a0b27db813f0159e1c8a45f48852afded501de2f527e7613c4dcf436ecf7/yarl-1.15.2-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:6e840553c9c494a35e449a987ca2c4f8372668ee954a03a9a9685075228e5036", size = 327155, upload-time = "2024-10-13T18:47:52.337Z" }, + { url = "https://files.pythonhosted.org/packages/97/4e/b3414dded12d0e2b52eb1964c21a8d8b68495b320004807de770f7b6b53a/yarl-1.15.2-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:458c0c65802d816a6b955cf3603186de79e8fdb46d4f19abaec4ef0a906f50a7", size = 320810, upload-time = "2024-10-13T18:47:55.067Z" }, + { url = "https://files.pythonhosted.org/packages/bb/ca/e5149c55d1c9dcf3d5b48acd7c71ca8622fd2f61322d0386fe63ba106774/yarl-1.15.2-cp39-cp39-win32.whl", hash = "sha256:5b48388ded01f6f2429a8c55012bdbd1c2a0c3735b3e73e221649e524c34a58d", size = 78686, upload-time = "2024-10-13T18:47:57Z" }, + { url = "https://files.pythonhosted.org/packages/b1/87/f56a80a1abaf65dbf138b821357b51b6cc061756bb7d93f08797950b3881/yarl-1.15.2-cp39-cp39-win_amd64.whl", hash = "sha256:81dadafb3aa124f86dc267a2168f71bbd2bfb163663661ab0038f6e4b8edb810", size = 84818, upload-time = "2024-10-13T18:47:58.76Z" }, + { url = "https://files.pythonhosted.org/packages/46/cf/a28c494decc9c8776b0d7b729c68d26fdafefcedd8d2eab5d9cd767376b2/yarl-1.15.2-py3-none-any.whl", hash = "sha256:0d3105efab7c5c091609abacad33afff33bdff0035bece164c98bcf5a85ef90a", size = 38891, upload-time = "2024-10-13T18:48:00.883Z" }, +] + +[[package]] +name = "yarl" +version = "1.22.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version >= '3.10'", + "python_full_version == '3.9.*'", +] +dependencies = [ + { name = "idna", marker = "python_full_version >= '3.9'" }, + { name = "multidict", version = "6.7.0", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, + { name = "propcache", version = "0.4.1", source = { registry = "https://pypi.org/simple" }, marker = "python_full_version >= '3.9'" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/57/63/0c6ebca57330cd313f6102b16dd57ffaf3ec4c83403dcb45dbd15c6f3ea1/yarl-1.22.0.tar.gz", hash = "sha256:bebf8557577d4401ba8bd9ff33906f1376c877aa78d1fe216ad01b4d6745af71", size = 187169, upload-time = "2025-10-06T14:12:55.963Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d1/43/a2204825342f37c337f5edb6637040fa14e365b2fcc2346960201d457579/yarl-1.22.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:c7bd6683587567e5a49ee6e336e0612bec8329be1b7d4c8af5687dcdeb67ee1e", size = 140517, upload-time = "2025-10-06T14:08:42.494Z" }, + { url = "https://files.pythonhosted.org/packages/44/6f/674f3e6f02266428c56f704cd2501c22f78e8b2eeb23f153117cc86fb28a/yarl-1.22.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:5cdac20da754f3a723cceea5b3448e1a2074866406adeb4ef35b469d089adb8f", size = 93495, upload-time = "2025-10-06T14:08:46.2Z" }, + { url = "https://files.pythonhosted.org/packages/b8/12/5b274d8a0f30c07b91b2f02cba69152600b47830fcfb465c108880fcee9c/yarl-1.22.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:07a524d84df0c10f41e3ee918846e1974aba4ec017f990dc735aad487a0bdfdf", size = 94400, upload-time = "2025-10-06T14:08:47.855Z" }, + { url = "https://files.pythonhosted.org/packages/e2/7f/df1b6949b1fa1aa9ff6de6e2631876ad4b73c4437822026e85d8acb56bb1/yarl-1.22.0-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e1b329cb8146d7b736677a2440e422eadd775d1806a81db2d4cded80a48efc1a", size = 347545, upload-time = "2025-10-06T14:08:49.683Z" }, + { url = "https://files.pythonhosted.org/packages/84/09/f92ed93bd6cd77872ab6c3462df45ca45cd058d8f1d0c9b4f54c1704429f/yarl-1.22.0-cp310-cp310-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:75976c6945d85dbb9ee6308cd7ff7b1fb9409380c82d6119bd778d8fcfe2931c", size = 319598, upload-time = "2025-10-06T14:08:51.215Z" }, + { url = "https://files.pythonhosted.org/packages/c3/97/ac3f3feae7d522cf7ccec3d340bb0b2b61c56cb9767923df62a135092c6b/yarl-1.22.0-cp310-cp310-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:80ddf7a5f8c86cb3eb4bc9028b07bbbf1f08a96c5c0bc1244be5e8fefcb94147", size = 363893, upload-time = "2025-10-06T14:08:53.144Z" }, + { url = "https://files.pythonhosted.org/packages/06/49/f3219097403b9c84a4d079b1d7bda62dd9b86d0d6e4428c02d46ab2c77fc/yarl-1.22.0-cp310-cp310-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:d332fc2e3c94dad927f2112395772a4e4fedbcf8f80efc21ed7cdfae4d574fdb", size = 371240, upload-time = "2025-10-06T14:08:55.036Z" }, + { url = "https://files.pythonhosted.org/packages/35/9f/06b765d45c0e44e8ecf0fe15c9eacbbde342bb5b7561c46944f107bfb6c3/yarl-1.22.0-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:0cf71bf877efeac18b38d3930594c0948c82b64547c1cf420ba48722fe5509f6", size = 346965, upload-time = "2025-10-06T14:08:56.722Z" }, + { url = "https://files.pythonhosted.org/packages/c5/69/599e7cea8d0fcb1694323b0db0dda317fa3162f7b90166faddecf532166f/yarl-1.22.0-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:663e1cadaddae26be034a6ab6072449a8426ddb03d500f43daf952b74553bba0", size = 342026, upload-time = "2025-10-06T14:08:58.563Z" }, + { url = "https://files.pythonhosted.org/packages/95/6f/9dfd12c8bc90fea9eab39832ee32ea48f8e53d1256252a77b710c065c89f/yarl-1.22.0-cp310-cp310-musllinux_1_2_armv7l.whl", hash = "sha256:6dcbb0829c671f305be48a7227918cfcd11276c2d637a8033a99a02b67bf9eda", size = 335637, upload-time = "2025-10-06T14:09:00.506Z" }, + { url = "https://files.pythonhosted.org/packages/57/2e/34c5b4eb9b07e16e873db5b182c71e5f06f9b5af388cdaa97736d79dd9a6/yarl-1.22.0-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:f0d97c18dfd9a9af4490631905a3f131a8e4c9e80a39353919e2cfed8f00aedc", size = 359082, upload-time = "2025-10-06T14:09:01.936Z" }, + { url = "https://files.pythonhosted.org/packages/31/71/fa7e10fb772d273aa1f096ecb8ab8594117822f683bab7d2c5a89914c92a/yarl-1.22.0-cp310-cp310-musllinux_1_2_s390x.whl", hash = "sha256:437840083abe022c978470b942ff832c3940b2ad3734d424b7eaffcd07f76737", size = 357811, upload-time = "2025-10-06T14:09:03.445Z" }, + { url = "https://files.pythonhosted.org/packages/26/da/11374c04e8e1184a6a03cf9c8f5688d3e5cec83ed6f31ad3481b3207f709/yarl-1.22.0-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:a899cbd98dce6f5d8de1aad31cb712ec0a530abc0a86bd6edaa47c1090138467", size = 351223, upload-time = "2025-10-06T14:09:05.401Z" }, + { url = "https://files.pythonhosted.org/packages/82/8f/e2d01f161b0c034a30410e375e191a5d27608c1f8693bab1a08b089ca096/yarl-1.22.0-cp310-cp310-win32.whl", hash = "sha256:595697f68bd1f0c1c159fcb97b661fc9c3f5db46498043555d04805430e79bea", size = 82118, upload-time = "2025-10-06T14:09:11.148Z" }, + { url = "https://files.pythonhosted.org/packages/62/46/94c76196642dbeae634c7a61ba3da88cd77bed875bf6e4a8bed037505aa6/yarl-1.22.0-cp310-cp310-win_amd64.whl", hash = "sha256:cb95a9b1adaa48e41815a55ae740cfda005758104049a640a398120bf02515ca", size = 86852, upload-time = "2025-10-06T14:09:12.958Z" }, + { url = "https://files.pythonhosted.org/packages/af/af/7df4f179d3b1a6dcb9a4bd2ffbc67642746fcafdb62580e66876ce83fff4/yarl-1.22.0-cp310-cp310-win_arm64.whl", hash = "sha256:b85b982afde6df99ecc996990d4ad7ccbdbb70e2a4ba4de0aecde5922ba98a0b", size = 82012, upload-time = "2025-10-06T14:09:14.664Z" }, + { url = "https://files.pythonhosted.org/packages/4d/27/5ab13fc84c76a0250afd3d26d5936349a35be56ce5785447d6c423b26d92/yarl-1.22.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:1ab72135b1f2db3fed3997d7e7dc1b80573c67138023852b6efb336a5eae6511", size = 141607, upload-time = "2025-10-06T14:09:16.298Z" }, + { url = "https://files.pythonhosted.org/packages/6a/a1/d065d51d02dc02ce81501d476b9ed2229d9a990818332242a882d5d60340/yarl-1.22.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:669930400e375570189492dc8d8341301578e8493aec04aebc20d4717f899dd6", size = 94027, upload-time = "2025-10-06T14:09:17.786Z" }, + { url = "https://files.pythonhosted.org/packages/c1/da/8da9f6a53f67b5106ffe902c6fa0164e10398d4e150d85838b82f424072a/yarl-1.22.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:792a2af6d58177ef7c19cbf0097aba92ca1b9cb3ffdd9c7470e156c8f9b5e028", size = 94963, upload-time = "2025-10-06T14:09:19.662Z" }, + { url = "https://files.pythonhosted.org/packages/68/fe/2c1f674960c376e29cb0bec1249b117d11738db92a6ccc4a530b972648db/yarl-1.22.0-cp311-cp311-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3ea66b1c11c9150f1372f69afb6b8116f2dd7286f38e14ea71a44eee9ec51b9d", size = 368406, upload-time = "2025-10-06T14:09:21.402Z" }, + { url = "https://files.pythonhosted.org/packages/95/26/812a540e1c3c6418fec60e9bbd38e871eaba9545e94fa5eff8f4a8e28e1e/yarl-1.22.0-cp311-cp311-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:3e2daa88dc91870215961e96a039ec73e4937da13cf77ce17f9cad0c18df3503", size = 336581, upload-time = "2025-10-06T14:09:22.98Z" }, + { url = "https://files.pythonhosted.org/packages/0b/f5/5777b19e26fdf98563985e481f8be3d8a39f8734147a6ebf459d0dab5a6b/yarl-1.22.0-cp311-cp311-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ba440ae430c00eee41509353628600212112cd5018d5def7e9b05ea7ac34eb65", size = 388924, upload-time = "2025-10-06T14:09:24.655Z" }, + { url = "https://files.pythonhosted.org/packages/86/08/24bd2477bd59c0bbd994fe1d93b126e0472e4e3df5a96a277b0a55309e89/yarl-1.22.0-cp311-cp311-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:e6438cc8f23a9c1478633d216b16104a586b9761db62bfacb6425bac0a36679e", size = 392890, upload-time = "2025-10-06T14:09:26.617Z" }, + { url = "https://files.pythonhosted.org/packages/46/00/71b90ed48e895667ecfb1eaab27c1523ee2fa217433ed77a73b13205ca4b/yarl-1.22.0-cp311-cp311-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:4c52a6e78aef5cf47a98ef8e934755abf53953379b7d53e68b15ff4420e6683d", size = 365819, upload-time = "2025-10-06T14:09:28.544Z" }, + { url = "https://files.pythonhosted.org/packages/30/2d/f715501cae832651d3282387c6a9236cd26bd00d0ff1e404b3dc52447884/yarl-1.22.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:3b06bcadaac49c70f4c88af4ffcfbe3dc155aab3163e75777818092478bcbbe7", size = 363601, upload-time = "2025-10-06T14:09:30.568Z" }, + { url = "https://files.pythonhosted.org/packages/f8/f9/a678c992d78e394e7126ee0b0e4e71bd2775e4334d00a9278c06a6cce96a/yarl-1.22.0-cp311-cp311-musllinux_1_2_armv7l.whl", hash = "sha256:6944b2dc72c4d7f7052683487e3677456050ff77fcf5e6204e98caf785ad1967", size = 358072, upload-time = "2025-10-06T14:09:32.528Z" }, + { url = "https://files.pythonhosted.org/packages/2c/d1/b49454411a60edb6fefdcad4f8e6dbba7d8019e3a508a1c5836cba6d0781/yarl-1.22.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:d5372ca1df0f91a86b047d1277c2aaf1edb32d78bbcefffc81b40ffd18f027ed", size = 385311, upload-time = "2025-10-06T14:09:34.634Z" }, + { url = "https://files.pythonhosted.org/packages/87/e5/40d7a94debb8448c7771a916d1861d6609dddf7958dc381117e7ba36d9e8/yarl-1.22.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:51af598701f5299012b8416486b40fceef8c26fc87dc6d7d1f6fc30609ea0aa6", size = 381094, upload-time = "2025-10-06T14:09:36.268Z" }, + { url = "https://files.pythonhosted.org/packages/35/d8/611cc282502381ad855448643e1ad0538957fc82ae83dfe7762c14069e14/yarl-1.22.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:b266bd01fedeffeeac01a79ae181719ff848a5a13ce10075adbefc8f1daee70e", size = 370944, upload-time = "2025-10-06T14:09:37.872Z" }, + { url = "https://files.pythonhosted.org/packages/2d/df/fadd00fb1c90e1a5a8bd731fa3d3de2e165e5a3666a095b04e31b04d9cb6/yarl-1.22.0-cp311-cp311-win32.whl", hash = "sha256:a9b1ba5610a4e20f655258d5a1fdc7ebe3d837bb0e45b581398b99eb98b1f5ca", size = 81804, upload-time = "2025-10-06T14:09:39.359Z" }, + { url = "https://files.pythonhosted.org/packages/b5/f7/149bb6f45f267cb5c074ac40c01c6b3ea6d8a620d34b337f6321928a1b4d/yarl-1.22.0-cp311-cp311-win_amd64.whl", hash = "sha256:078278b9b0b11568937d9509b589ee83ef98ed6d561dfe2020e24a9fd08eaa2b", size = 86858, upload-time = "2025-10-06T14:09:41.068Z" }, + { url = "https://files.pythonhosted.org/packages/2b/13/88b78b93ad3f2f0b78e13bfaaa24d11cbc746e93fe76d8c06bf139615646/yarl-1.22.0-cp311-cp311-win_arm64.whl", hash = "sha256:b6a6f620cfe13ccec221fa312139135166e47ae169f8253f72a0abc0dae94376", size = 81637, upload-time = "2025-10-06T14:09:42.712Z" }, + { url = "https://files.pythonhosted.org/packages/75/ff/46736024fee3429b80a165a732e38e5d5a238721e634ab41b040d49f8738/yarl-1.22.0-cp312-cp312-macosx_10_13_universal2.whl", hash = "sha256:e340382d1afa5d32b892b3ff062436d592ec3d692aeea3bef3a5cfe11bbf8c6f", size = 142000, upload-time = "2025-10-06T14:09:44.631Z" }, + { url = "https://files.pythonhosted.org/packages/5a/9a/b312ed670df903145598914770eb12de1bac44599549b3360acc96878df8/yarl-1.22.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:f1e09112a2c31ffe8d80be1b0988fa6a18c5d5cad92a9ffbb1c04c91bfe52ad2", size = 94338, upload-time = "2025-10-06T14:09:46.372Z" }, + { url = "https://files.pythonhosted.org/packages/ba/f5/0601483296f09c3c65e303d60c070a5c19fcdbc72daa061e96170785bc7d/yarl-1.22.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:939fe60db294c786f6b7c2d2e121576628468f65453d86b0fe36cb52f987bd74", size = 94909, upload-time = "2025-10-06T14:09:48.648Z" }, + { url = "https://files.pythonhosted.org/packages/60/41/9a1fe0b73dbcefce72e46cf149b0e0a67612d60bfc90fb59c2b2efdfbd86/yarl-1.22.0-cp312-cp312-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:e1651bf8e0398574646744c1885a41198eba53dc8a9312b954073f845c90a8df", size = 372940, upload-time = "2025-10-06T14:09:50.089Z" }, + { url = "https://files.pythonhosted.org/packages/17/7a/795cb6dfee561961c30b800f0ed616b923a2ec6258b5def2a00bf8231334/yarl-1.22.0-cp312-cp312-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:b8a0588521a26bf92a57a1705b77b8b59044cdceccac7151bd8d229e66b8dedb", size = 345825, upload-time = "2025-10-06T14:09:52.142Z" }, + { url = "https://files.pythonhosted.org/packages/d7/93/a58f4d596d2be2ae7bab1a5846c4d270b894958845753b2c606d666744d3/yarl-1.22.0-cp312-cp312-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:42188e6a615c1a75bcaa6e150c3fe8f3e8680471a6b10150c5f7e83f47cc34d2", size = 386705, upload-time = "2025-10-06T14:09:54.128Z" }, + { url = "https://files.pythonhosted.org/packages/61/92/682279d0e099d0e14d7fd2e176bd04f48de1484f56546a3e1313cd6c8e7c/yarl-1.22.0-cp312-cp312-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:f6d2cb59377d99718913ad9a151030d6f83ef420a2b8f521d94609ecc106ee82", size = 396518, upload-time = "2025-10-06T14:09:55.762Z" }, + { url = "https://files.pythonhosted.org/packages/db/0f/0d52c98b8a885aeda831224b78f3be7ec2e1aa4a62091f9f9188c3c65b56/yarl-1.22.0-cp312-cp312-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:50678a3b71c751d58d7908edc96d332af328839eea883bb554a43f539101277a", size = 377267, upload-time = "2025-10-06T14:09:57.958Z" }, + { url = "https://files.pythonhosted.org/packages/22/42/d2685e35908cbeaa6532c1fc73e89e7f2efb5d8a7df3959ea8e37177c5a3/yarl-1.22.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:1e8fbaa7cec507aa24ea27a01456e8dd4b6fab829059b69844bd348f2d467124", size = 365797, upload-time = "2025-10-06T14:09:59.527Z" }, + { url = "https://files.pythonhosted.org/packages/a2/83/cf8c7bcc6355631762f7d8bdab920ad09b82efa6b722999dfb05afa6cfac/yarl-1.22.0-cp312-cp312-musllinux_1_2_armv7l.whl", hash = "sha256:433885ab5431bc3d3d4f2f9bd15bfa1614c522b0f1405d62c4f926ccd69d04fa", size = 365535, upload-time = "2025-10-06T14:10:01.139Z" }, + { url = "https://files.pythonhosted.org/packages/25/e1/5302ff9b28f0c59cac913b91fe3f16c59a033887e57ce9ca5d41a3a94737/yarl-1.22.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:b790b39c7e9a4192dc2e201a282109ed2985a1ddbd5ac08dc56d0e121400a8f7", size = 382324, upload-time = "2025-10-06T14:10:02.756Z" }, + { url = "https://files.pythonhosted.org/packages/bf/cd/4617eb60f032f19ae3a688dc990d8f0d89ee0ea378b61cac81ede3e52fae/yarl-1.22.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:31f0b53913220599446872d757257be5898019c85e7971599065bc55065dc99d", size = 383803, upload-time = "2025-10-06T14:10:04.552Z" }, + { url = "https://files.pythonhosted.org/packages/59/65/afc6e62bb506a319ea67b694551dab4a7e6fb7bf604e9bd9f3e11d575fec/yarl-1.22.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:a49370e8f711daec68d09b821a34e1167792ee2d24d405cbc2387be4f158b520", size = 374220, upload-time = "2025-10-06T14:10:06.489Z" }, + { url = "https://files.pythonhosted.org/packages/e7/3d/68bf18d50dc674b942daec86a9ba922d3113d8399b0e52b9897530442da2/yarl-1.22.0-cp312-cp312-win32.whl", hash = "sha256:70dfd4f241c04bd9239d53b17f11e6ab672b9f1420364af63e8531198e3f5fe8", size = 81589, upload-time = "2025-10-06T14:10:09.254Z" }, + { url = "https://files.pythonhosted.org/packages/c8/9a/6ad1a9b37c2f72874f93e691b2e7ecb6137fb2b899983125db4204e47575/yarl-1.22.0-cp312-cp312-win_amd64.whl", hash = "sha256:8884d8b332a5e9b88e23f60bb166890009429391864c685e17bd73a9eda9105c", size = 87213, upload-time = "2025-10-06T14:10:11.369Z" }, + { url = "https://files.pythonhosted.org/packages/44/c5/c21b562d1680a77634d748e30c653c3ca918beb35555cff24986fff54598/yarl-1.22.0-cp312-cp312-win_arm64.whl", hash = "sha256:ea70f61a47f3cc93bdf8b2f368ed359ef02a01ca6393916bc8ff877427181e74", size = 81330, upload-time = "2025-10-06T14:10:13.112Z" }, + { url = "https://files.pythonhosted.org/packages/ea/f3/d67de7260456ee105dc1d162d43a019ecad6b91e2f51809d6cddaa56690e/yarl-1.22.0-cp313-cp313-macosx_10_13_universal2.whl", hash = "sha256:8dee9c25c74997f6a750cd317b8ca63545169c098faee42c84aa5e506c819b53", size = 139980, upload-time = "2025-10-06T14:10:14.601Z" }, + { url = "https://files.pythonhosted.org/packages/01/88/04d98af0b47e0ef42597b9b28863b9060bb515524da0a65d5f4db160b2d5/yarl-1.22.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:01e73b85a5434f89fc4fe27dcda2aff08ddf35e4d47bbbea3bdcd25321af538a", size = 93424, upload-time = "2025-10-06T14:10:16.115Z" }, + { url = "https://files.pythonhosted.org/packages/18/91/3274b215fd8442a03975ce6bee5fe6aa57a8326b29b9d3d56234a1dca244/yarl-1.22.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:22965c2af250d20c873cdbee8ff958fb809940aeb2e74ba5f20aaf6b7ac8c70c", size = 93821, upload-time = "2025-10-06T14:10:17.993Z" }, + { url = "https://files.pythonhosted.org/packages/61/3a/caf4e25036db0f2da4ca22a353dfeb3c9d3c95d2761ebe9b14df8fc16eb0/yarl-1.22.0-cp313-cp313-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b4f15793aa49793ec8d1c708ab7f9eded1aa72edc5174cae703651555ed1b601", size = 373243, upload-time = "2025-10-06T14:10:19.44Z" }, + { url = "https://files.pythonhosted.org/packages/6e/9e/51a77ac7516e8e7803b06e01f74e78649c24ee1021eca3d6a739cb6ea49c/yarl-1.22.0-cp313-cp313-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:e5542339dcf2747135c5c85f68680353d5cb9ffd741c0f2e8d832d054d41f35a", size = 342361, upload-time = "2025-10-06T14:10:21.124Z" }, + { url = "https://files.pythonhosted.org/packages/d4/f8/33b92454789dde8407f156c00303e9a891f1f51a0330b0fad7c909f87692/yarl-1.22.0-cp313-cp313-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:5c401e05ad47a75869c3ab3e35137f8468b846770587e70d71e11de797d113df", size = 387036, upload-time = "2025-10-06T14:10:22.902Z" }, + { url = "https://files.pythonhosted.org/packages/d9/9a/c5db84ea024f76838220280f732970aa4ee154015d7f5c1bfb60a267af6f/yarl-1.22.0-cp313-cp313-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:243dda95d901c733f5b59214d28b0120893d91777cb8aa043e6ef059d3cddfe2", size = 397671, upload-time = "2025-10-06T14:10:24.523Z" }, + { url = "https://files.pythonhosted.org/packages/11/c9/cd8538dc2e7727095e0c1d867bad1e40c98f37763e6d995c1939f5fdc7b1/yarl-1.22.0-cp313-cp313-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bec03d0d388060058f5d291a813f21c011041938a441c593374da6077fe21b1b", size = 377059, upload-time = "2025-10-06T14:10:26.406Z" }, + { url = "https://files.pythonhosted.org/packages/a1/b9/ab437b261702ced75122ed78a876a6dec0a1b0f5e17a4ac7a9a2482d8abe/yarl-1.22.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:b0748275abb8c1e1e09301ee3cf90c8a99678a4e92e4373705f2a2570d581273", size = 365356, upload-time = "2025-10-06T14:10:28.461Z" }, + { url = "https://files.pythonhosted.org/packages/b2/9d/8e1ae6d1d008a9567877b08f0ce4077a29974c04c062dabdb923ed98e6fe/yarl-1.22.0-cp313-cp313-musllinux_1_2_armv7l.whl", hash = "sha256:47fdb18187e2a4e18fda2c25c05d8251a9e4a521edaed757fef033e7d8498d9a", size = 361331, upload-time = "2025-10-06T14:10:30.541Z" }, + { url = "https://files.pythonhosted.org/packages/ca/5a/09b7be3905962f145b73beb468cdd53db8aa171cf18c80400a54c5b82846/yarl-1.22.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:c7044802eec4524fde550afc28edda0dd5784c4c45f0be151a2d3ba017daca7d", size = 382590, upload-time = "2025-10-06T14:10:33.352Z" }, + { url = "https://files.pythonhosted.org/packages/aa/7f/59ec509abf90eda5048b0bc3e2d7b5099dffdb3e6b127019895ab9d5ef44/yarl-1.22.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:139718f35149ff544caba20fce6e8a2f71f1e39b92c700d8438a0b1d2a631a02", size = 385316, upload-time = "2025-10-06T14:10:35.034Z" }, + { url = "https://files.pythonhosted.org/packages/e5/84/891158426bc8036bfdfd862fabd0e0fa25df4176ec793e447f4b85cf1be4/yarl-1.22.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:e1b51bebd221006d3d2f95fbe124b22b247136647ae5dcc8c7acafba66e5ee67", size = 374431, upload-time = "2025-10-06T14:10:37.76Z" }, + { url = "https://files.pythonhosted.org/packages/bb/49/03da1580665baa8bef5e8ed34c6df2c2aca0a2f28bf397ed238cc1bbc6f2/yarl-1.22.0-cp313-cp313-win32.whl", hash = "sha256:d3e32536234a95f513bd374e93d717cf6b2231a791758de6c509e3653f234c95", size = 81555, upload-time = "2025-10-06T14:10:39.649Z" }, + { url = "https://files.pythonhosted.org/packages/9a/ee/450914ae11b419eadd067c6183ae08381cfdfcb9798b90b2b713bbebddda/yarl-1.22.0-cp313-cp313-win_amd64.whl", hash = "sha256:47743b82b76d89a1d20b83e60d5c20314cbd5ba2befc9cda8f28300c4a08ed4d", size = 86965, upload-time = "2025-10-06T14:10:41.313Z" }, + { url = "https://files.pythonhosted.org/packages/98/4d/264a01eae03b6cf629ad69bae94e3b0e5344741e929073678e84bf7a3e3b/yarl-1.22.0-cp313-cp313-win_arm64.whl", hash = "sha256:5d0fcda9608875f7d052eff120c7a5da474a6796fe4d83e152e0e4d42f6d1a9b", size = 81205, upload-time = "2025-10-06T14:10:43.167Z" }, + { url = "https://files.pythonhosted.org/packages/88/fc/6908f062a2f77b5f9f6d69cecb1747260831ff206adcbc5b510aff88df91/yarl-1.22.0-cp313-cp313t-macosx_10_13_universal2.whl", hash = "sha256:719ae08b6972befcba4310e49edb1161a88cdd331e3a694b84466bd938a6ab10", size = 146209, upload-time = "2025-10-06T14:10:44.643Z" }, + { url = "https://files.pythonhosted.org/packages/65/47/76594ae8eab26210b4867be6f49129861ad33da1f1ebdf7051e98492bf62/yarl-1.22.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:47d8a5c446df1c4db9d21b49619ffdba90e77c89ec6e283f453856c74b50b9e3", size = 95966, upload-time = "2025-10-06T14:10:46.554Z" }, + { url = "https://files.pythonhosted.org/packages/ab/ce/05e9828a49271ba6b5b038b15b3934e996980dd78abdfeb52a04cfb9467e/yarl-1.22.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:cfebc0ac8333520d2d0423cbbe43ae43c8838862ddb898f5ca68565e395516e9", size = 97312, upload-time = "2025-10-06T14:10:48.007Z" }, + { url = "https://files.pythonhosted.org/packages/d1/c5/7dffad5e4f2265b29c9d7ec869c369e4223166e4f9206fc2243ee9eea727/yarl-1.22.0-cp313-cp313t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:4398557cbf484207df000309235979c79c4356518fd5c99158c7d38203c4da4f", size = 361967, upload-time = "2025-10-06T14:10:49.997Z" }, + { url = "https://files.pythonhosted.org/packages/50/b2/375b933c93a54bff7fc041e1a6ad2c0f6f733ffb0c6e642ce56ee3b39970/yarl-1.22.0-cp313-cp313t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:2ca6fd72a8cd803be290d42f2dec5cdcd5299eeb93c2d929bf060ad9efaf5de0", size = 323949, upload-time = "2025-10-06T14:10:52.004Z" }, + { url = "https://files.pythonhosted.org/packages/66/50/bfc2a29a1d78644c5a7220ce2f304f38248dc94124a326794e677634b6cf/yarl-1.22.0-cp313-cp313t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:ca1f59c4e1ab6e72f0a23c13fca5430f889634166be85dbf1013683e49e3278e", size = 361818, upload-time = "2025-10-06T14:10:54.078Z" }, + { url = "https://files.pythonhosted.org/packages/46/96/f3941a46af7d5d0f0498f86d71275696800ddcdd20426298e572b19b91ff/yarl-1.22.0-cp313-cp313t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:6c5010a52015e7c70f86eb967db0f37f3c8bd503a695a49f8d45700144667708", size = 372626, upload-time = "2025-10-06T14:10:55.767Z" }, + { url = "https://files.pythonhosted.org/packages/c1/42/8b27c83bb875cd89448e42cd627e0fb971fa1675c9ec546393d18826cb50/yarl-1.22.0-cp313-cp313t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:9d7672ecf7557476642c88497c2f8d8542f8e36596e928e9bcba0e42e1e7d71f", size = 341129, upload-time = "2025-10-06T14:10:57.985Z" }, + { url = "https://files.pythonhosted.org/packages/49/36/99ca3122201b382a3cf7cc937b95235b0ac944f7e9f2d5331d50821ed352/yarl-1.22.0-cp313-cp313t-musllinux_1_2_aarch64.whl", hash = "sha256:3b7c88eeef021579d600e50363e0b6ee4f7f6f728cd3486b9d0f3ee7b946398d", size = 346776, upload-time = "2025-10-06T14:10:59.633Z" }, + { url = "https://files.pythonhosted.org/packages/85/b4/47328bf996acd01a4c16ef9dcd2f59c969f495073616586f78cd5f2efb99/yarl-1.22.0-cp313-cp313t-musllinux_1_2_armv7l.whl", hash = "sha256:f4afb5c34f2c6fecdcc182dfcfc6af6cccf1aa923eed4d6a12e9d96904e1a0d8", size = 334879, upload-time = "2025-10-06T14:11:01.454Z" }, + { url = "https://files.pythonhosted.org/packages/c2/ad/b77d7b3f14a4283bffb8e92c6026496f6de49751c2f97d4352242bba3990/yarl-1.22.0-cp313-cp313t-musllinux_1_2_ppc64le.whl", hash = "sha256:59c189e3e99a59cf8d83cbb31d4db02d66cda5a1a4374e8a012b51255341abf5", size = 350996, upload-time = "2025-10-06T14:11:03.452Z" }, + { url = "https://files.pythonhosted.org/packages/81/c8/06e1d69295792ba54d556f06686cbd6a7ce39c22307100e3fb4a2c0b0a1d/yarl-1.22.0-cp313-cp313t-musllinux_1_2_s390x.whl", hash = "sha256:5a3bf7f62a289fa90f1990422dc8dff5a458469ea71d1624585ec3a4c8d6960f", size = 356047, upload-time = "2025-10-06T14:11:05.115Z" }, + { url = "https://files.pythonhosted.org/packages/4b/b8/4c0e9e9f597074b208d18cef227d83aac36184bfbc6eab204ea55783dbc5/yarl-1.22.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:de6b9a04c606978fdfe72666fa216ffcf2d1a9f6a381058d4378f8d7b1e5de62", size = 342947, upload-time = "2025-10-06T14:11:08.137Z" }, + { url = "https://files.pythonhosted.org/packages/e0/e5/11f140a58bf4c6ad7aca69a892bff0ee638c31bea4206748fc0df4ebcb3a/yarl-1.22.0-cp313-cp313t-win32.whl", hash = "sha256:1834bb90991cc2999f10f97f5f01317f99b143284766d197e43cd5b45eb18d03", size = 86943, upload-time = "2025-10-06T14:11:10.284Z" }, + { url = "https://files.pythonhosted.org/packages/31/74/8b74bae38ed7fe6793d0c15a0c8207bbb819cf287788459e5ed230996cdd/yarl-1.22.0-cp313-cp313t-win_amd64.whl", hash = "sha256:ff86011bd159a9d2dfc89c34cfd8aff12875980e3bd6a39ff097887520e60249", size = 93715, upload-time = "2025-10-06T14:11:11.739Z" }, + { url = "https://files.pythonhosted.org/packages/69/66/991858aa4b5892d57aef7ee1ba6b4d01ec3b7eb3060795d34090a3ca3278/yarl-1.22.0-cp313-cp313t-win_arm64.whl", hash = "sha256:7861058d0582b847bc4e3a4a4c46828a410bca738673f35a29ba3ca5db0b473b", size = 83857, upload-time = "2025-10-06T14:11:13.586Z" }, + { url = "https://files.pythonhosted.org/packages/46/b3/e20ef504049f1a1c54a814b4b9bed96d1ac0e0610c3b4da178f87209db05/yarl-1.22.0-cp314-cp314-macosx_10_13_universal2.whl", hash = "sha256:34b36c2c57124530884d89d50ed2c1478697ad7473efd59cfd479945c95650e4", size = 140520, upload-time = "2025-10-06T14:11:15.465Z" }, + { url = "https://files.pythonhosted.org/packages/e4/04/3532d990fdbab02e5ede063676b5c4260e7f3abea2151099c2aa745acc4c/yarl-1.22.0-cp314-cp314-macosx_10_13_x86_64.whl", hash = "sha256:0dd9a702591ca2e543631c2a017e4a547e38a5c0f29eece37d9097e04a7ac683", size = 93504, upload-time = "2025-10-06T14:11:17.106Z" }, + { url = "https://files.pythonhosted.org/packages/11/63/ff458113c5c2dac9a9719ac68ee7c947cb621432bcf28c9972b1c0e83938/yarl-1.22.0-cp314-cp314-macosx_11_0_arm64.whl", hash = "sha256:594fcab1032e2d2cc3321bb2e51271e7cd2b516c7d9aee780ece81b07ff8244b", size = 94282, upload-time = "2025-10-06T14:11:19.064Z" }, + { url = "https://files.pythonhosted.org/packages/a7/bc/315a56aca762d44a6aaaf7ad253f04d996cb6b27bad34410f82d76ea8038/yarl-1.22.0-cp314-cp314-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:f3d7a87a78d46a2e3d5b72587ac14b4c16952dd0887dbb051451eceac774411e", size = 372080, upload-time = "2025-10-06T14:11:20.996Z" }, + { url = "https://files.pythonhosted.org/packages/3f/3f/08e9b826ec2e099ea6e7c69a61272f4f6da62cb5b1b63590bb80ca2e4a40/yarl-1.22.0-cp314-cp314-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:852863707010316c973162e703bddabec35e8757e67fcb8ad58829de1ebc8590", size = 338696, upload-time = "2025-10-06T14:11:22.847Z" }, + { url = "https://files.pythonhosted.org/packages/e3/9f/90360108e3b32bd76789088e99538febfea24a102380ae73827f62073543/yarl-1.22.0-cp314-cp314-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:131a085a53bfe839a477c0845acf21efc77457ba2bcf5899618136d64f3303a2", size = 387121, upload-time = "2025-10-06T14:11:24.889Z" }, + { url = "https://files.pythonhosted.org/packages/98/92/ab8d4657bd5b46a38094cfaea498f18bb70ce6b63508fd7e909bd1f93066/yarl-1.22.0-cp314-cp314-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:078a8aefd263f4d4f923a9677b942b445a2be970ca24548a8102689a3a8ab8da", size = 394080, upload-time = "2025-10-06T14:11:27.307Z" }, + { url = "https://files.pythonhosted.org/packages/f5/e7/d8c5a7752fef68205296201f8ec2bf718f5c805a7a7e9880576c67600658/yarl-1.22.0-cp314-cp314-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:bca03b91c323036913993ff5c738d0842fc9c60c4648e5c8d98331526df89784", size = 372661, upload-time = "2025-10-06T14:11:29.387Z" }, + { url = "https://files.pythonhosted.org/packages/b6/2e/f4d26183c8db0bb82d491b072f3127fb8c381a6206a3a56332714b79b751/yarl-1.22.0-cp314-cp314-musllinux_1_2_aarch64.whl", hash = "sha256:68986a61557d37bb90d3051a45b91fa3d5c516d177dfc6dd6f2f436a07ff2b6b", size = 364645, upload-time = "2025-10-06T14:11:31.423Z" }, + { url = "https://files.pythonhosted.org/packages/80/7c/428e5812e6b87cd00ee8e898328a62c95825bf37c7fa87f0b6bb2ad31304/yarl-1.22.0-cp314-cp314-musllinux_1_2_armv7l.whl", hash = "sha256:4792b262d585ff0dff6bcb787f8492e40698443ec982a3568c2096433660c694", size = 355361, upload-time = "2025-10-06T14:11:33.055Z" }, + { url = "https://files.pythonhosted.org/packages/ec/2a/249405fd26776f8b13c067378ef4d7dd49c9098d1b6457cdd152a99e96a9/yarl-1.22.0-cp314-cp314-musllinux_1_2_ppc64le.whl", hash = "sha256:ebd4549b108d732dba1d4ace67614b9545b21ece30937a63a65dd34efa19732d", size = 381451, upload-time = "2025-10-06T14:11:35.136Z" }, + { url = "https://files.pythonhosted.org/packages/67/a8/fb6b1adbe98cf1e2dd9fad71003d3a63a1bc22459c6e15f5714eb9323b93/yarl-1.22.0-cp314-cp314-musllinux_1_2_s390x.whl", hash = "sha256:f87ac53513d22240c7d59203f25cc3beac1e574c6cd681bbfd321987b69f95fd", size = 383814, upload-time = "2025-10-06T14:11:37.094Z" }, + { url = "https://files.pythonhosted.org/packages/d9/f9/3aa2c0e480fb73e872ae2814c43bc1e734740bb0d54e8cb2a95925f98131/yarl-1.22.0-cp314-cp314-musllinux_1_2_x86_64.whl", hash = "sha256:22b029f2881599e2f1b06f8f1db2ee63bd309e2293ba2d566e008ba12778b8da", size = 370799, upload-time = "2025-10-06T14:11:38.83Z" }, + { url = "https://files.pythonhosted.org/packages/50/3c/af9dba3b8b5eeb302f36f16f92791f3ea62e3f47763406abf6d5a4a3333b/yarl-1.22.0-cp314-cp314-win32.whl", hash = "sha256:6a635ea45ba4ea8238463b4f7d0e721bad669f80878b7bfd1f89266e2ae63da2", size = 82990, upload-time = "2025-10-06T14:11:40.624Z" }, + { url = "https://files.pythonhosted.org/packages/ac/30/ac3a0c5bdc1d6efd1b41fa24d4897a4329b3b1e98de9449679dd327af4f0/yarl-1.22.0-cp314-cp314-win_amd64.whl", hash = "sha256:0d6e6885777af0f110b0e5d7e5dda8b704efed3894da26220b7f3d887b839a79", size = 88292, upload-time = "2025-10-06T14:11:42.578Z" }, + { url = "https://files.pythonhosted.org/packages/df/0a/227ab4ff5b998a1b7410abc7b46c9b7a26b0ca9e86c34ba4b8d8bc7c63d5/yarl-1.22.0-cp314-cp314-win_arm64.whl", hash = "sha256:8218f4e98d3c10d683584cb40f0424f4b9fd6e95610232dd75e13743b070ee33", size = 82888, upload-time = "2025-10-06T14:11:44.863Z" }, + { url = "https://files.pythonhosted.org/packages/06/5e/a15eb13db90abd87dfbefb9760c0f3f257ac42a5cac7e75dbc23bed97a9f/yarl-1.22.0-cp314-cp314t-macosx_10_13_universal2.whl", hash = "sha256:45c2842ff0e0d1b35a6bf1cd6c690939dacb617a70827f715232b2e0494d55d1", size = 146223, upload-time = "2025-10-06T14:11:46.796Z" }, + { url = "https://files.pythonhosted.org/packages/18/82/9665c61910d4d84f41a5bf6837597c89e665fa88aa4941080704645932a9/yarl-1.22.0-cp314-cp314t-macosx_10_13_x86_64.whl", hash = "sha256:d947071e6ebcf2e2bee8fce76e10faca8f7a14808ca36a910263acaacef08eca", size = 95981, upload-time = "2025-10-06T14:11:48.845Z" }, + { url = "https://files.pythonhosted.org/packages/5d/9a/2f65743589809af4d0a6d3aa749343c4b5f4c380cc24a8e94a3c6625a808/yarl-1.22.0-cp314-cp314t-macosx_11_0_arm64.whl", hash = "sha256:334b8721303e61b00019474cc103bdac3d7b1f65e91f0bfedeec2d56dfe74b53", size = 97303, upload-time = "2025-10-06T14:11:50.897Z" }, + { url = "https://files.pythonhosted.org/packages/b0/ab/5b13d3e157505c43c3b43b5a776cbf7b24a02bc4cccc40314771197e3508/yarl-1.22.0-cp314-cp314t-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:1e7ce67c34138a058fd092f67d07a72b8e31ff0c9236e751957465a24b28910c", size = 361820, upload-time = "2025-10-06T14:11:52.549Z" }, + { url = "https://files.pythonhosted.org/packages/fb/76/242a5ef4677615cf95330cfc1b4610e78184400699bdda0acb897ef5e49a/yarl-1.22.0-cp314-cp314t-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:d77e1b2c6d04711478cb1c4ab90db07f1609ccf06a287d5607fcd90dc9863acf", size = 323203, upload-time = "2025-10-06T14:11:54.225Z" }, + { url = "https://files.pythonhosted.org/packages/8c/96/475509110d3f0153b43d06164cf4195c64d16999e0c7e2d8a099adcd6907/yarl-1.22.0-cp314-cp314t-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:c4647674b6150d2cae088fc07de2738a84b8bcedebef29802cf0b0a82ab6face", size = 363173, upload-time = "2025-10-06T14:11:56.069Z" }, + { url = "https://files.pythonhosted.org/packages/c9/66/59db471aecfbd559a1fd48aedd954435558cd98c7d0da8b03cc6c140a32c/yarl-1.22.0-cp314-cp314t-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:efb07073be061c8f79d03d04139a80ba33cbd390ca8f0297aae9cce6411e4c6b", size = 373562, upload-time = "2025-10-06T14:11:58.783Z" }, + { url = "https://files.pythonhosted.org/packages/03/1f/c5d94abc91557384719da10ff166b916107c1b45e4d0423a88457071dd88/yarl-1.22.0-cp314-cp314t-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:e51ac5435758ba97ad69617e13233da53908beccc6cfcd6c34bbed8dcbede486", size = 339828, upload-time = "2025-10-06T14:12:00.686Z" }, + { url = "https://files.pythonhosted.org/packages/5f/97/aa6a143d3afba17b6465733681c70cf175af89f76ec8d9286e08437a7454/yarl-1.22.0-cp314-cp314t-musllinux_1_2_aarch64.whl", hash = "sha256:33e32a0dd0c8205efa8e83d04fc9f19313772b78522d1bdc7d9aed706bfd6138", size = 347551, upload-time = "2025-10-06T14:12:02.628Z" }, + { url = "https://files.pythonhosted.org/packages/43/3c/45a2b6d80195959239a7b2a8810506d4eea5487dce61c2a3393e7fc3c52e/yarl-1.22.0-cp314-cp314t-musllinux_1_2_armv7l.whl", hash = "sha256:bf4a21e58b9cde0e401e683ebd00f6ed30a06d14e93f7c8fd059f8b6e8f87b6a", size = 334512, upload-time = "2025-10-06T14:12:04.871Z" }, + { url = "https://files.pythonhosted.org/packages/86/a0/c2ab48d74599c7c84cb104ebd799c5813de252bea0f360ffc29d270c2caa/yarl-1.22.0-cp314-cp314t-musllinux_1_2_ppc64le.whl", hash = "sha256:e4b582bab49ac33c8deb97e058cd67c2c50dac0dd134874106d9c774fd272529", size = 352400, upload-time = "2025-10-06T14:12:06.624Z" }, + { url = "https://files.pythonhosted.org/packages/32/75/f8919b2eafc929567d3d8411f72bdb1a2109c01caaab4ebfa5f8ffadc15b/yarl-1.22.0-cp314-cp314t-musllinux_1_2_s390x.whl", hash = "sha256:0b5bcc1a9c4839e7e30b7b30dd47fe5e7e44fb7054ec29b5bb8d526aa1041093", size = 357140, upload-time = "2025-10-06T14:12:08.362Z" }, + { url = "https://files.pythonhosted.org/packages/cf/72/6a85bba382f22cf78add705d8c3731748397d986e197e53ecc7835e76de7/yarl-1.22.0-cp314-cp314t-musllinux_1_2_x86_64.whl", hash = "sha256:c0232bce2170103ec23c454e54a57008a9a72b5d1c3105dc2496750da8cfa47c", size = 341473, upload-time = "2025-10-06T14:12:10.994Z" }, + { url = "https://files.pythonhosted.org/packages/35/18/55e6011f7c044dc80b98893060773cefcfdbf60dfefb8cb2f58b9bacbd83/yarl-1.22.0-cp314-cp314t-win32.whl", hash = "sha256:8009b3173bcd637be650922ac455946197d858b3630b6d8787aa9e5c4564533e", size = 89056, upload-time = "2025-10-06T14:12:13.317Z" }, + { url = "https://files.pythonhosted.org/packages/f9/86/0f0dccb6e59a9e7f122c5afd43568b1d31b8ab7dda5f1b01fb5c7025c9a9/yarl-1.22.0-cp314-cp314t-win_amd64.whl", hash = "sha256:9fb17ea16e972c63d25d4a97f016d235c78dd2344820eb35bc034bc32012ee27", size = 96292, upload-time = "2025-10-06T14:12:15.398Z" }, + { url = "https://files.pythonhosted.org/packages/48/b7/503c98092fb3b344a179579f55814b613c1fbb1c23b3ec14a7b008a66a6e/yarl-1.22.0-cp314-cp314t-win_arm64.whl", hash = "sha256:9f6d73c1436b934e3f01df1e1b21ff765cd1d28c77dfb9ace207f746d4610ee1", size = 85171, upload-time = "2025-10-06T14:12:16.935Z" }, + { url = "https://files.pythonhosted.org/packages/94/fd/6480106702a79bcceda5fd9c63cb19a04a6506bd5ce7fd8d9b63742f0021/yarl-1.22.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:3aa27acb6de7a23785d81557577491f6c38a5209a254d1191519d07d8fe51748", size = 141301, upload-time = "2025-10-06T14:12:19.01Z" }, + { url = "https://files.pythonhosted.org/packages/42/e1/6d95d21b17a93e793e4ec420a925fe1f6a9342338ca7a563ed21129c0990/yarl-1.22.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:af74f05666a5e531289cb1cc9c883d1de2088b8e5b4de48004e5ca8a830ac859", size = 93864, upload-time = "2025-10-06T14:12:21.05Z" }, + { url = "https://files.pythonhosted.org/packages/32/58/b8055273c203968e89808413ea4c984988b6649baabf10f4522e67c22d2f/yarl-1.22.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:62441e55958977b8167b2709c164c91a6363e25da322d87ae6dd9c6019ceecf9", size = 94706, upload-time = "2025-10-06T14:12:23.287Z" }, + { url = "https://files.pythonhosted.org/packages/18/91/d7bfbc28a88c2895ecd0da6a874def0c147de78afc52c773c28e1aa233a3/yarl-1.22.0-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:b580e71cac3f8113d3135888770903eaf2f507e9421e5697d6ee6d8cd1c7f054", size = 347100, upload-time = "2025-10-06T14:12:28.527Z" }, + { url = "https://files.pythonhosted.org/packages/bd/e8/37a1e7b99721c0564b1fc7b0a4d1f595ef6fb8060d82ca61775b644185f7/yarl-1.22.0-cp39-cp39-manylinux2014_armv7l.manylinux_2_17_armv7l.manylinux_2_31_armv7l.whl", hash = "sha256:e81fda2fb4a07eda1a2252b216aa0df23ebcd4d584894e9612e80999a78fd95b", size = 318902, upload-time = "2025-10-06T14:12:30.528Z" }, + { url = "https://files.pythonhosted.org/packages/1c/ef/34724449d7ef2db4f22df644f2dac0b8a275d20f585e526937b3ae47b02d/yarl-1.22.0-cp39-cp39-manylinux2014_ppc64le.manylinux_2_17_ppc64le.manylinux_2_28_ppc64le.whl", hash = "sha256:99b6fc1d55782461b78221e95fc357b47ad98b041e8e20f47c1411d0aacddc60", size = 363302, upload-time = "2025-10-06T14:12:32.295Z" }, + { url = "https://files.pythonhosted.org/packages/8a/04/88a39a5dad39889f192cce8d66cc4c58dbeca983e83f9b6bf23822a7ed91/yarl-1.22.0-cp39-cp39-manylinux2014_s390x.manylinux_2_17_s390x.manylinux_2_28_s390x.whl", hash = "sha256:088e4e08f033db4be2ccd1f34cf29fe994772fb54cfe004bbf54db320af56890", size = 370816, upload-time = "2025-10-06T14:12:34.01Z" }, + { url = "https://files.pythonhosted.org/packages/6b/1f/5e895e547129413f56c76be2c3ce4b96c797d2d0ff3e16a817d9269b12e6/yarl-1.22.0-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2e4e1f6f0b4da23e61188676e3ed027ef0baa833a2e633c29ff8530800edccba", size = 346465, upload-time = "2025-10-06T14:12:35.977Z" }, + { url = "https://files.pythonhosted.org/packages/11/13/a750e9fd6f9cc9ed3a52a70fe58ffe505322f0efe0d48e1fd9ffe53281f5/yarl-1.22.0-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:84fc3ec96fce86ce5aa305eb4aa9358279d1aa644b71fab7b8ed33fe3ba1a7ca", size = 341506, upload-time = "2025-10-06T14:12:37.788Z" }, + { url = "https://files.pythonhosted.org/packages/3c/67/bb6024de76e7186611ebe626aec5b71a2d2ecf9453e795f2dbd80614784c/yarl-1.22.0-cp39-cp39-musllinux_1_2_armv7l.whl", hash = "sha256:5dbeefd6ca588b33576a01b0ad58aa934bc1b41ef89dee505bf2932b22ddffba", size = 335030, upload-time = "2025-10-06T14:12:39.775Z" }, + { url = "https://files.pythonhosted.org/packages/a2/be/50b38447fd94a7992996a62b8b463d0579323fcfc08c61bdba949eef8a5d/yarl-1.22.0-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:14291620375b1060613f4aab9ebf21850058b6b1b438f386cc814813d901c60b", size = 358560, upload-time = "2025-10-06T14:12:41.547Z" }, + { url = "https://files.pythonhosted.org/packages/e2/89/c020b6f547578c4e3dbb6335bf918f26e2f34ad0d1e515d72fd33ac0c635/yarl-1.22.0-cp39-cp39-musllinux_1_2_s390x.whl", hash = "sha256:a4fcfc8eb2c34148c118dfa02e6427ca278bfd0f3df7c5f99e33d2c0e81eae3e", size = 357290, upload-time = "2025-10-06T14:12:43.861Z" }, + { url = "https://files.pythonhosted.org/packages/8c/52/c49a619ee35a402fa3a7019a4fa8d26878fec0d1243f6968bbf516789578/yarl-1.22.0-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:029866bde8d7b0878b9c160e72305bbf0a7342bcd20b9999381704ae03308dc8", size = 350700, upload-time = "2025-10-06T14:12:46.868Z" }, + { url = "https://files.pythonhosted.org/packages/ab/c9/f5042d87777bf6968435f04a2bbb15466b2f142e6e47fa4f34d1a3f32f0c/yarl-1.22.0-cp39-cp39-win32.whl", hash = "sha256:4dcc74149ccc8bba31ce1944acee24813e93cfdee2acda3c172df844948ddf7b", size = 82323, upload-time = "2025-10-06T14:12:48.633Z" }, + { url = "https://files.pythonhosted.org/packages/fd/58/d00f7cad9eba20c4eefac2682f34661d1d1b3a942fc0092eb60e78cfb733/yarl-1.22.0-cp39-cp39-win_amd64.whl", hash = "sha256:10619d9fdee46d20edc49d3479e2f8269d0779f1b031e6f7c2aa1c76be04b7ed", size = 87145, upload-time = "2025-10-06T14:12:50.241Z" }, + { url = "https://files.pythonhosted.org/packages/c2/a3/70904f365080780d38b919edd42d224b8c4ce224a86950d2eaa2a24366ad/yarl-1.22.0-cp39-cp39-win_arm64.whl", hash = "sha256:dd7afd3f8b0bfb4e0d9fc3c31bfe8a4ec7debe124cfd90619305def3c8ca8cd2", size = 82173, upload-time = "2025-10-06T14:12:51.869Z" }, + { url = "https://files.pythonhosted.org/packages/73/ae/b48f95715333080afb75a4504487cbe142cae1268afc482d06692d605ae6/yarl-1.22.0-py3-none-any.whl", hash = "sha256:1380560bdba02b6b6c90de54133c81c9f2a453dee9912fe58c1dcced1edb7cff", size = 46814, upload-time = "2025-10-06T14:12:53.872Z" }, +] + +[[package]] +name = "zipp" +version = "3.20.2" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version < '3.9'", +] +sdist = { url = "https://files.pythonhosted.org/packages/54/bf/5c0000c44ebc80123ecbdddba1f5dcd94a5ada602a9c225d84b5aaa55e86/zipp-3.20.2.tar.gz", hash = "sha256:bc9eb26f4506fda01b81bcde0ca78103b6e62f991b381fec825435c836edbc29", size = 24199, upload-time = "2024-09-13T13:44:16.101Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/62/8b/5ba542fa83c90e09eac972fc9baca7a88e7e7ca4b221a89251954019308b/zipp-3.20.2-py3-none-any.whl", hash = "sha256:a817ac80d6cf4b23bf7f2828b7cabf326f15a001bea8b1f9b49631780ba28350", size = 9200, upload-time = "2024-09-13T13:44:14.38Z" }, +] + +[[package]] +name = "zipp" +version = "3.23.0" +source = { registry = "https://pypi.org/simple" } +resolution-markers = [ + "python_full_version == '3.9.*'", +] +sdist = { url = "https://files.pythonhosted.org/packages/e3/02/0f2892c661036d50ede074e376733dca2ae7c6eb617489437771209d4180/zipp-3.23.0.tar.gz", hash = "sha256:a07157588a12518c9d4034df3fbbee09c814741a33ff63c05fa29d26a2404166", size = 25547, upload-time = "2025-06-08T17:06:39.4Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/2e/54/647ade08bf0db230bfea292f893923872fd20be6ac6f53b2b936ba839d75/zipp-3.23.0-py3-none-any.whl", hash = "sha256:071652d6115ed432f5ce1d34c336c0adfd6a884660d1e9712a256d3d3bd4b14e", size = 10276, upload-time = "2025-06-08T17:06:38.034Z" }, +] From 57190a8b1b04813f6210af41a6ec275ea0c728c1 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Fri, 24 Oct 2025 10:13:47 -0700 Subject: [PATCH 10/21] Switch to socket mode, but it doesn't work Signed-off-by: Sid Murching --- slack-bot/.env.example | 5 + slack-bot/RUN.md | 5 +- slack-bot/SOCKET_MODE_SETUP.md | 242 +++++++++++++++++++++++++++++++++ slack-bot/app.py | 52 +++++-- slack-bot/config.py | 3 + 5 files changed, 292 insertions(+), 15 deletions(-) create mode 100644 slack-bot/SOCKET_MODE_SETUP.md diff --git a/slack-bot/.env.example b/slack-bot/.env.example index 43428d1..f297428 100644 --- a/slack-bot/.env.example +++ b/slack-bot/.env.example @@ -4,6 +4,11 @@ SLACK_SIGNING_SECRET=your-signing-secret-here SLACK_CLIENT_ID=your-client-id-here SLACK_CLIENT_SECRET=your-client-secret-here +# Socket Mode (recommended for local development) +# Create app-level token with connections:write scope +# Get from: https://api.slack.com/apps/YOUR_APP_ID/app-level-tokens +SLACK_APP_TOKEN=xapp-your-app-level-token-here + # Databricks Configuration DATABRICKS_HOST=https://your-workspace.databricks.com # For local development with databricks-cli, leave CLIENT_ID and CLIENT_SECRET empty diff --git a/slack-bot/RUN.md b/slack-bot/RUN.md index 84aee2e..786a8cc 100644 --- a/slack-bot/RUN.md +++ b/slack-bot/RUN.md @@ -38,11 +38,14 @@ cp .env.example .env The app will automatically load from `.env.local` first, then fall back to `.env`. Required variables: -- `SLACK_BOT_TOKEN` - Your Slack bot token +- `SLACK_BOT_TOKEN` - Your Slack bot token (xoxb-...) - `SLACK_SIGNING_SECRET` - Your Slack signing secret +- `SLACK_APP_TOKEN` - Your app-level token for Socket Mode (xapp-...) - `DATABRICKS_HOST` - Your Databricks workspace URL - `SERVING_ENDPOINT_NAME` - Your agent endpoint name +**Important**: You need `SLACK_APP_TOKEN` for Socket Mode (local development). See `SOCKET_MODE_SETUP.md` for detailed setup instructions. + ## Development Workflow ### Install with dev dependencies diff --git a/slack-bot/SOCKET_MODE_SETUP.md b/slack-bot/SOCKET_MODE_SETUP.md new file mode 100644 index 0000000..13e3fa9 --- /dev/null +++ b/slack-bot/SOCKET_MODE_SETUP.md @@ -0,0 +1,242 @@ +# Socket Mode Setup for Local Development + +Socket Mode allows you to run the Slack bot on your laptop without exposing a public URL. It uses WebSockets to connect to Slack. + +## Why Socket Mode? + +**Without Socket Mode (HTTP Mode):** +- āŒ Need public URL (ngrok, port forwarding, etc.) +- āŒ Configure Event Subscriptions URL in Slack +- āŒ Complex firewall setup +- āŒ Harder to debug + +**With Socket Mode:** +- āœ… No public URL needed +- āœ… Runs entirely on localhost +- āœ… Perfect for development +- āœ… Easy to debug + +## Setup Steps + +### 1. Enable Socket Mode in Slack App + +1. Go to [api.slack.com/apps](https://api.slack.com/apps) +2. Select your app +3. Click **Socket Mode** in the left sidebar +4. **Enable Socket Mode** +5. You'll be prompted to create an app-level token + +### 2. Create App-Level Token + +1. Click **Generate Token and Scopes** +2. **Token Name**: `socket-mode-token` (or any name) +3. **Add Scopes**: + - `connections:write` (required for Socket Mode) +4. Click **Generate** +5. **Copy the token** (starts with `xapp-`) + - āš ļø You can only see this once! Save it securely. + +### 3. Update Your `.env.local` + +Add the app-level token: + +```bash +# Existing tokens +SLACK_BOT_TOKEN=xoxb-your-bot-token +SLACK_SIGNING_SECRET=your-signing-secret +SLACK_CLIENT_ID=your-client-id +SLACK_CLIENT_SECRET=your-client-secret + +# Add this for Socket Mode +SLACK_APP_TOKEN=xapp-1-A07xxxxx-7724xxxxx-xxxxxxxxxxxxx + +# Other config +DATABRICKS_HOST=https://your-workspace.databricks.com +SERVING_ENDPOINT_NAME=your-endpoint-name +PORT=3000 +``` + +### 4. Disable Event Subscriptions URL (Optional) + +Since you're using Socket Mode, you don't need the Event Subscriptions URL: + +1. Go to **Event Subscriptions** +2. You can leave **Enable Events** ON (Socket Mode still uses these subscriptions) +3. But **Request URL** is not needed (can be left empty or disabled) + +### 5. Run the Bot + +```bash +python app.py +``` + +You should see: + +``` +Loaded environment from: .env.local +INFO - Starting bot in Socket Mode (WebSocket)... +INFO - OAuth callback available at: http://localhost:3000/oauth/callback +INFO - OAuth callback server running on port 3000 +INFO - Socket Mode handler starting... +INFO - A new session has been established +``` + +## How It Works + +### Architecture + +``` +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” WebSocket ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ │◄──────────────────────────►│ │ +│ Your Bot │ Slack Events & Commands │ Slack │ +│ (localhost) │ │ │ +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ + │ + │ HTTP (localhost only) + ā–¼ +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ Browser │ OAuth Callback +│ (User) │ http://localhost:3000/oauth/callback +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ +``` + +### Two Connections + +1. **WebSocket (Socket Mode)**: + - Slack events (messages, mentions, etc.) → Your bot + - Bot responses → Slack + - No public URL needed + +2. **HTTP (OAuth Callback)**: + - User clicks "Sign in to Databricks" + - Browser redirects to Databricks OAuth + - Databricks redirects back to `http://localhost:3000/oauth/callback` + - Only needs localhost, no public URL + +## Verification + +### Check Socket Mode is Working + +When you start the bot, look for: + +``` +INFO - Starting bot in Socket Mode (WebSocket)... +INFO - Socket Mode handler starting... +``` + +NOT this: +``` +INFO - No SLACK_APP_TOKEN found, starting in HTTP mode... +``` + +### Test the Bot + +1. **Send a message** in Slack to a channel where the bot is installed +2. Check bot logs - you should see: + ``` + INFO - Processing message from user U123456: hello + ``` + +3. If authentication is needed, bot will respond with ephemeral message (only you can see) + +## Troubleshooting + +### "No SLACK_APP_TOKEN found, starting in HTTP mode" + +**Cause**: App-level token not set or not loaded from `.env.local` + +**Solution**: +```bash +# Check .env.local has the token +cat .env.local | grep SLACK_APP_TOKEN + +# Should show: +SLACK_APP_TOKEN=xapp-1-... + +# If missing, add it and restart +``` + +### "Connection to Slack failed" + +**Cause**: Socket Mode not enabled or token invalid + +**Solution**: +1. Verify Socket Mode is enabled in Slack app settings +2. Verify token starts with `xapp-` +3. Regenerate token if needed + +### "Events not being received" + +**Cause**: Event subscriptions not configured + +**Solution**: +1. Go to **Event Subscriptions** in Slack app +2. Verify you're subscribed to events like: + - `message.channels` + - `message.groups` + - `message.im` + - `app_mention` + +### OAuth callback not working + +**Cause**: Flask server not starting or port blocked + +**Solution**: +```bash +# Check port is free +lsof -i :3000 + +# Try different port in .env.local +PORT=3001 +``` + +## HTTP Mode (Alternative) + +If you prefer HTTP mode (requires public URL): + +1. **Don't set** `SLACK_APP_TOKEN` in `.env.local` +2. **Expose your localhost** using ngrok: + ```bash + ngrok http 3000 + ``` +3. **Configure Event Subscriptions URL** in Slack: + ``` + https://your-ngrok-url.ngrok.io/slack/events + ``` +4. **Run the bot**: + ```bash + python app.py + ``` + +Bot will automatically use HTTP mode if no `SLACK_APP_TOKEN` is found. + +## Production Deployment + +For production, you have two options: + +### Option 1: Socket Mode (Easier) +- Keep using Socket Mode +- Deploy on any server (doesn't need public URL) +- Just set `SLACK_APP_TOKEN` in production environment + +### Option 2: HTTP Mode (More Scalable) +- Don't set `SLACK_APP_TOKEN` +- Set up public HTTPS endpoint +- Configure Event Subscriptions URL in Slack +- Use production WSGI server (Gunicorn, uWSGI) + +Socket Mode is simpler but HTTP mode can be more scalable for high-traffic bots. + +## Security Notes + +- āœ… App-level tokens are sensitive - never commit to git +- āœ… `.env.local` is in `.gitignore` +- āœ… Use separate tokens for dev/staging/prod +- āœ… Rotate tokens if compromised +- āœ… Socket Mode connection is encrypted (WSS) + +## References + +- [Slack Socket Mode Documentation](https://api.slack.com/apis/connections/socket) +- [Socket Mode vs HTTP Mode](https://api.slack.com/apis/connections/socket-implement) +- [App-Level Tokens](https://api.slack.com/authentication/token-types#app) diff --git a/slack-bot/app.py b/slack-bot/app.py index e84e437..843c1a0 100644 --- a/slack-bot/app.py +++ b/slack-bot/app.py @@ -322,20 +322,44 @@ async def custom_error_handler(error, body, logger): def main(): """Main entry point""" try: - # HTTP mode with Flask for OAuth callback support - logger.info(f"Starting bot in HTTP mode on port {CONFIG.PORT}...") - logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") - - from slack_bolt.adapter.flask import SlackRequestHandler - handler = SlackRequestHandler(app) - - # Register Slack events endpoint - @flask_app.route("/slack/events", methods=["POST"]) - def slack_events(): - return handler.handle(request) - - # Start Flask server - flask_app.run(port=CONFIG.PORT, debug=False) + # Use Socket Mode for local development (recommended) + # Socket Mode uses WebSocket connection, no public URL needed + slack_app_token = getattr(CONFIG, 'SLACK_APP_TOKEN', None) + + if slack_app_token: + logger.info("Starting bot in Socket Mode (WebSocket)...") + logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") + + # Start Flask server in background thread for OAuth callbacks + import threading + flask_thread = threading.Thread( + target=lambda: flask_app.run(port=CONFIG.PORT, debug=False, use_reloader=False), + daemon=True + ) + flask_thread.start() + logger.info(f"OAuth callback server running on port {CONFIG.PORT}") + + # Start Socket Mode handler (blocking) + handler = SocketModeHandler(app, slack_app_token) + logger.info("Socket Mode handler starting...") + handler.start() + + else: + # Fallback to HTTP mode if no app token provided + logger.info("No SLACK_APP_TOKEN found, starting in HTTP mode...") + logger.info(f"HTTP mode requires public URL for Slack events") + logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") + + from slack_bolt.adapter.flask import SlackRequestHandler + handler = SlackRequestHandler(app) + + # Register Slack events endpoint + @flask_app.route("/slack/events", methods=["POST"]) + def slack_events(): + return handler.handle(request) + + # Start Flask server + flask_app.run(port=CONFIG.PORT, debug=False) except KeyboardInterrupt: logger.info("Bot stopped by user") diff --git a/slack-bot/config.py b/slack-bot/config.py index 6db270c..61f69f8 100644 --- a/slack-bot/config.py +++ b/slack-bot/config.py @@ -16,6 +16,9 @@ class DefaultConfig: SLACK_CLIENT_ID = os.environ.get("SLACK_CLIENT_ID", "") SLACK_CLIENT_SECRET = os.environ.get("SLACK_CLIENT_SECRET", "") + # Socket Mode (for local development - no public URL needed) + SLACK_APP_TOKEN = os.environ.get("SLACK_APP_TOKEN", "") + # OAuth settings for Databricks DATABRICKS_HOST = os.environ.get("DATABRICKS_HOST", "") DATABRICKS_CLIENT_ID = os.environ.get("DATABRICKS_CLIENT_ID", "") From 8fdfd522ff5783c5ace3a8764cb2d07e09de961f Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Fri, 24 Oct 2025 11:16:33 -0700 Subject: [PATCH 11/21] Bot replies work, but oauth flow does not Signed-off-by: Sid Murching --- slack-bot/RUN.md | 208 ++++++++++---- slack-bot/app.py | 383 ++------------------------ slack-bot/handlers/message_handler.py | 17 +- 3 files changed, 199 insertions(+), 409 deletions(-) diff --git a/slack-bot/RUN.md b/slack-bot/RUN.md index 786a8cc..66b2e8a 100644 --- a/slack-bot/RUN.md +++ b/slack-bot/RUN.md @@ -1,6 +1,6 @@ -# Running the Slack Bot with uv +# Running the Slack Bot (Updated Architecture) -## Quick Start with uv +## Quick Start ### 1. Install dependencies (one-time setup) @@ -8,43 +8,125 @@ uv pip install -r requirements.txt ``` -### 2. Run the bot +### 2. Configure environment ```bash -python app.py +cp .env.example .env.local +# Edit .env.local with your credentials ``` -### 3. Alternative: One-liner (uv handles everything) +### 3. Run the bot ```bash -# uv will handle dependencies automatically -uv run --with slack-bolt --with slack-sdk --with Flask --with databricks-sdk --with httpx --with python-dotenv python app.py +python app.py # Auto-detects mode +# OR +python app_socket.py # Force Socket Mode (recommended) ``` -## Setup Environment Variables +## File Structure + +``` +app.py # šŸš€ Auto-detects and launches +app_socket.py # ⚔ Socket Mode (clean, minimal) +http_mode.py # 🌐 HTTP/Events API (production) +oauth_server.py # šŸ” OAuth callback handler (shared) +``` + +## Modes + +### Socket Mode (Recommended) -Before running, create a `.env` or `.env.local` file: +**Best for:** Local development, simple deployments +**What you need:** ```bash -# Option 1: Use .env.local (recommended for local development) -cp .env.example .env.local -# Edit .env.local with your credentials +SLACK_BOT_TOKEN=xoxb-... +SLACK_APP_TOKEN=xapp-... # Required! +DATABRICKS_HOST=https://... +SERVING_ENDPOINT_NAME=... +``` + +**Run:** +```bash +python app_socket.py +``` + +**Advantages:** +- āœ… No public URL needed +- āœ… Works on laptop +- āœ… Easy to debug +- āœ… Clean, minimal code + +### HTTP Mode + +**Best for:** Production deployments + +**What you need:** +- Public HTTPS URL +- Event Subscriptions configured in Slack + +**Run:** +```bash +python http_mode.py +``` + +## Environment Variables + +### Required + +```bash +# Slack +SLACK_BOT_TOKEN=xoxb-... +SLACK_APP_TOKEN=xapp-... # For Socket Mode +SLACK_SIGNING_SECRET=... + +# Databricks +DATABRICKS_HOST=https://your-workspace.databricks.com +SERVING_ENDPOINT_NAME=your-endpoint-name +``` + +### Optional -# Option 2: Use .env -cp .env.example .env -# Edit .env with your credentials +```bash +# OAuth (defaults to databricks-cli) +DATABRICKS_CLIENT_ID= +DATABRICKS_CLIENT_SECRET= + +# Server +PORT=3000 + +# State storage +STATE_STORAGE=memory +REDIS_URL=redis://localhost:6379 ``` -The app will automatically load from `.env.local` first, then fall back to `.env`. +**Important:** +- Don't set `SLACK_CLIENT_ID` or `SLACK_CLIENT_SECRET` (only needed for multi-workspace OAuth) +- Leave `DATABRICKS_CLIENT_ID` empty to use `databricks-cli` (easier for development) + +## Setup Guide -Required variables: -- `SLACK_BOT_TOKEN` - Your Slack bot token (xoxb-...) -- `SLACK_SIGNING_SECRET` - Your Slack signing secret -- `SLACK_APP_TOKEN` - Your app-level token for Socket Mode (xapp-...) -- `DATABRICKS_HOST` - Your Databricks workspace URL -- `SERVING_ENDPOINT_NAME` - Your agent endpoint name +See `SOCKET_MODE_SETUP.md` for detailed Socket Mode setup. -**Important**: You need `SLACK_APP_TOKEN` for Socket Mode (local development). See `SOCKET_MODE_SETUP.md` for detailed setup instructions. +See `OAUTH_SETUP.md` for Databricks OAuth configuration. + +See `README_ARCHITECTURE.md` for architecture details. + +## Troubleshooting + +See `TROUBLESHOOTING.md` for common issues. + +**Quick checks:** +```bash +# Test configuration +python test_connection.py + +# Check bot is running +ps aux | grep python + +# View logs (Socket Mode) +# Logs print to stdout +``` ## Development Workflow @@ -72,44 +154,74 @@ uv run ruff check . - **Fast**: 10-100x faster than pip - **Reliable**: Deterministic dependency resolution - **Simple**: No need for venv management -- **Modern**: Built in Rust, replaces pip, pip-tools, and virtualenv +- **Modern**: Built in Rust -## Troubleshooting - -### "uv: command not found" +## What Changed? -Install uv: -```bash -curl -LsSf https://astral.sh/uv/install.sh | sh +### Old Structure +``` +app.py (500+ lines) +ā”œā”€ Socket Mode code +ā”œā”€ HTTP Mode code +ā”œā”€ OAuth callback +└─ Complex branching ``` -Or with pip: -```bash -pip install uv +### New Structure +``` +app.py (50 lines) # Simple launcher +app_socket.py (180 lines) # Clean Socket Mode +http_mode.py (150 lines) # Clean HTTP Mode +oauth_server.py (100 lines) # Shared OAuth ``` -### Dependencies not found +**Benefits:** +- āœ… Easier to understand +- āœ… Easier to debug +- āœ… Cleaner separation +- āœ… Better for troubleshooting + +## Examples -Make sure you're in the slack-bot directory: +### Start Socket Mode ```bash -cd slack-bot -uv sync +$ python app_socket.py + +āœ… Loaded environment from: .env.local +INFO - āœ… Slack bot initialized successfully +INFO - šŸš€ Starting bot in Socket Mode... +INFO - šŸ” OAuth callback server running on port 3000 +INFO - ⚔ Socket Mode handler starting... +INFO - A new session has been established +INFO - āš”ļø Bolt app is running! ``` -### Port already in use - -Change the port in `.env`: +### Start HTTP Mode ```bash -PORT=3001 +$ python http_mode.py + +INFO - Starting bot in HTTP mode on port 3000... +INFO - Slack events endpoint: http://localhost:3000/slack/events +WARNING - HTTP mode requires public HTTPS URL for production! ``` -## Alternative: Traditional venv +### Auto-detect Mode +```bash +$ python app.py -If you prefer traditional venv: +Loaded environment from: .env.local +šŸš€ Socket Mode detected (SLACK_APP_TOKEN found) + Starting Socket Mode bot... -```bash -python -m venv venv -source venv/bin/activate # On Windows: venv\Scripts\activate -pip install -r requirements.txt -python app.py +INFO - āœ… Slack bot initialized successfully +... ``` + +## Next Steps + +1. āœ… Run `python app_socket.py` +2. āœ… Invite bot to Slack channel: `/invite @YourBot` +3. āœ… Tag the bot: `@YourBot hello` +4. āœ… Check logs for `šŸ“Ø Message from user...` +5. āœ… Click "Sign in to Databricks" when prompted +6. āœ… Start chatting! diff --git a/slack-bot/app.py b/slack-bot/app.py index 843c1a0..6dd9ab9 100644 --- a/slack-bot/app.py +++ b/slack-bot/app.py @@ -1,24 +1,23 @@ #!/usr/bin/env python3 """ -Slack bot for Databricks Agent integration. +Slack Bot Launcher - Auto-detects mode and runs appropriate version. -This bot provides a Slack interface to Databricks agents, handling: -- OAuth authentication with Databricks -- Message handling in channels and threads -- Conversation history management -- Tool call visualization +Modes: +- Socket Mode (default): For local development, no public URL needed +- HTTP Mode: For production with public HTTPS endpoint + +Usage: + python app.py # Auto-detect based on SLACK_APP_TOKEN + python app_socket.py # Force Socket Mode + python http_mode.py # Force HTTP Mode """ -import logging -import sys import os +import sys from pathlib import Path from dotenv import load_dotenv -from slack_bolt import App -from slack_bolt.adapter.socket_mode import SocketModeHandler -from flask import Flask, request, redirect -# Load environment variables from .env.local or .env +# Load environment variables env_file = Path(__file__).parent / ".env.local" if env_file.exists(): load_dotenv(env_file) @@ -28,347 +27,25 @@ if env_file.exists(): load_dotenv(env_file) print(f"Loaded environment from: {env_file}") - else: - print("No .env or .env.local file found, using system environment variables") - -from config import DefaultConfig -from storage import StateManager -from handlers import AuthHandler, handle_message_event -from client import DatabricksClient - -# Configure logging -logging.basicConfig( - level=logging.INFO, - format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', - handlers=[ - logging.StreamHandler(sys.stdout) - ] -) -logger = logging.getLogger(__name__) - -# Load configuration -CONFIG = DefaultConfig() - -# Validate required configuration -required_configs = [ - "SLACK_BOT_TOKEN", - "SLACK_SIGNING_SECRET", - "DATABRICKS_HOST", - "SERVING_ENDPOINT_NAME" -] -missing_configs = [c for c in required_configs if not getattr(CONFIG, c)] -if missing_configs: - logger.error(f"Missing required configuration: {', '.join(missing_configs)}") - sys.exit(1) - -# Initialize Slack app -app = App( - token=CONFIG.SLACK_BOT_TOKEN, - signing_secret=CONFIG.SLACK_SIGNING_SECRET -) - -# Initialize components -state_manager = StateManager() -auth_handler = AuthHandler(CONFIG, state_manager) -databricks_client = DatabricksClient( - CONFIG.DATABRICKS_HOST, - CONFIG.SERVING_ENDPOINT_NAME -) - -logger.info("Slack bot initialized successfully") - - -@app.event("message") -async def message_handler(message, say, client): - """ - Handle incoming message events. - This is the main entry point for user messages. - """ - await handle_message_event( - message=message, - say=say, - client=client, - state_manager=state_manager, - auth_handler=auth_handler, - databricks_client=databricks_client, - logger=logger - ) - - -@app.event("app_mention") -async def handle_app_mention(event, say, client): - """ - Handle @mentions of the bot. - Treat as regular message. - """ - await handle_message_event( - message=event, - say=say, - client=client, - state_manager=state_manager, - auth_handler=auth_handler, - databricks_client=databricks_client, - logger=logger - ) - - -@app.command("/databricks-logout") -async def handle_logout_command(ack, command, respond): - """Handle /databricks-logout slash command""" - await ack() - user_id = command["user_id"] - auth_handler.handle_logout(user_id) - await respond("You have been logged out from Databricks.") - - -@app.command("/databricks-clear") -async def handle_clear_command(ack, command, respond): - """Handle /databricks-clear slash command to clear conversation history""" - await ack() - channel_id = command["channel_id"] - # For commands, we don't have a specific thread, so clear all history - # In production, you might want to handle this differently - await respond("Conversation history management via slash command is not fully implemented. Use 'clear' in a message instead.") - - -@app.event("app_home_opened") -async def handle_app_home_opened(event, client): - """ - Handle app home opened event. - Show welcome message and authentication status. - """ - user_id = event["user"] - is_authenticated = auth_handler.is_authenticated(user_id) - - blocks = [ - { - "type": "header", - "text": { - "type": "plain_text", - "text": "Welcome to Databricks Agent Bot" - } - }, - { - "type": "section", - "text": { - "type": "mrkdwn", - "text": "*What I can do:*\n• Answer questions using Databricks agents\n• Execute tool calls and show results\n• Maintain conversation context in threads" - } - }, - { - "type": "divider" - } - ] - - if is_authenticated: - blocks.append({ - "type": "section", - "text": { - "type": "mrkdwn", - "text": "āœ… *Status:* Authenticated\n\nYou can start chatting with me in any channel where I'm added!" - } - }) - else: - auth_message = auth_handler.create_auth_message(user_id, "app_home") - blocks.extend(auth_message["blocks"]) - - try: - client.views_publish( - user_id=user_id, - view={ - "type": "home", - "blocks": blocks - } - ) - except Exception as e: - logger.error(f"Error publishing home view: {e}") - - -# Custom route for OAuth callback -flask_app = Flask(__name__) - -@flask_app.route("/oauth/callback", methods=["GET"]) -async def oauth_callback(): - """ - Handle OAuth callback from Databricks. - Exchanges authorization code for access token. - """ - try: - # Get authorization code and state from query parameters - code = request.args.get("code") - state = request.args.get("state") - error = request.args.get("error") +# Auto-detect mode based on SLACK_APP_TOKEN +slack_app_token = os.environ.get("SLACK_APP_TOKEN", "") - if error: - logger.error(f"OAuth error: {error}") - return f""" - - -

āŒ Authentication Failed

-

Error: {error}

-

Please close this window and try again in Slack.

- - - """, 400 - - if not code or not state: - logger.error("Missing code or state in OAuth callback") - return """ - - -

āŒ Invalid Request

-

Missing required parameters.

-

Please close this window and try again in Slack.

- - - """, 400 - - # Parse state to get user_id - try: - user_id, channel_id = state.split(":") - except ValueError: - logger.error(f"Invalid state format: {state}") - return """ - - -

āŒ Invalid State

-

Please close this window and try again in Slack.

- - - """, 400 - - # Validate state - if not auth_handler.validate_state(state, user_id): - logger.error(f"State validation failed for user {user_id}") - return """ - - -

āŒ Security Validation Failed

-

Please close this window and try again in Slack.

- - - """, 400 - - # Exchange code for token - access_token, expires_in = await auth_handler.exchange_code_for_token(code, user_id) - - if not access_token: - logger.error(f"Failed to exchange code for token for user {user_id}") - return """ - - -

āŒ Token Exchange Failed

-

Unable to complete authentication.

-

Please close this window and try again in Slack.

- - - """, 500 - - # Store token - state_manager.store_user_token(user_id, access_token, expires_in) - - logger.info(f"Successfully authenticated user {user_id}") - - # Send success message to Slack - try: - from slack_sdk import WebClient - slack_client = WebClient(token=CONFIG.SLACK_BOT_TOKEN) - slack_client.chat_postMessage( - channel=channel_id, - text=f"āœ… <@{user_id}> You're now authenticated! Send me a message to get started.", - ) - except Exception as e: - logger.error(f"Failed to send success message to Slack: {e}") - - # Return success page - return """ - - -

āœ… Authentication Successful!

-

You're now connected to Databricks.

-

You can close this window and return to Slack.

- - - - """, 200 - - except Exception as e: - logger.error(f"Error in OAuth callback: {e}") - import traceback - traceback.print_exc() - return """ - - -

āŒ Error

-

An unexpected error occurred.

-

Please close this window and try again in Slack.

- - - """, 500 - - -@app.error -async def custom_error_handler(error, body, logger): - """Global error handler""" - logger.error(f"Error: {error}") - logger.error(f"Request body: {body}") - - -def main(): - """Main entry point""" - try: - # Use Socket Mode for local development (recommended) - # Socket Mode uses WebSocket connection, no public URL needed - slack_app_token = getattr(CONFIG, 'SLACK_APP_TOKEN', None) - - if slack_app_token: - logger.info("Starting bot in Socket Mode (WebSocket)...") - logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") - - # Start Flask server in background thread for OAuth callbacks - import threading - flask_thread = threading.Thread( - target=lambda: flask_app.run(port=CONFIG.PORT, debug=False, use_reloader=False), - daemon=True - ) - flask_thread.start() - logger.info(f"OAuth callback server running on port {CONFIG.PORT}") - - # Start Socket Mode handler (blocking) - handler = SocketModeHandler(app, slack_app_token) - logger.info("Socket Mode handler starting...") - handler.start() - - else: - # Fallback to HTTP mode if no app token provided - logger.info("No SLACK_APP_TOKEN found, starting in HTTP mode...") - logger.info(f"HTTP mode requires public URL for Slack events") - logger.info(f"OAuth callback available at: http://localhost:{CONFIG.PORT}/oauth/callback") - - from slack_bolt.adapter.flask import SlackRequestHandler - handler = SlackRequestHandler(app) - - # Register Slack events endpoint - @flask_app.route("/slack/events", methods=["POST"]) - def slack_events(): - return handler.handle(request) - - # Start Flask server - flask_app.run(port=CONFIG.PORT, debug=False) - - except KeyboardInterrupt: - logger.info("Bot stopped by user") - except Exception as e: - logger.error(f"Error starting bot: {e}") - import traceback - traceback.print_exc() - sys.exit(1) - - -if __name__ == "__main__": +if slack_app_token: + print("šŸš€ Socket Mode detected (SLACK_APP_TOKEN found)") + print(" Starting Socket Mode bot...") + print() + from app_socket import main + main() +else: + print("āš ļø HTTP Mode detected (no SLACK_APP_TOKEN)") + print(" HTTP Mode requires public HTTPS URL") + print(" For local development, set SLACK_APP_TOKEN and use Socket Mode") + print() + response = input("Continue with HTTP mode? (y/N): ") + if response.lower() != 'y': + print("Exiting. Set SLACK_APP_TOKEN in .env.local for Socket Mode") + sys.exit(0) + + from http_mode import main main() diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index ea82f97..7507226 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -1,11 +1,12 @@ """Message event handler for Slack bot""" +import asyncio import logging from typing import Dict, List from slack_sdk import WebClient -async def handle_message_event( +def handle_message_event( message: Dict, say, client: WebClient, @@ -33,11 +34,11 @@ async def handle_message_event( # Handle special commands if text.lower() == "logout": - await handle_logout(user_id, channel_id, thread_ts, say, auth_handler) + handle_logout(user_id, channel_id, thread_ts, say, auth_handler) return if text.lower() == "clear": - await handle_clear_history(thread_ts, channel_id, user_id, say, state_manager) + handle_clear_history(thread_ts, channel_id, user_id, say, state_manager) return # Check authentication @@ -62,10 +63,10 @@ async def handle_message_event( # Call Databricks agent logger.info(f"Calling Databricks agent for user {user_id}") - response = await databricks_client.call_model_endpoint(text, user_token, history) + response = asyncio.run(databricks_client.call_model_endpoint(text, user_token, history)) # Process and send responses - await send_agent_responses( + send_agent_responses( response=response, text=text, channel_id=channel_id, @@ -92,7 +93,7 @@ async def handle_message_event( pass -async def send_agent_responses( +def send_agent_responses( response: List[Dict], text: str, channel_id: str, @@ -172,7 +173,7 @@ def create_tool_call_block(tool_call_info: Dict) -> List[Dict]: ] -async def handle_logout(user_id: str, channel_id: str, thread_ts: str, say, auth_handler): +def handle_logout(user_id: str, channel_id: str, thread_ts: str, say, auth_handler): """Handle logout command""" auth_handler.handle_logout(user_id) say( @@ -181,7 +182,7 @@ async def handle_logout(user_id: str, channel_id: str, thread_ts: str, say, auth ) -async def handle_clear_history(thread_ts: str, channel_id: str, user_id: str, say, state_manager): +def handle_clear_history(thread_ts: str, channel_id: str, user_id: str, say, state_manager): """Handle clear history command""" state_manager.clear_conversation_history(thread_ts) say( From 4742f755bbc056523d80231a215528f7c782f6e6 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Fri, 24 Oct 2025 11:43:31 -0700 Subject: [PATCH 12/21] Basic POC works, just has some broken response parsing Signed-off-by: Sid Murching --- slack-bot/client/databricks_client.py | 57 +++++++++++++++++++-------- slack-bot/handlers/auth_handler.py | 9 +++++ slack-bot/handlers/message_handler.py | 18 ++++++--- 3 files changed, 62 insertions(+), 22 deletions(-) diff --git a/slack-bot/client/databricks_client.py b/slack-bot/client/databricks_client.py index 8bf55d2..494c249 100644 --- a/slack-bot/client/databricks_client.py +++ b/slack-bot/client/databricks_client.py @@ -2,6 +2,7 @@ # Licensed under the MIT License. import logging +import os import uuid import httpx @@ -9,26 +10,26 @@ from databricks_ai_bridge.genie import Genie class DatabricksClient: - def __init__(self, databricks_host: str, request_timeout: float = 300): + def __init__(self, databricks_host: str, serving_endpoint_name: str, request_timeout: float = 300): self.databricks_host = databricks_host + self.serving_endpoint_name = serving_endpoint_name self.client = httpx.AsyncClient( timeout=httpx.Timeout(request_timeout), ) async def exchange_token(self, provider_oauth_token: str): - - url = f"{self.databricks_host}/oidc/v1/token" - - data = { - "subject_token": provider_oauth_token, # replace with your JWT token variable - "subject_token_type": "urn:ietf:params:oauth:token-type:jwt", - "grant_type": "urn:ietf:params:oauth:grant-type:token-exchange", - "scope": "all-apis" - } - - response = await self.client.post(url, data=data) - - return response.json()['access_token'] + """ + Exchange provider OAuth token for Databricks token. + For direct Databricks OAuth, the token is already valid - no exchange needed. + """ + # If this is already a Databricks OAuth token (starts with dapi or similar), + # no exchange is needed + logging.info("Token exchange requested") + logging.info(f" Token prefix: {provider_oauth_token[:20]}...") + + # For Databricks OAuth, the token is already a Databricks token + # Skip the exchange and return the token directly + return provider_oauth_token def _throw_unexpected_endpoint_format(self): raise Exception("This app can only run against ChatModel, ChatAgent, or ResponsesAgent endpoints") @@ -166,7 +167,19 @@ async def call_model_endpoint(self, oauth_db_token = await self.exchange_token(provider_oauth_token) - workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + # Create workspace client with token only (don't use OAuth credentials) + # Temporarily unset OAuth env vars to prevent SDK from reading them + old_client_id = os.environ.pop('DATABRICKS_CLIENT_ID', None) + old_client_secret = os.environ.pop('DATABRICKS_CLIENT_SECRET', None) + + try: + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + finally: + # Restore env vars + if old_client_id: + os.environ['DATABRICKS_CLIENT_ID'] = old_client_id + if old_client_secret: + os.environ['DATABRICKS_CLIENT_SECRET'] = old_client_secret task_type = self._get_endpoint_task_type(workspace_client, serving_endpoint_name) @@ -190,7 +203,19 @@ async def call_genie_space(self, question: str, oauth_db_token = await self.exchange_token(provider_oauth_token) - workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + # Create workspace client with token only (don't use OAuth credentials) + # Temporarily unset OAuth env vars to prevent SDK from reading them + old_client_id = os.environ.pop('DATABRICKS_CLIENT_ID', None) + old_client_secret = os.environ.pop('DATABRICKS_CLIENT_SECRET', None) + + try: + workspace_client = WorkspaceClient(host=self.databricks_host, token=oauth_db_token) + finally: + # Restore env vars + if old_client_id: + os.environ['DATABRICKS_CLIENT_ID'] = old_client_id + if old_client_secret: + os.environ['DATABRICKS_CLIENT_SECRET'] = old_client_secret genie = Genie(genie_space_id, workspace_client) diff --git a/slack-bot/handlers/auth_handler.py b/slack-bot/handlers/auth_handler.py index 4ce6f23..6b4f097 100644 --- a/slack-bot/handlers/auth_handler.py +++ b/slack-bot/handlers/auth_handler.py @@ -44,6 +44,11 @@ def get_auth_url(self, user_id: str, channel_id: str) -> str: self.state_manager.set_user_data(user_id, "oauth_state", state) self.state_manager.set_user_data(user_id, "code_verifier", code_verifier) + logging.info(f"Generated PKCE for user {user_id}") + logging.info(f" Code verifier: {code_verifier[:20]}...") + logging.info(f" Code challenge: {code_challenge[:20]}...") + logging.info(f" Stored verifier: {self.state_manager.get_user_data(user_id, 'code_verifier')[:20]}...") + # Build OAuth URL with PKCE # Use databricks-cli as client_id if no custom client_id is configured client_id = self.config.DATABRICKS_CLIENT_ID or "databricks-cli" @@ -124,6 +129,10 @@ async def exchange_code_for_token(self, code: str, user_id: str) -> Tuple[Option logging.error(f"Code verifier not found for user {user_id}") return None, None + logging.info(f"Exchanging code for token for user {user_id}") + logging.info(f" Retrieved code verifier: {code_verifier[:20]}...") + logging.info(f" Authorization code: {code[:20]}...") + # Use databricks-cli as client_id if no custom client_id is configured client_id = self.config.DATABRICKS_CLIENT_ID or "databricks-cli" diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index 7507226..aa1fc57 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -63,7 +63,12 @@ def handle_message_event( # Call Databricks agent logger.info(f"Calling Databricks agent for user {user_id}") - response = asyncio.run(databricks_client.call_model_endpoint(text, user_token, history)) + response = asyncio.run(databricks_client.call_model_endpoint( + databricks_client.serving_endpoint_name, + text, + user_token, + history + )) # Process and send responses send_agent_responses( @@ -110,19 +115,20 @@ def send_agent_responses( tool_calls = {} for item in response: - if item["type"] == "message": + # Handle response format: [{'role': 'assistant', 'content': '...'}] + if item.get("role") == "assistant" and "content" in item: # Send message in thread (visible to everyone) say( - text=item["text"], + text=item["content"], thread_ts=thread_ts ) - state_manager.add_message_to_history(thread_ts, "assistant", item["text"]) + state_manager.add_message_to_history(thread_ts, "assistant", item["content"]) - elif item["type"] == "tool_call": + elif item.get("type") == "tool_call": # Store tool call for pairing with result tool_calls[item["call_id"]] = item - elif item["type"] == "tool_result": + elif item.get("type") == "tool_result": # Pair with tool call and send as block call_id = item["call_id"] if call_id in tool_calls: From 3914e20594aeda33419b6edce1de85357630c070 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Tue, 28 Oct 2025 20:23:43 -0700 Subject: [PATCH 13/21] Implement ephemeral responses with share button and auto-respond after OAuth MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Major features: - Ephemeral messages: Bot responses appear privately to user with share button - Auto-respond after OAuth: Bot stores pending messages and automatically processes them after authentication - Share to channel: Users can click button to post response publicly in thread - Socket mode: Refactored to use Socket Mode for local development Technical changes: - Added oauth_server.py: Standalone OAuth callback handler with auto-respond logic - Updated app_socket.py: Socket Mode implementation with share button handler - Enhanced message_handler.py: Send ephemeral messages, store pending messages before OAuth - Fixed PKCE issues: Deduplicated event handlers to prevent double processing - Fixed token exchange: Skip unnecessary exchange for direct Databricks OAuth tokens - Fixed WorkspaceClient conflicts: Temporarily unset OAuth env vars šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/DEVLOOP.md | 341 ++++++++++++++++++++++ slack-bot/QUICK_FIX.md | 83 ++++++ slack-bot/README_ARCHITECTURE.md | 357 +++++++++++++++++++++++ slack-bot/TROUBLESHOOTING.md | 229 +++++++++++++++ slack-bot/app_socket.py | 292 ++++++++++++++++++ slack-bot/dev_tools/__init__.py | 1 + slack-bot/dev_tools/devloop.sh | 193 ++++++++++++ slack-bot/dev_tools/send_test_message.py | 138 +++++++++ slack-bot/dev_tools/test_bot_flow.py | 230 +++++++++++++++ slack-bot/dev_tools/watch_logs.sh | 53 ++++ slack-bot/handlers/message_handler.py | 88 +++++- slack-bot/http_mode.py | 161 ++++++++++ slack-bot/oauth_server.py | 142 +++++++++ slack-bot/test_connection.py | 68 +++++ 14 files changed, 2368 insertions(+), 8 deletions(-) create mode 100644 slack-bot/DEVLOOP.md create mode 100644 slack-bot/QUICK_FIX.md create mode 100644 slack-bot/README_ARCHITECTURE.md create mode 100644 slack-bot/TROUBLESHOOTING.md create mode 100644 slack-bot/app_socket.py create mode 100644 slack-bot/dev_tools/__init__.py create mode 100755 slack-bot/dev_tools/devloop.sh create mode 100644 slack-bot/dev_tools/send_test_message.py create mode 100644 slack-bot/dev_tools/test_bot_flow.py create mode 100755 slack-bot/dev_tools/watch_logs.sh create mode 100644 slack-bot/http_mode.py create mode 100644 slack-bot/oauth_server.py create mode 100644 slack-bot/test_connection.py diff --git a/slack-bot/DEVLOOP.md b/slack-bot/DEVLOOP.md new file mode 100644 index 0000000..375849f --- /dev/null +++ b/slack-bot/DEVLOOP.md @@ -0,0 +1,341 @@ +# Development Loop Guide + +This guide shows how to set up an automated development loop where you can send test messages to Slack, observe bot behavior, and iterate quickly. + +## Tools Created + +``` +dev_tools/ +ā”œā”€ā”€ send_test_message.py # Send messages to Slack via API +ā”œā”€ā”€ test_bot_flow.py # Automated end-to-end tests +ā”œā”€ā”€ devloop.sh # Interactive dev loop script +└── watch_logs.sh # Colorized log watching +``` + +## Quick Start + +### 1. Set Test Channel + +Add to your `.env.local`: +```bash +SLACK_TEST_CHANNEL=test-chatbot-app +``` + +### 2. Run Interactive Dev Loop + +```bash +./dev_tools/devloop.sh +``` + +This gives you a menu: +``` +Choose an action: + 1) Start bot + 2) Stop bot + 3) Restart bot + 4) Send test message + 5) Run automated tests + 6) Watch logs + 7) View recent logs + q) Quit +``` + +### 3. Or Run Automated Tests + +```bash +./dev_tools/devloop.sh auto +``` + +This will: +1. Start the bot +2. Run automated tests +3. Stop the bot +4. Show results + +## Using Individual Tools + +### Send Test Messages + +```bash +# Simple message (mentions bot automatically) +python dev_tools/send_test_message.py --mention "Hello bot!" + +# Message to specific channel +python dev_tools/send_test_message.py --mention "Test" --channel test-channel + +# Send as DM +python dev_tools/send_test_message.py --dm "Private message" + +# Plain message (no mention) +python dev_tools/send_test_message.py "Just a message" +``` + +### Run Automated Tests + +```bash +# Run all tests +python dev_tools/test_bot_flow.py + +# Verbose mode +python dev_tools/test_bot_flow.py --verbose + +# Specific channel +python dev_tools/test_bot_flow.py --channel my-test-channel +``` + +Tests include: +- āœ… Simple message response +- āœ… Authentication flow +- āœ… Command execution + +### Watch Logs + +```bash +# Colorized log watching +./dev_tools/watch_logs.sh + +# Or plain tail +tail -f bot.log +``` + +## Complete Dev Loop Example + +### Terminal 1: Start bot and watch logs + +```bash +# Start bot +python app_socket.py > bot.log 2>&1 & + +# Watch logs +tail -f bot.log +``` + +### Terminal 2: Send test messages + +```bash +# Test 1: Simple message +python dev_tools/send_test_message.py --mention "Hello!" + +# Test 2: Check authentication +python dev_tools/send_test_message.py --mention "What can you do?" + +# Test 3: Databricks query (if authenticated) +python dev_tools/send_test_message.py --mention "Tell me about Q4 sales" +``` + +### What to observe in logs: + +``` +āœ… Message sent to #test-chatbot-app +šŸ“Ø Message from user U09N9G6PQ69 +šŸ“¢ Bot mentioned by user U09N9G6PQ69 +INFO - Processing message from user... +``` + +## Automated Testing Workflow + +For testing code changes: + +```bash +# 1. Make code changes to handlers/message_handler.py + +# 2. Run automated tests +./dev_tools/devloop.sh auto + +# 3. Check results +🧪 SLACK BOT END-TO-END TESTS +āœ… PASS - Simple Message +āœ… PASS - Authentication Flow +āœ… PASS - Logout Command +Results: 3/3 tests passed +šŸŽ‰ All tests passed! + +# 4. If tests fail, check logs and iterate +tail -50 bot.log +``` + +## Integration with Claude Code + +You (Claude) can now: + +1. **Send test messages**: + ```bash + python dev_tools/send_test_message.py --mention "test query" + ``` + +2. **Check logs**: + ```bash + tail -20 bot.log | grep "Message from" + ``` + +3. **Run tests**: + ```bash + python dev_tools/test_bot_flow.py --verbose + ``` + +4. **Iterate on failures**: + - Read error from logs + - Edit code + - Restart bot + - Re-test + +## Example: Full Debug Cycle + +```bash +# 1. Start bot with logging +python app_socket.py 2>&1 | tee bot.log & + +# 2. Send test message +python dev_tools/send_test_message.py --mention "Hello bot" + +# 3. Check if bot responded +tail -10 bot.log | grep -E "(Message from|Bot mentioned)" + +# 4. If no response, check for errors +tail -20 bot.log | grep -E "(ERROR|Exception)" + +# 5. Fix issue in code + +# 6. Restart bot +pkill -f "python app_socket.py" +python app_socket.py 2>&1 | tee bot.log & + +# 7. Re-test +python dev_tools/send_test_message.py --mention "Hello bot" +``` + +## Advanced: Continuous Testing + +Create a test loop: + +```bash +#!/bin/bash +# test_loop.sh + +while true; do + echo "Running tests..." + python dev_tools/test_bot_flow.py + + if [ $? -eq 0 ]; then + echo "āœ… Tests passed" + break + else + echo "āŒ Tests failed. Retrying in 5s..." + sleep 5 + fi +done +``` + +## Environment Variables + +Add these to `.env.local`: + +```bash +# Test channel (where test messages will be sent) +SLACK_TEST_CHANNEL=test-chatbot-app + +# Optional: Specific user ID for testing +SLACK_TEST_USER=U09N9G6PQ69 +``` + +## Tips + +### 1. Use Test Channel + +Create a dedicated test channel in Slack: +``` +/channel create test-chatbot-app +/invite @YourBot +``` + +### 2. Filter Logs + +```bash +# Only show message events +tail -f bot.log | grep "šŸ“Ø\|šŸ“¢" + +# Only show errors +tail -f bot.log | grep "ERROR\|āŒ" + +# Only show Databricks API calls +tail -f bot.log | grep "Databricks" +``` + +### 3. Quick Restart + +```bash +# Restart bot quickly +pkill -f "python app" && sleep 1 && python app_socket.py > bot.log 2>&1 & +``` + +### 4. Test Specific Scenarios + +Create test scripts for common scenarios: + +```python +# test_scenarios/test_auth_flow.py +import sys +sys.path.append('..') +from dev_tools.send_test_message import send_mention + +# Test auth required +send_mention("What is 2+2?") + +# User would authenticate here... + +# Test after auth +send_mention("Now what is 2+2?") +``` + +## Troubleshooting + +### Bot not responding to test messages + +Check: +```bash +# 1. Bot is running +ps aux | grep "python app" + +# 2. Bot connected to Slack +tail -5 bot.log | grep "session has been established" + +# 3. Bot received the message +tail -10 bot.log | grep "Message from" +``` + +### Test messages not appearing + +Check: +```bash +# 1. Test channel exists +python -c "from slack_sdk import WebClient; import os; print(WebClient(token=os.getenv('SLACK_BOT_TOKEN')).conversations_list())" + +# 2. Bot token is valid +python test_connection.py +``` + +### Tests timeout + +Increase timeout in `test_bot_flow.py`: +```python +def wait_for_response(self, after_ts, timeout=30): # Increase to 30s +``` + +## Next Steps + +1. āœ… Run `./dev_tools/devloop.sh` to get started +2. āœ… Send test messages with `send_test_message.py` +3. āœ… Run automated tests with `test_bot_flow.py` +4. āœ… Watch logs with `tail -f bot.log` +5. āœ… Iterate quickly on code changes! + +## For Claude Code Users + +This setup allows me (Claude) to: +- āœ… Send test messages via Bash commands +- āœ… Read bot logs to see responses +- āœ… Run automated tests +- āœ… Iterate on failures +- āœ… Complete full dev loop without manual Slack interaction + +Perfect for rapid debugging and development! šŸš€ diff --git a/slack-bot/QUICK_FIX.md b/slack-bot/QUICK_FIX.md new file mode 100644 index 0000000..f7f6f93 --- /dev/null +++ b/slack-bot/QUICK_FIX.md @@ -0,0 +1,83 @@ +# Quick Fix: Bot Not Receiving Messages + +## Problem + +The bot is in OAuth installation mode because `SLACK_CLIENT_ID` and `SLACK_CLIENT_SECRET` are set in `.env.local`. + +This causes the warning: +``` +As `installation_store` or `authorize` has been used, `token` will be ignored. +``` + +## Solution + +**Remove or comment out** `SLACK_CLIENT_ID` and `SLACK_CLIENT_SECRET` from your `.env.local`: + +```bash +# These are only needed for multi-workspace Slack OAuth (not needed for single bot) +# SLACK_CLIENT_ID=... +# SLACK_CLIENT_SECRET=... +``` + +Keep these (required): +```bash +SLACK_BOT_TOKEN=xoxb-... +SLACK_APP_TOKEN=xapp-... +SLACK_SIGNING_SECRET=... +DATABRICKS_HOST=... +SERVING_ENDPOINT_NAME=... +``` + +## Quick Test + +1. Edit `.env.local` and comment out SLACK_CLIENT_ID and SLACK_CLIENT_SECRET +2. Restart bot: + ```bash + python app.py + ``` + +3. You should see: + ``` + āœ… NO warning about "installation_store" + āœ… INFO - Slack bot initialized successfully + āœ… INFO - A new session has been established + ``` + +4. Tag the bot in Slack: + ``` + @BotName hello + ``` + +5. You should see logs: + ``` + INFO - šŸ“¢ Bot mentioned by user U123... + INFO - Processing message... + ``` + +## Why This Happens + +- **Slack OAuth** (SLACK_CLIENT_ID/SECRET) = Install bot to multiple workspaces +- **Single Bot Token** (SLACK_BOT_TOKEN) = Bot only in one workspace + +For local development, you just need the bot token, not OAuth credentials. + +The `SLACK_CLIENT_ID` and `SLACK_CLIENT_SECRET` confused Slack Bolt into thinking you want multi-workspace OAuth, which requires a different setup. + +## Verification + +After removing those vars and restarting, check the logs - you should NOT see: +``` +āŒ As `installation_store` or `authorize` has been used... +``` + +You SHOULD see: +``` +āœ… Slack bot initialized successfully +āœ… A new session has been established +``` + +Then tag your bot in Slack and you'll see: +``` +āœ… šŸ“¢ Bot mentioned by user... +āœ… Processing message... +``` diff --git a/slack-bot/README_ARCHITECTURE.md b/slack-bot/README_ARCHITECTURE.md new file mode 100644 index 0000000..95eb383 --- /dev/null +++ b/slack-bot/README_ARCHITECTURE.md @@ -0,0 +1,357 @@ +# Architecture - Simplified Socket Mode Design + +## File Structure + +``` +slack-bot/ +ā”œā”€ā”€ app.py # šŸš€ Launcher - auto-detects mode +ā”œā”€ā”€ app_socket.py # ⚔ Socket Mode (recommended for dev) +ā”œā”€ā”€ http_mode.py # 🌐 HTTP/Events API mode (production) +ā”œā”€ā”€ oauth_server.py # šŸ” OAuth callback handler (shared) +│ +ā”œā”€ā”€ config.py # Configuration +ā”œā”€ā”€ handlers/ # Event handlers +│ ā”œā”€ā”€ message_handler.py # Message processing +│ └── auth_handler.py # Databricks OAuth +ā”œā”€ā”€ storage/ # State management +│ └── state_manager.py # In-memory state +└── client/ # External integrations + └── databricks_client.py # Databricks API +``` + +## Modes + +### Socket Mode (Default - `app_socket.py`) + +**Best for:** Local development, simple deployment + +**How it works:** +``` +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” WebSocket ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ Bot │◄──────────────────────────►│ Slack │ +│ (localhost) │ Events via WebSocket │ │ +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”¬ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ + │ + │ Background Thread + ā–¼ +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ OAuth │ http://localhost:3000/oauth/callback +│ Server │ (Flask in background) +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ +``` + +**Advantages:** +- āœ… No public URL needed +- āœ… Easy to debug locally +- āœ… Simple setup +- āœ… Works behind firewall + +**Run:** +```bash +python app.py # Auto-detect +python app_socket.py # Force Socket Mode +``` + +### HTTP Mode (`http_mode.py`) + +**Best for:** Production deployments + +**How it works:** +``` +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” HTTPS POST ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ Bot │◄──────────────────────────►│ Slack │ +│ (public) │ /slack/events │ │ +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”¬ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ + │ + │ Same Process + ā–¼ +ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” +│ OAuth │ https://yourbot.com/oauth/callback +│ Server │ (Flask on same port) +ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ +``` + +**Advantages:** +- āœ… Production-ready +- āœ… More scalable +- āœ… Standard architecture + +**Requirements:** +- āŒ Public HTTPS URL +- āŒ Event Subscriptions configured +- āŒ ngrok/tunnel for local testing + +**Run:** +```bash +python http_mode.py +``` + +## Components + +### 1. **OAuth Server** (`oauth_server.py`) + +Handles Databricks OAuth callbacks: +- Receives authorization code from Databricks +- Exchanges code for access token (with PKCE) +- Stores token in state manager +- Sends success message to Slack + +**Shared by both modes** - runs in background thread (Socket) or same process (HTTP). + +### 2. **Event Handlers** (`handlers/`) + +#### Message Handler (`message_handler.py`) +- Processes user messages +- Checks authentication +- Calls Databricks agent +- Formats and sends responses +- Handles special commands (logout, clear) + +#### Auth Handler (`auth_handler.py`) +- Generates OAuth URLs with PKCE +- Validates OAuth state (CSRF protection) +- Exchanges authorization codes for tokens +- Manages user authentication state + +### 3. **State Manager** (`storage/state_manager.py`) + +In-memory storage for: +- User OAuth tokens +- Conversation history (per thread) +- User session data + +**Note:** For production, replace with Redis/DynamoDB. + +### 4. **Databricks Client** (`client/databricks_client.py`) + +- Exchanges user tokens for Databricks tokens +- Calls serving endpoints +- Parses agent responses (messages, tool calls) + +## Event Flow + +### User Sends Message + +``` +1. User types in Slack: "@BotName hello" + +2. Socket Mode: + Slack → WebSocket → app_socket.py → message_handler() + + HTTP Mode: + Slack → HTTPS POST → http_mode.py → message_handler() + +3. message_handler.py: + - Check if user authenticated + - If not: Send ephemeral auth prompt + - If yes: Call Databricks agent + +4. databricks_client.py: + - Exchange user token for Databricks token + - Call serving endpoint + - Parse response + +5. message_handler.py: + - Format response + - Send back to Slack in thread +``` + +### OAuth Flow + +``` +1. User clicks "Sign in to Databricks" + +2. auth_handler.py: + - Generate OAuth URL with PKCE + - Store code_verifier in state + +3. User authenticates in browser + +4. Databricks redirects to: + http://localhost:3000/oauth/callback?code=...&state=... + +5. oauth_server.py: + - Validate state (CSRF check) + - Exchange code + code_verifier for token + - Store token in state_manager + - Send success message to Slack + +6. User returns to Slack, sends message + +7. message_handler.py: + - User now authenticated + - Calls Databricks agent +``` + +## Key Design Decisions + +### Why Separate Socket and HTTP? + +**Before:** +- Single `app.py` with complex branching +- Hard to understand which mode is active +- HTTP code mixed with Socket code + +**After:** +- Clear separation: `app_socket.py` vs `http_mode.py` +- Each file is minimal and focused +- Easy to understand what's happening + +### Why OAuth Server is Shared? + +OAuth callback logic is **identical** for both modes: +- Same URL format +- Same token exchange +- Same security validation + +Only difference: HTTP mode uses Flask directly, Socket mode runs Flask in thread. + +### Why In-Memory State? + +For simplicity during development. In production: + +```python +# storage/redis_state_manager.py +class RedisStateManager: + def __init__(self, redis_url): + self.redis = redis.from_url(redis_url) + + def store_user_token(self, user_id, token, expires_in): + self.redis.setex(f"token:{user_id}", expires_in, token) +``` + +Then swap in `config.py`: +```python +if CONFIG.STATE_STORAGE == "redis": + state_manager = RedisStateManager(CONFIG.REDIS_URL) +else: + state_manager = StateManager() # In-memory +``` + +## Configuration + +### Required Environment Variables + +**Socket Mode:** +```bash +SLACK_BOT_TOKEN=xoxb-... +SLACK_APP_TOKEN=xapp-... # Required for Socket Mode +DATABRICKS_HOST=https://... +SERVING_ENDPOINT_NAME=... +``` + +**HTTP Mode:** +```bash +SLACK_BOT_TOKEN=xoxb-... +SLACK_SIGNING_SECRET=... # Required for HTTP +DATABRICKS_HOST=https://... +SERVING_ENDPOINT_NAME=... +``` + +### Optional Configuration + +```bash +PORT=3000 # OAuth callback port +DATABRICKS_CLIENT_ID=... # For custom OAuth app +DATABRICKS_CLIENT_SECRET=... # For custom OAuth app +STATE_STORAGE=memory # Or redis, dynamodb +REDIS_URL=redis://... # If using Redis +``` + +## Testing + +### Test Socket Mode + +```bash +# Start bot +python app_socket.py + +# In Slack: +/invite @BotName +@BotName hello + +# Check logs: +āœ… Socket Mode handler starting... +āœ… A new session has been established +šŸ“Ø Message from user U123... +``` + +### Test HTTP Mode + +```bash +# Start bot +python http_mode.py + +# Expose with ngrok: +ngrok http 3000 + +# Configure in Slack app: +Event Subscriptions URL: https://your-ngrok.ngrok.io/slack/events + +# In Slack: +@BotName hello +``` + +## Debugging + +### Enable Debug Logs + +Edit `app_socket.py`: +```python +logging.basicConfig(level=logging.DEBUG) +``` + +### Check What's Running + +```bash +ps aux | grep python +lsof -i :3000 +``` + +### Common Issues + +**Bot not responding:** +- Check `SLACK_APP_TOKEN` is set (Socket Mode) +- Verify bot is invited to channel +- Check event subscriptions configured + +**OAuth fails:** +- Check `http://localhost:3000/oauth/callback` is accessible +- Verify Databricks OAuth app has correct redirect URI +- Check `DATABRICKS_HOST` is correct + +## Production Deployment + +### Checklist + +- [ ] Use HTTP Mode (`http_mode.py`) +- [ ] Set up persistent state storage (Redis/DynamoDB) +- [ ] Use production WSGI server (Gunicorn) +- [ ] Configure HTTPS endpoint +- [ ] Set Event Subscriptions URL in Slack +- [ ] Enable rate limiting +- [ ] Set up monitoring/logging +- [ ] Encrypt tokens at rest +- [ ] Use secret management (AWS Secrets Manager) + +### Example Production Stack + +``` +Internet + │ + ā–¼ +ALB (HTTPS) + │ + ā–¼ +ECS/Fargate (http_mode.py + Gunicorn) + │ + ā”œā”€ā–¶ ElastiCache (Redis) - State storage + └─▶ Databricks API +``` + +## Next Steps + +1. **For development:** Use `app_socket.py` - it just works! +2. **For production:** Adapt `http_mode.py` with proper infrastructure +3. **For multi-tenant:** Add per-workspace token storage +4. **For scale:** Add Redis, message queues, etc. diff --git a/slack-bot/TROUBLESHOOTING.md b/slack-bot/TROUBLESHOOTING.md new file mode 100644 index 0000000..9835ff2 --- /dev/null +++ b/slack-bot/TROUBLESHOOTING.md @@ -0,0 +1,229 @@ +# Troubleshooting Guide + +## Bot Not Responding to Messages + +### 1. Check if bot is running + +```bash +ps aux | grep "python app.py" +``` + +If not running, start it: +```bash +python app.py +``` + +### 2. Check logs when starting + +You should see: +``` +Loaded environment from: .env.local +INFO - Slack bot initialized successfully +INFO - Starting bot in Socket Mode (WebSocket)... +INFO - OAuth callback server running on port 3000 +INFO - Socket Mode handler starting... +INFO - A new session has been established (app_id: A07xxxxx) +``` + +**If you see:** +``` +No SLACK_APP_TOKEN found, starting in HTTP mode... +``` +→ Add `SLACK_APP_TOKEN` to `.env.local` + +### 3. Check Event Subscriptions in Slack + +Go to [api.slack.com/apps](https://api.slack.com/apps) → Your App → **Event Subscriptions** + +Make sure these are enabled: +- āœ… `app_mention` - Bot is @mentioned +- āœ… `message.channels` - Messages in public channels +- āœ… `message.groups` - Messages in private channels +- āœ… `message.im` - Direct messages + +### 4. Check Bot Scopes + +Go to **OAuth & Permissions** → **Scopes** → **Bot Token Scopes** + +Required scopes: +- āœ… `app_mentions:read` +- āœ… `chat:write` +- āœ… `channels:history` +- āœ… `groups:history` +- āœ… `im:history` +- āœ… `users:read` + +If you added new scopes, **reinstall the app** to your workspace. + +### 5. Check Bot is in Channel + +``` +/invite @YourBotName +``` + +The bot must be invited to channels to see messages. + +### 6. Test with Direct Message + +Try sending a DM to the bot. This helps isolate if it's a channel issue vs bot issue. + +### 7. Enable Debug Logging + +Edit `app.py` and change logging level: + +```python +logging.basicConfig( + level=logging.DEBUG, # Changed from INFO + format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', + handlers=[ + logging.StreamHandler(sys.stdout) + ] +) +``` + +Restart and check for detailed logs. + +### 8. Common Issues + +#### "Authentication failed" error + +**Cause**: Missing or invalid Databricks credentials + +**Solution**: Check `.env.local` has: +```bash +DATABRICKS_HOST=https://your-workspace.databricks.com +SERVING_ENDPOINT_NAME=your-endpoint-name +``` + +#### Bot responds but says "Please authenticate" + +**Expected behavior** - User needs to authenticate with Databricks first. + +Click the "Sign in to Databricks" button (only visible to you). + +#### Socket Mode connection issues + +**Symptoms**: Bot starts but no events received + +**Check**: +1. Socket Mode enabled in Slack app settings +2. `SLACK_APP_TOKEN` is correct (starts with `xapp-`) +3. Token has `connections:write` scope + +**Regenerate token** if needed: +1. Go to **Socket Mode** in Slack app +2. Delete old token +3. Create new one with `connections:write` scope +4. Update `.env.local` + +#### "Message not found" or handler not triggered + +**Check bot user ID**: +```python +# Add this to app.py temporarily for debugging +@app.event("message") +async def log_all_messages(message, logger): + logger.info(f"Received message event: {message}") +``` + +This will log ALL messages the bot sees. + +### 9. Test OAuth Flow + +Try the OAuth flow separately: + +1. Open browser to: `http://localhost:3000/oauth/callback?test=1` +2. Should see the Flask server responding (or error page) + +### 10. Verify Configuration + +Create a test script: + +```python +# test_config.py +from config import DefaultConfig + +config = DefaultConfig() + +print(f"SLACK_BOT_TOKEN: {config.SLACK_BOT_TOKEN[:20]}...") +print(f"SLACK_APP_TOKEN: {config.SLACK_APP_TOKEN[:20]}...") +print(f"DATABRICKS_HOST: {config.DATABRICKS_HOST}") +print(f"SERVING_ENDPOINT_NAME: {config.SERVING_ENDPOINT_NAME}") +``` + +Run: +```bash +python test_config.py +``` + +All values should be set (not empty). + +## Quick Diagnostic Checklist + +Run through this checklist: + +- [ ] Bot is running (`python app.py`) +- [ ] Socket Mode enabled in Slack app +- [ ] `SLACK_APP_TOKEN` set in `.env.local` +- [ ] Event subscriptions configured (`app_mention`, `message.channels`) +- [ ] Bot token scopes include `app_mentions:read`, `chat:write` +- [ ] Bot invited to channel (`/invite @BotName`) +- [ ] Logs show "A new session has been established" +- [ ] Databricks credentials configured +- [ ] Port 3000 not blocked by firewall + +## Still Not Working? + +### Check Slack API Status + +Visit [status.slack.com](https://status.slack.com) - Slack might be having issues. + +### Enable Verbose Logging + +Add this to see ALL Slack events: + +```python +# In app.py, before app.event decorators +import os +os.environ["SLACK_BOLT_LOG_LEVEL"] = "DEBUG" +``` + +### Restart Everything + +1. Stop the bot (Ctrl+C) +2. Clear any cached connections: + ```bash + rm -rf .slack_bolt_* + ``` +3. Restart: + ```bash + python app.py + ``` + +### Check Network Issues + +```bash +# Test connectivity to Slack +curl -I https://slack.com + +# Check DNS resolution +nslookup slack.com +``` + +## Getting Help + +When asking for help, provide: + +1. **Bot startup logs** (first 20 lines) +2. **Slack app configuration screenshot** (Event Subscriptions page) +3. **Environment check**: + ```bash + cat .env.local | grep -v SECRET | grep -v TOKEN + ``` +4. **Error messages** from logs when you tag the bot + +## Contact + +- Check Slack SDK docs: https://slack.dev/bolt-python +- Slack API community: https://api.slack.com/community +- File issue with bot logs if it's a code problem diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py new file mode 100644 index 0000000..a3a8818 --- /dev/null +++ b/slack-bot/app_socket.py @@ -0,0 +1,292 @@ +#!/usr/bin/env python3 +""" +Slack bot using Socket Mode (WebSocket) - Simple and minimal. + +This version is optimized for local development: +- No public URL needed +- OAuth callbacks handled in background thread +- Clean Socket Mode connection + +For HTTP/Events API mode, see http_mode.py +""" + +import logging +import re +import sys +import threading +from pathlib import Path +from dotenv import load_dotenv +from slack_bolt import App +from slack_bolt.adapter.socket_mode import SocketModeHandler + +# Load environment variables +env_file = Path(__file__).parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) + print(f"āœ… Loaded environment from: {env_file}") +else: + env_file = Path(__file__).parent / ".env" + if env_file.exists(): + load_dotenv(env_file) + print(f"āœ… Loaded environment from: {env_file}") + else: + print("āš ļø No .env file found, using system environment variables") + +from config import DefaultConfig +from storage import StateManager +from handlers import AuthHandler, handle_message_event +from client import DatabricksClient +from oauth_server import create_oauth_server + +# Configure logging +logging.basicConfig( + level=logging.INFO, + format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', + handlers=[logging.StreamHandler(sys.stdout)] +) +logger = logging.getLogger(__name__) + +# Reduce Slack SDK noise +logging.getLogger("slack_sdk").setLevel(logging.WARNING) +logging.getLogger("werkzeug").setLevel(logging.WARNING) + +# Load configuration +CONFIG = DefaultConfig() + +# Validate required configuration +required_configs = [ + "SLACK_BOT_TOKEN", + "SLACK_APP_TOKEN", # Required for Socket Mode + "DATABRICKS_HOST", + "SERVING_ENDPOINT_NAME" +] +missing_configs = [c for c in required_configs if not getattr(CONFIG, c)] +if missing_configs: + logger.error(f"āŒ Missing required configuration: {', '.join(missing_configs)}") + logger.error("Make sure SLACK_APP_TOKEN is set for Socket Mode") + sys.exit(1) + +# Initialize Slack app (Socket Mode) +app = App( + token=CONFIG.SLACK_BOT_TOKEN, + signing_secret=CONFIG.SLACK_SIGNING_SECRET, +) + +# Initialize components +state_manager = StateManager() +auth_handler = AuthHandler(CONFIG, state_manager) +databricks_client = DatabricksClient( + CONFIG.DATABRICKS_HOST, + CONFIG.SERVING_ENDPOINT_NAME +) + +logger.info("āœ… Slack bot initialized successfully") + + +# Event Handlers + +@app.event("app_mention") +def handle_app_mention(event, say, client): + """Handle @mentions of the bot""" + logger.info(f"šŸ“¢ Bot mentioned by user {event.get('user')}") + + handle_message_event( + message=event, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.event("message") +def message_handler(message, say, client): + """Handle direct messages (non-mention messages in DMs)""" + # Only handle DMs here, app_mention handles channel mentions + if message.get('channel_type') == 'im': + logger.info(f"šŸ“Ø DM from user {message.get('user')}") + handle_message_event( + message=message, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.action(re.compile(r"^share_response_.*")) +def handle_share_response(ack, body, client): + """Handle share to channel button click""" + ack() + + user_id = body["user"]["id"] + action = body["actions"][0] + response_id = action["value"] + + # Retrieve the stored response + response_data = state_manager.get_user_data(user_id, f"pending_share_{response_id}") + + if response_data: + # Post the message publicly in the thread + client.chat_postMessage( + channel=response_data["channel_id"], + thread_ts=response_data["thread_ts"], + text=f"<@{user_id}> shared:\n\n{response_data['content']}" + ) + + # Update the ephemeral message to show it was shared + client.chat_update( + channel=body["channel"]["id"], + ts=body["message"]["ts"], + text=response_data["content"], + blocks=[ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": response_data["content"] + } + }, + { + "type": "context", + "elements": [ + { + "type": "mrkdwn", + "text": "āœ… _Shared to channel_" + } + ] + } + ] + ) + + # Clean up stored data + state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) + + logger.info(f"User {user_id} shared response {response_id} to channel") + + +@app.command("/databricks-logout") +def handle_logout_command(ack, command, respond): + """Handle /databricks-logout slash command""" + ack() + user_id = command["user_id"] + auth_handler.handle_logout(user_id) + respond("You have been logged out from Databricks.") + + +@app.event("app_home_opened") +def handle_app_home_opened(event, client): + """Handle app home opened event""" + user_id = event["user"] + is_authenticated = auth_handler.is_authenticated(user_id) + + blocks = [ + { + "type": "header", + "text": {"type": "plain_text", "text": "Welcome to Databricks Agent Bot"} + }, + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": "*What I can do:*\n• Answer questions using Databricks agents\n• Execute tool calls and show results\n• Maintain conversation context in threads" + } + }, + {"type": "divider"} + ] + + if is_authenticated: + blocks.append({ + "type": "section", + "text": { + "type": "mrkdwn", + "text": "āœ… *Status:* Authenticated\n\nYou can start chatting with me in any channel where I'm added!" + } + }) + else: + auth_message = auth_handler.create_auth_message(user_id, "app_home") + blocks.extend(auth_message["blocks"]) + + try: + client.views_publish( + user_id=user_id, + view={"type": "home", "blocks": blocks} + ) + except Exception as e: + logger.error(f"Error publishing home view: {e}") + + +@app.error +def custom_error_handler(error, body, logger): + """Global error handler""" + logger.error(f"āŒ Error: {error}") + logger.debug(f"Request body: {body}") + + +def start_oauth_server(): + """Start OAuth callback server in background thread""" + # Create a callback wrapper for processing messages after OAuth + def process_message_after_oauth(message, channel_id, thread_ts): + """Process a pending message after OAuth completion""" + # Use the app's client to send the response + handle_message_event( + message=message, + say=lambda **kwargs: app.client.chat_postMessage( + channel=channel_id, + thread_ts=kwargs.get("thread_ts"), + text=kwargs.get("text", ""), + blocks=kwargs.get("blocks") + ), + client=app.client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + flask_app = create_oauth_server( + CONFIG, + auth_handler, + state_manager, + databricks_client, + app.client, + process_message_after_oauth + ) + + def run_server(): + flask_app.run(port=CONFIG.PORT, debug=False, use_reloader=False) + + thread = threading.Thread(target=run_server, daemon=True) + thread.start() + logger.info(f"šŸ” OAuth callback server running on port {CONFIG.PORT}") + logger.info(f" → http://localhost:{CONFIG.PORT}/oauth/callback") + + +def main(): + """Main entry point""" + try: + logger.info("šŸš€ Starting bot in Socket Mode...") + + # Start OAuth callback server + start_oauth_server() + + # Start Socket Mode handler (blocking) + handler = SocketModeHandler(app, CONFIG.SLACK_APP_TOKEN) + logger.info("⚔ Socket Mode handler starting...") + handler.start() + + except KeyboardInterrupt: + logger.info("šŸ‘‹ Bot stopped by user") + except Exception as e: + logger.error(f"āŒ Error starting bot: {e}") + import traceback + traceback.print_exc() + sys.exit(1) + + +if __name__ == "__main__": + main() diff --git a/slack-bot/dev_tools/__init__.py b/slack-bot/dev_tools/__init__.py new file mode 100644 index 0000000..04373b8 --- /dev/null +++ b/slack-bot/dev_tools/__init__.py @@ -0,0 +1 @@ +# Dev tools package \ No newline at end of file diff --git a/slack-bot/dev_tools/devloop.sh b/slack-bot/dev_tools/devloop.sh new file mode 100755 index 0000000..e7882ba --- /dev/null +++ b/slack-bot/dev_tools/devloop.sh @@ -0,0 +1,193 @@ +#!/bin/bash +# Complete dev loop: Start bot, send test messages, observe logs +# +# Usage: +# ./dev_tools/devloop.sh # Interactive mode +# ./dev_tools/devloop.sh auto # Run automated tests + +set -e + +# Colors +RED='\033[0;31m' +GREEN='\033[0;32m' +YELLOW='\033[1;33m' +BLUE='\033[0;34m' +CYAN='\033[0;36m' +NC='\033[0m' + +# Config +BOT_SCRIPT="app_socket.py" +BOT_LOG="bot.log" + +echo -e "${BLUE}╔════════════════════════════════════════╗${NC}" +echo -e "${BLUE}ā•‘ Slack Bot Development Loop ā•‘${NC}" +echo -e "${BLUE}ā•šā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•ā•${NC}" +echo "" + +# Check if bot is running +check_bot() { + BOT_PID=$(ps aux | grep "[p]ython.*app" | awk '{print $2}') + if [ -z "$BOT_PID" ]; then + return 1 + fi + return 0 +} + +# Start bot +start_bot() { + echo -e "${YELLOW}šŸš€ Starting bot...${NC}" + + if check_bot; then + echo -e "${GREEN}āœ… Bot already running (PID: $BOT_PID)${NC}" + return 0 + fi + + # Start bot in background + nohup python "$BOT_SCRIPT" > "$BOT_LOG" 2>&1 & + BOT_PID=$! + + echo " Waiting for bot to initialize..." + sleep 3 + + if check_bot; then + echo -e "${GREEN}āœ… Bot started (PID: $BOT_PID)${NC}" + echo " Logs: tail -f $BOT_LOG" + else + echo -e "${RED}āŒ Failed to start bot${NC}" + echo "Check logs: cat $BOT_LOG" + exit 1 + fi +} + +# Stop bot +stop_bot() { + echo -e "${YELLOW}šŸ›‘ Stopping bot...${NC}" + + if check_bot; then + kill $BOT_PID + sleep 1 + echo -e "${GREEN}āœ… Bot stopped${NC}" + else + echo -e "${CYAN}ā„¹ļø Bot not running${NC}" + fi +} + +# Send test message +send_test() { + local message="$1" + echo -e "${CYAN}šŸ“¤ Sending: ${message}${NC}" + python dev_tools/send_test_message.py --mention "$message" +} + +# Run automated tests +run_tests() { + echo -e "${YELLOW}🧪 Running automated tests...${NC}" + echo "" + python dev_tools/test_bot_flow.py --verbose +} + +# Watch logs +watch_logs() { + echo -e "${CYAN}šŸ‘€ Watching logs (Ctrl+C to stop)...${NC}" + echo "" + tail -f "$BOT_LOG" | while IFS= read -r line; do + case "$line" in + *ERROR*|*āŒ*) + echo -e "${RED}${line}${NC}" + ;; + *WARNING*|*āš ļø*) + echo -e "${YELLOW}${line}${NC}" + ;; + *šŸ“Ø*|*šŸ“¢*) + echo -e "${CYAN}${line}${NC}" + ;; + *) + echo "$line" + ;; + esac + done +} + +# Interactive mode +interactive() { + while true; do + echo "" + echo -e "${BLUE}═══════════════════════════════════════${NC}" + echo "Choose an action:" + echo " 1) Start bot" + echo " 2) Stop bot" + echo " 3) Restart bot" + echo " 4) Send test message" + echo " 5) Run automated tests" + echo " 6) Watch logs" + echo " 7) View recent logs" + echo " q) Quit" + echo "" + read -p "Enter choice: " choice + + case $choice in + 1) + start_bot + ;; + 2) + stop_bot + ;; + 3) + stop_bot + sleep 1 + start_bot + ;; + 4) + read -p "Enter message: " msg + send_test "$msg" + ;; + 5) + run_tests + ;; + 6) + watch_logs + ;; + 7) + echo "" + tail -20 "$BOT_LOG" + ;; + q|Q) + echo "Bye!" + exit 0 + ;; + *) + echo "Invalid choice" + ;; + esac + done +} + +# Main +case "${1:-interactive}" in + auto|test) + start_bot + echo "" + run_tests + TEST_EXIT=$? + echo "" + stop_bot + exit $TEST_EXIT + ;; + start) + start_bot + ;; + stop) + stop_bot + ;; + restart) + stop_bot + sleep 1 + start_bot + ;; + logs) + watch_logs + ;; + interactive|*) + interactive + ;; +esac diff --git a/slack-bot/dev_tools/send_test_message.py b/slack-bot/dev_tools/send_test_message.py new file mode 100644 index 0000000..687bf3b --- /dev/null +++ b/slack-bot/dev_tools/send_test_message.py @@ -0,0 +1,138 @@ +#!/usr/bin/env python3 +""" +Send test messages to Slack as a user for testing the bot. + +Usage: + python dev_tools/send_test_message.py "Hello bot!" + python dev_tools/send_test_message.py "@BotName what is 2+2?" + python dev_tools/send_test_message.py "test message" --channel C123456 +""" + +import os +import sys +from pathlib import Path +from dotenv import load_dotenv +from slack_sdk import WebClient +from slack_sdk.errors import SlackApiError +import argparse + +# Load environment +env_file = Path(__file__).parent.parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) + +def send_message(text, channel=None, as_user=True): + """Send a message to Slack""" + + # Use user token to send message as a real user (not bot) + token = os.environ.get("SLACK_USER_TOKEN") + if not token: + print("āŒ SLACK_USER_TOKEN not found in environment") + print(" Add SLACK_USER_TOKEN to .env.local") + sys.exit(1) + + client = WebClient(token=token) + + # Get default test channel if not specified + if not channel: + channel = os.environ.get("SLACK_TEST_CHANNEL", "test-chatbot-app") + + try: + # Send message as user (don't use as_user parameter with user tokens) + response = client.chat_postMessage( + channel=channel, + text=text + ) + + print(f"āœ… Message sent to #{channel}") + print(f" Message: {text}") + print(f" Timestamp: {response['ts']}") + print(f" Thread: {response.get('thread_ts', response['ts'])}") + + return response + + except SlackApiError as e: + print(f"āŒ Error sending message: {e.response['error']}") + print(f" Details: {e.response}") + sys.exit(1) + + +def send_mention(text, channel=None, bot_user_id=None): + """Send a message mentioning the bot""" + + if not bot_user_id: + # Get bot user ID from bot token + bot_token = os.environ.get("SLACK_BOT_TOKEN") + client = WebClient(token=bot_token) + + try: + auth_response = client.auth_test() + bot_user_id = auth_response["user_id"] + bot_name = auth_response["user"] + print(f"šŸ¤– Bot: {bot_name} ({bot_user_id})") + except SlackApiError as e: + print(f"āŒ Error getting bot info: {e.response['error']}") + sys.exit(1) + + # Add mention to text + mention_text = f"<@{bot_user_id}> {text}" + + return send_message(mention_text, channel) + + +def send_dm(text, user_id=None): + """Send a DM to the bot""" + + token = os.environ.get("SLACK_BOT_TOKEN") + client = WebClient(token=token) + + if not user_id: + # Get current user from token (if user token) or use bot + try: + auth_response = client.auth_test() + user_id = auth_response["user_id"] + except SlackApiError as e: + print(f"āŒ Error: {e.response['error']}") + sys.exit(1) + + try: + # Open DM conversation + response = client.conversations_open(users=[user_id]) + dm_channel = response["channel"]["id"] + + print(f"šŸ’¬ Opened DM: {dm_channel}") + + # Send message + return send_message(text, channel=dm_channel) + + except SlackApiError as e: + print(f"āŒ Error opening DM: {e.response['error']}") + sys.exit(1) + + +def main(): + parser = argparse.ArgumentParser(description="Send test messages to Slack") + parser.add_argument("text", help="Message text to send") + parser.add_argument("--channel", "-c", help="Channel to send to (default: SLACK_TEST_CHANNEL env var)") + parser.add_argument("--mention", "-m", action="store_true", help="Mention the bot") + parser.add_argument("--dm", "-d", action="store_true", help="Send as DM") + parser.add_argument("--thread", "-t", help="Thread timestamp to reply to") + + args = parser.parse_args() + + print("šŸš€ Sending test message to Slack...") + print() + + if args.dm: + send_dm(args.text) + elif args.mention: + send_mention(args.text, channel=args.channel) + else: + send_message(args.text, channel=args.channel) + + print() + print("āœ… Done! Check the Slack channel and bot logs.") + + +if __name__ == "__main__": + main() diff --git a/slack-bot/dev_tools/test_bot_flow.py b/slack-bot/dev_tools/test_bot_flow.py new file mode 100644 index 0000000..ad9a058 --- /dev/null +++ b/slack-bot/dev_tools/test_bot_flow.py @@ -0,0 +1,230 @@ +#!/usr/bin/env python3 +""" +End-to-end test flow for the Slack bot. + +Tests: +1. Send message mentioning bot +2. Wait for response +3. Validate response +4. Check logs + +Usage: + python dev_tools/test_bot_flow.py + python dev_tools/test_bot_flow.py --verbose +""" + +import os +import sys +import time +from pathlib import Path +from dotenv import load_dotenv +from slack_sdk import WebClient +from slack_sdk.errors import SlackApiError +import argparse + +# Load environment +env_file = Path(__file__).parent.parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) + + +class BotTester: + def __init__(self, verbose=False): + self.verbose = verbose + self.user_token = os.environ.get("SLACK_USER_TOKEN") + self.bot_token = os.environ.get("SLACK_BOT_TOKEN") + + if not self.user_token: + raise Exception("SLACK_USER_TOKEN not found. Add to .env.local") + + self.client = WebClient(token=self.user_token) # Use user token to send messages + self.test_channel = os.environ.get("SLACK_TEST_CHANNEL", "test-chatbot-app") + + # Get bot info from bot token + bot_client = WebClient(token=self.bot_token) + auth_response = bot_client.auth_test() + self.bot_user_id = auth_response["user_id"] + self.bot_name = auth_response["user"] + + self.log(f"šŸ¤– Testing bot: {self.bot_name} ({self.bot_user_id})") + self.log(f"šŸ“ Test channel: #{self.test_channel}") + + def log(self, message, always=False): + """Log message if verbose or always""" + if self.verbose or always: + print(message) + + def send_message(self, text, mention=True): + """Send a test message""" + if mention: + text = f"<@{self.bot_user_id}> {text}" + + self.log(f"\nšŸ“¤ Sending: {text}") + + try: + response = self.client.chat_postMessage( + channel=self.test_channel, + text=text + ) + + self.log(f"āœ… Sent at {response['ts']}") + return response + + except SlackApiError as e: + print(f"āŒ Error sending message: {e.response['error']}") + raise + + def wait_for_response(self, after_ts, timeout=10): + """Wait for bot response after a timestamp""" + self.log(f"ā³ Waiting for response (timeout: {timeout}s)...") + + start_time = time.time() + + while (time.time() - start_time) < timeout: + try: + # Get recent messages + response = self.client.conversations_history( + channel=self.test_channel, + oldest=after_ts, + limit=10 + ) + + # Look for bot response + for message in response['messages']: + if message.get('user') == self.bot_user_id or message.get('bot_id'): + self.log(f"āœ… Got response: {message.get('text', '(attachment)')[:100]}") + return message + + time.sleep(1) + + except SlackApiError as e: + print(f"āŒ Error checking messages: {e.response['error']}") + raise + + self.log(f"āš ļø No response after {timeout}s", always=True) + return None + + def test_simple_message(self): + """Test 1: Simple message""" + print("\n" + "="*60) + print("TEST 1: Simple Message") + print("="*60) + + sent = self.send_message("Hello! This is a test.") + response = self.wait_for_response(sent['ts']) + + if response: + print("āœ… Test 1 PASSED: Bot responded") + return True + else: + print("āŒ Test 1 FAILED: No response") + return False + + def test_authentication_flow(self): + """Test 2: Authentication prompt""" + print("\n" + "="*60) + print("TEST 2: Authentication Flow") + print("="*60) + + sent = self.send_message("What can you do?") + response = self.wait_for_response(sent['ts']) + + if response: + # Check if it's an auth prompt or response + text = response.get('text', '').lower() + has_auth = 'authenticate' in text or 'sign in' in text + + if has_auth: + print("āœ… Test 2 PASSED: Bot prompted authentication") + else: + print("āœ… Test 2 PASSED: Bot responded (already authenticated?)") + return True + else: + print("āŒ Test 2 FAILED: No response") + return False + + def test_command(self): + """Test 3: Command""" + print("\n" + "="*60) + print("TEST 3: Logout Command") + print("="*60) + + # Try logout command + try: + response = self.client.chat_command( + channel=self.test_channel, + command="/databricks-logout" + ) + print("āœ… Test 3 PASSED: Command executed") + return True + except SlackApiError as e: + if e.response['error'] == 'invalid_command': + print("āš ļø Test 3 SKIPPED: Command not installed") + return True + else: + print(f"āŒ Test 3 FAILED: {e.response['error']}") + return False + + def run_all_tests(self): + """Run all tests""" + print("\n" + "="*60) + print("🧪 SLACK BOT END-TO-END TESTS") + print("="*60) + print(f"Bot: {self.bot_name}") + print(f"Channel: #{self.test_channel}") + print() + + results = [] + + # Run tests + results.append(("Simple Message", self.test_simple_message())) + results.append(("Authentication Flow", self.test_authentication_flow())) + results.append(("Logout Command", self.test_command())) + + # Summary + print("\n" + "="*60) + print("SUMMARY") + print("="*60) + + passed = sum(1 for _, result in results if result) + total = len(results) + + for test_name, result in results: + status = "āœ… PASS" if result else "āŒ FAIL" + print(f"{status} - {test_name}") + + print() + print(f"Results: {passed}/{total} tests passed") + + if passed == total: + print("šŸŽ‰ All tests passed!") + return 0 + else: + print("āŒ Some tests failed") + return 1 + + +def main(): + parser = argparse.ArgumentParser(description="Run end-to-end bot tests") + parser.add_argument("--verbose", "-v", action="store_true", help="Verbose output") + parser.add_argument("--channel", "-c", help="Test channel (default: SLACK_TEST_CHANNEL env)") + + args = parser.parse_args() + + if args.channel: + os.environ["SLACK_TEST_CHANNEL"] = args.channel + + try: + tester = BotTester(verbose=args.verbose) + exit_code = tester.run_all_tests() + sys.exit(exit_code) + + except Exception as e: + print(f"\nāŒ Fatal error: {e}") + import traceback + traceback.print_exc() + sys.exit(1) + + +if __name__ == "__main__": + main() diff --git a/slack-bot/dev_tools/watch_logs.sh b/slack-bot/dev_tools/watch_logs.sh new file mode 100755 index 0000000..565ace7 --- /dev/null +++ b/slack-bot/dev_tools/watch_logs.sh @@ -0,0 +1,53 @@ +#!/bin/bash +# Watch bot logs in real-time with color highlighting +# +# Usage: +# ./dev_tools/watch_logs.sh + +# Colors +RED='\033[0;31m' +GREEN='\033[0;32m' +YELLOW='\033[1;33m' +BLUE='\033[0;34m' +MAGENTA='\033[0;35m' +CYAN='\033[0;36m' +NC='\033[0m' # No Color + +echo "šŸ‘€ Watching bot logs..." +echo "Press Ctrl+C to stop" +echo "" + +# Find bot process +BOT_PID=$(ps aux | grep "[p]ython app" | awk '{print $2}') + +if [ -z "$BOT_PID" ]; then + echo "āŒ Bot not running. Start it with: python app_socket.py" + exit 1 +fi + +echo "Found bot process: $BOT_PID" +echo "" + +# Tail logs and colorize +tail -f /dev/stdin | while IFS= read -r line; do + case "$line" in + *ERROR*|*āŒ*) + echo -e "${RED}${line}${NC}" + ;; + *WARNING*|*āš ļø*) + echo -e "${YELLOW}${line}${NC}" + ;; + *INFO*|*āœ…*) + echo -e "${GREEN}${line}${NC}" + ;; + *DEBUG*) + echo -e "${CYAN}${line}${NC}" + ;; + *šŸ“Ø*|*šŸ“¢*) + echo -e "${MAGENTA}${line}${NC}" + ;; + *) + echo "$line" + ;; + esac +done diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index aa1fc57..ac4b26e 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -1,7 +1,9 @@ """Message event handler for Slack bot""" import asyncio +import json import logging +import uuid from typing import Dict, List from slack_sdk import WebClient @@ -44,6 +46,16 @@ def handle_message_event( # Check authentication if not auth_handler.is_authenticated(user_id): logger.info(f"User {user_id} not authenticated") + + # Store the pending message to auto-respond after OAuth + pending_data = { + "text": text, + "channel_id": channel_id, + "thread_ts": thread_ts, + } + state_manager.set_user_data(user_id, "pending_message", pending_data) + logger.info(f"Stored pending message for user {user_id}: channel={channel_id}, thread_ts={thread_ts}") + auth_message = auth_handler.create_auth_message(user_id, channel_id) # Send ephemeral message (only visible to user) @@ -79,7 +91,8 @@ def handle_message_event( user_id=user_id, say=say, client=client, - state_manager=state_manager + state_manager=state_manager, + logger=logger ) except Exception as e: @@ -106,7 +119,8 @@ def send_agent_responses( user_id: str, say, client: WebClient, - state_manager + state_manager, + logger ): """Send agent responses to Slack""" # Add user message to history @@ -116,12 +130,68 @@ def send_agent_responses( for item in response: # Handle response format: [{'role': 'assistant', 'content': '...'}] - if item.get("role") == "assistant" and "content" in item: - # Send message in thread (visible to everyone) - say( - text=item["content"], - thread_ts=thread_ts - ) + logger.info(f"Processing response item: role={item.get('role')}, has_content={bool(item.get('content'))}") + if item.get("role") == "assistant" and item.get("content"): + # Send ephemeral message (only visible to user) with share button + logger.info(f"Sending ephemeral message to user {user_id}") + # Generate unique ID for this response + response_id = str(uuid.uuid4()) + + # Store the response for later sharing + state_manager.set_user_data(user_id, f"pending_share_{response_id}", { + "content": item["content"], + "thread_ts": thread_ts, + "channel_id": channel_id + }) + + # Truncate content if too long for Slack blocks (max 3000 chars) + display_content = item["content"] + if len(display_content) > 2900: + display_content = display_content[:2900] + "... (truncated)" + + try: + logger.info(f"Ephemeral params: channel={channel_id}, user={user_id}, thread_ts={thread_ts}") + # Don't use thread_ts for ephemeral messages - they appear in channel view + result = client.chat_postEphemeral( + channel=channel_id, + user=user_id, + text=item["content"], # Fallback text + blocks=[ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": display_content + } + }, + { + "type": "actions", + "block_id": f"share_block_{response_id}", + "elements": [ + { + "type": "button", + "text": { + "type": "plain_text", + "text": "šŸ“¢ Share to channel", + "emoji": True + }, + "style": "primary", + "value": response_id, + "action_id": f"share_response_{response_id}" + } + ] + } + ] + ) + logger.info(f"Ephemeral message sent successfully with response_id: {response_id}") + logger.info(f"Ephemeral result: {result}") + except Exception as e: + logger.error(f"Failed to send ephemeral message: {e}") + import traceback + traceback.print_exc() + # Fallback: send as regular message + logger.info("Falling back to regular message") + say(text=item["content"], thread_ts=thread_ts) state_manager.add_message_to_history(thread_ts, "assistant", item["content"]) elif item.get("type") == "tool_call": @@ -145,6 +215,8 @@ def send_agent_responses( thread_ts=thread_ts ) + logger.info(f"Finished processing {len(response)} response items") + def create_tool_call_block(tool_call_info: Dict) -> List[Dict]: """Create Slack blocks for tool call display""" diff --git a/slack-bot/http_mode.py b/slack-bot/http_mode.py new file mode 100644 index 0000000..560ad6e --- /dev/null +++ b/slack-bot/http_mode.py @@ -0,0 +1,161 @@ +#!/usr/bin/env python3 +""" +Slack bot using HTTP mode with Events API. + +This version requires: +- Public HTTPS URL for Slack events +- Event Subscriptions configured in Slack app +- For production deployments only + +For local development, use app_socket.py instead. +""" + +import logging +import sys +from pathlib import Path +from dotenv import load_dotenv +from flask import Flask, request +from slack_bolt import App +from slack_bolt.adapter.flask import SlackRequestHandler + +# Load environment variables +env_file = Path(__file__).parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) +else: + load_dotenv(Path(__file__).parent / ".env") + +from config import DefaultConfig +from storage import StateManager +from handlers import AuthHandler, handle_message_event +from client import DatabricksClient +from oauth_server import create_oauth_server + +# Configure logging +logging.basicConfig( + level=logging.INFO, + format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', + handlers=[logging.StreamHandler(sys.stdout)] +) +logger = logging.getLogger(__name__) + +# Load configuration +CONFIG = DefaultConfig() + +# Validate required configuration +required_configs = [ + "SLACK_BOT_TOKEN", + "SLACK_SIGNING_SECRET", + "DATABRICKS_HOST", + "SERVING_ENDPOINT_NAME" +] +missing_configs = [c for c in required_configs if not getattr(CONFIG, c)] +if missing_configs: + logger.error(f"Missing required configuration: {', '.join(missing_configs)}") + sys.exit(1) + +# Initialize Slack app +app = App( + token=CONFIG.SLACK_BOT_TOKEN, + signing_secret=CONFIG.SLACK_SIGNING_SECRET, +) + +# Initialize components +state_manager = StateManager() +auth_handler = AuthHandler(CONFIG, state_manager) +databricks_client = DatabricksClient( + CONFIG.DATABRICKS_HOST, + CONFIG.SERVING_ENDPOINT_NAME +) + +logger.info("Slack bot initialized successfully") + + +# Event Handlers (same as Socket Mode) + +@app.event("message") +async def message_handler(message, say, client): + """Handle incoming message events""" + await handle_message_event( + message=message, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.event("app_mention") +async def handle_app_mention(event, say, client): + """Handle @mentions of the bot""" + await handle_message_event( + message=event, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + +@app.command("/databricks-logout") +async def handle_logout_command(ack, command, respond): + """Handle /databricks-logout slash command""" + await ack() + user_id = command["user_id"] + auth_handler.handle_logout(user_id) + await respond("You have been logged out from Databricks.") + + +@app.error +async def custom_error_handler(error, body, logger): + """Global error handler""" + logger.error(f"Error: {error}") + + +# Create Flask app for HTTP mode +flask_app = Flask(__name__) +slack_handler = SlackRequestHandler(app) + +# OAuth callback server +oauth_app = create_oauth_server(CONFIG, auth_handler, state_manager) +flask_app.register_blueprint(oauth_app) + + +@flask_app.route("/slack/events", methods=["POST"]) +def slack_events(): + """Handle Slack events via HTTP""" + return slack_handler.handle(request) + + +@flask_app.route("/health", methods=["GET"]) +def health_check(): + """Health check endpoint""" + return {"status": "ok"}, 200 + + +def main(): + """Main entry point""" + try: + logger.info(f"Starting bot in HTTP mode on port {CONFIG.PORT}...") + logger.info(f"Slack events endpoint: http://localhost:{CONFIG.PORT}/slack/events") + logger.info(f"OAuth callback: http://localhost:{CONFIG.PORT}/oauth/callback") + logger.warning("āš ļø HTTP mode requires public HTTPS URL for production!") + logger.warning(" For local development, use app_socket.py instead") + + flask_app.run(port=CONFIG.PORT, debug=False) + + except KeyboardInterrupt: + logger.info("Bot stopped by user") + except Exception as e: + logger.error(f"Error starting bot: {e}") + import traceback + traceback.print_exc() + sys.exit(1) + + +if __name__ == "__main__": + main() diff --git a/slack-bot/oauth_server.py b/slack-bot/oauth_server.py new file mode 100644 index 0000000..08bfce5 --- /dev/null +++ b/slack-bot/oauth_server.py @@ -0,0 +1,142 @@ +#!/usr/bin/env python3 +""" +Standalone OAuth callback server for Databricks authentication. +Runs in background thread alongside Socket Mode. +""" + +import asyncio +import logging +from flask import Flask, request + +logger = logging.getLogger(__name__) + + +def create_oauth_server(config, auth_handler, state_manager, databricks_client, slack_client, handle_message_callback): + """Create Flask app for OAuth callbacks""" + + app = Flask(__name__) + + @app.route("/oauth/callback", methods=["GET"]) + def oauth_callback(): + """ + Handle OAuth callback from Databricks. + Exchanges authorization code for access token. + """ + try: + # Get authorization code and state from query parameters + code = request.args.get("code") + state = request.args.get("state") + error = request.args.get("error") + + if error: + logger.error(f"OAuth error: {error}") + return _error_page(f"Error: {error}"), 400 + + if not code or not state: + logger.error("Missing code or state in OAuth callback") + return _error_page("Missing required parameters"), 400 + + # Parse state to get user_id + try: + user_id, channel_id = state.split(":") + except ValueError: + logger.error(f"Invalid state format: {state}") + return _error_page("Invalid state"), 400 + + # Validate state (CSRF protection) + if not auth_handler.validate_state(state, user_id): + logger.error(f"State validation failed for user {user_id}") + return _error_page("Security validation failed"), 400 + + # Exchange code for token (run async function in sync context) + access_token, expires_in = asyncio.run(auth_handler.exchange_code_for_token(code, user_id)) + + if not access_token: + logger.error(f"Failed to exchange code for token for user {user_id}") + return _error_page("Token exchange failed"), 500 + + # Store token + state_manager.store_user_token(user_id, access_token, expires_in) + logger.info(f"Successfully authenticated user {user_id}") + + # Check for pending message and auto-respond + pending_message = state_manager.get_user_data(user_id, "pending_message") + if pending_message: + logger.info(f"Processing pending message for user {user_id}") + try: + # Process the pending message + handle_message_callback( + message={ + "user": user_id, + "text": pending_message["text"], + "channel": pending_message["channel_id"], + "ts": pending_message["thread_ts"], + "thread_ts": pending_message["thread_ts"] + }, + channel_id=pending_message["channel_id"], + thread_ts=pending_message["thread_ts"] + ) + # Clear pending message after processing + state_manager.set_user_data(user_id, "pending_message", None) + logger.info(f"Successfully processed pending message for user {user_id}") + except Exception as e: + logger.error(f"Failed to process pending message: {e}") + import traceback + traceback.print_exc() + + # Send success message to Slack + try: + success_text = f"āœ… <@{user_id}> You're now authenticated!" + if pending_message: + success_text += " Processing your message..." + else: + success_text += " Send me a message to get started." + + slack_client.chat_postMessage( + channel=channel_id, + text=success_text, + ) + except Exception as e: + logger.error(f"Failed to send success message to Slack: {e}") + + # Return success page + return _success_page(), 200 + + except Exception as e: + logger.error(f"Error in OAuth callback: {e}") + import traceback + traceback.print_exc() + return _error_page("An unexpected error occurred"), 500 + + return app + + +def _success_page(): + """Success page HTML""" + return """ + + +

āœ… Authentication Successful!

+

You're now connected to Databricks.

+

You can close this window and return to Slack.

+ + + + """ + + +def _error_page(message): + """Error page HTML""" + return f""" + + +

āŒ Authentication Failed

+

{message}

+

Please close this window and try again in Slack.

+ + + """ diff --git a/slack-bot/test_connection.py b/slack-bot/test_connection.py new file mode 100644 index 0000000..e21c386 --- /dev/null +++ b/slack-bot/test_connection.py @@ -0,0 +1,68 @@ +#!/usr/bin/env python3 +""" +Quick test to verify Slack connection and configuration +""" + +import os +from pathlib import Path +from dotenv import load_dotenv + +# Load env +env_file = Path(__file__).parent / ".env.local" +if env_file.exists(): + load_dotenv(env_file) + print(f"āœ… Loaded: {env_file}") +else: + print("āŒ No .env.local found") + exit(1) + +# Check required vars +required_vars = [ + "SLACK_BOT_TOKEN", + "SLACK_APP_TOKEN", + "SLACK_SIGNING_SECRET", + "DATABRICKS_HOST", + "SERVING_ENDPOINT_NAME" +] + +print("\nšŸ“‹ Configuration Check:") +all_set = True +for var in required_vars: + value = os.environ.get(var, "") + if value: + masked = value[:10] + "..." if len(value) > 10 else value + print(f" āœ… {var}: {masked}") + else: + print(f" āŒ {var}: NOT SET") + all_set = False + +if not all_set: + print("\nāŒ Missing required variables!") + exit(1) + +# Test Slack connection +print("\nšŸ”Œ Testing Slack API connection...") +try: + from slack_sdk import WebClient + client = WebClient(token=os.environ["SLACK_BOT_TOKEN"]) + + auth_response = client.auth_test() + print(f" āœ… Connected to Slack!") + print(f" šŸ“± Bot User ID: {auth_response['user_id']}") + print(f" šŸ‘¤ Bot Name: {auth_response['user']}") + print(f" šŸ¢ Team: {auth_response['team']}") + +except Exception as e: + print(f" āŒ Slack connection failed: {e}") + exit(1) + +# Check Socket Mode token +print("\nšŸ” Testing Socket Mode token...") +app_token = os.environ.get("SLACK_APP_TOKEN", "") +if app_token.startswith("xapp-"): + print(f" āœ… App token format correct: {app_token[:15]}...") +else: + print(f" āŒ App token should start with 'xapp-', got: {app_token[:10]}...") + +print("\nāœ… All checks passed! Ready to run the bot.") +print("\nRun: python app.py") From edda0a61d866210d40a2695ff2fbdadeeb476246 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Tue, 28 Oct 2025 20:46:17 -0700 Subject: [PATCH 14/21] Add rich UX features: reactions, status messages, and share/generate buttons MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Features: - Eyes emoji reaction on message receipt for instant acknowledgment - Public "working on it" status message after authentication - Ephemeral responses with Share to channel button - Public "Generate my own answer" button for other users - Auto-hide buttons after use UX improvements: - Share posts raw content without user attribution prefix - Generate button creates personalized answer for each user - Both buttons hide and show completion status after use šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/app_socket.py | 81 ++++++++++++++++++++++++++- slack-bot/handlers/message_handler.py | 59 ++++++++++++++++++- 2 files changed, 136 insertions(+), 4 deletions(-) diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py index a3a8818..6e8b4a2 100644 --- a/slack-bot/app_socket.py +++ b/slack-bot/app_socket.py @@ -131,14 +131,14 @@ def handle_share_response(ack, body, client): response_data = state_manager.get_user_data(user_id, f"pending_share_{response_id}") if response_data: - # Post the message publicly in the thread + # Post the message publicly in the thread (raw content without prefix) client.chat_postMessage( channel=response_data["channel_id"], thread_ts=response_data["thread_ts"], - text=f"<@{user_id}> shared:\n\n{response_data['content']}" + text=response_data['content'] ) - # Update the ephemeral message to show it was shared + # Update the ephemeral message to hide the share button client.chat_update( channel=body["channel"]["id"], ts=body["message"]["ts"], @@ -169,6 +169,81 @@ def handle_share_response(ack, body, client): logger.info(f"User {user_id} shared response {response_id} to channel") +@app.action("generate_own_answer") +def handle_generate_own_answer(ack, body, client, say): + """Handle Generate my own answer button click""" + ack() + + user_id = body["user"]["id"] + action = body["actions"][0] + value = action["value"] + + # Parse thread_ts and original text from value + try: + thread_ts, original_text = value.split(":", 1) + except ValueError: + logger.error(f"Invalid action value format: {value}") + return + + channel_id = body["channel"]["id"] + + logger.info(f"User {user_id} clicked Generate button for message: {original_text}") + + # Update the message to hide the button and show it was used + try: + original_message = body["message"] + # Find the original text block + text_block = None + for block in original_message.get("blocks", []): + if block.get("type") == "section": + text_block = block + break + + updated_blocks = [] + if text_block: + updated_blocks.append(text_block) + + # Add a context showing the button was used + updated_blocks.append({ + "type": "context", + "elements": [ + { + "type": "mrkdwn", + "text": f"āœ… <@{user_id}> generated their own answer" + } + ] + }) + + client.chat_update( + channel=channel_id, + ts=original_message["ts"], + text=original_message.get("text", ""), + blocks=updated_blocks + ) + except Exception as e: + logger.error(f"Failed to update Generate button message: {e}") + + # Create a synthetic message event for this user + synthetic_message = { + "user": user_id, + "text": original_text, + "channel": channel_id, + "ts": thread_ts, + "thread_ts": thread_ts + } + + # Process the message for this user + handle_message_event( + message=synthetic_message, + say=say, + client=client, + state_manager=state_manager, + auth_handler=auth_handler, + databricks_client=databricks_client, + logger=logger + ) + + @app.command("/databricks-logout") def handle_logout_command(ack, command, respond): """Handle /databricks-logout slash command""" diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index ac4b26e..883202b 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -27,6 +27,7 @@ def handle_message_event( text = message.get("text", "").strip() channel_id = message.get("channel") thread_ts = message.get("thread_ts") or message.get("ts") + message_ts = message.get("ts") # Ignore bot messages and empty messages if not user_id or not text or message.get("bot_id"): @@ -34,6 +35,16 @@ def handle_message_event( logger.info(f"Processing message from user {user_id}: {text}") + # React with eyes emoji to acknowledge receipt + try: + client.reactions_add( + channel=channel_id, + timestamp=message_ts, + name="eyes" + ) + except Exception as e: + logger.error(f"Failed to add reaction: {e}") + # Handle special commands if text.lower() == "logout": handle_logout(user_id, channel_id, thread_ts, say, auth_handler) @@ -70,6 +81,16 @@ def handle_message_event( # Get user's OAuth token user_token = state_manager.get_user_token(user_id) + # Send public "working on it" message + try: + client.chat_postMessage( + channel=channel_id, + thread_ts=thread_ts, + text=f"Hi <@{user_id}>, I'm asking Databricks agents to respond to your message..." + ) + except Exception as e: + logger.error(f"Failed to send working message: {e}") + # Get conversation history for this thread history = state_manager.get_conversation_history(thread_ts) @@ -151,10 +172,10 @@ def send_agent_responses( try: logger.info(f"Ephemeral params: channel={channel_id}, user={user_id}, thread_ts={thread_ts}") - # Don't use thread_ts for ephemeral messages - they appear in channel view result = client.chat_postEphemeral( channel=channel_id, user=user_id, + thread_ts=thread_ts, text=item["content"], # Fallback text blocks=[ { @@ -185,6 +206,42 @@ def send_agent_responses( ) logger.info(f"Ephemeral message sent successfully with response_id: {response_id}") logger.info(f"Ephemeral result: {result}") + + # Send public message with Generate button for others + try: + client.chat_postMessage( + channel=channel_id, + thread_ts=thread_ts, + text=f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful.", + blocks=[ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful." + } + }, + { + "type": "actions", + "elements": [ + { + "type": "button", + "text": { + "type": "plain_text", + "text": "šŸŽÆ Generate my own answer", + "emoji": True + }, + "style": "primary", + "value": f"{thread_ts}:{text}", + "action_id": "generate_own_answer" + } + ] + } + ] + ) + except Exception as e: + logger.error(f"Failed to send generate button message: {e}") + except Exception as e: logger.error(f"Failed to send ephemeral message: {e}") import traceback From 2ddd17a198f9c8d9df24ea2690b8fafe6592c3b4 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Tue, 28 Oct 2025 20:52:16 -0700 Subject: [PATCH 15/21] Clean up UX: remove duplicate messages, auto-delete status, instant button updates MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Improvements: - Remove duplicate OAuth success message in channel - Auto-delete "working on it" message after response is sent - Instant button removal on Share click using response_action="update" - Cleaner, less cluttered message flow šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/app_socket.py | 25 ++++++++++++------------- slack-bot/handlers/message_handler.py | 23 +++++++++++++++++++---- slack-bot/oauth_server.py | 15 +-------------- 3 files changed, 32 insertions(+), 31 deletions(-) diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py index 6e8b4a2..c13149c 100644 --- a/slack-bot/app_socket.py +++ b/slack-bot/app_socket.py @@ -121,8 +121,6 @@ def message_handler(message, say, client): @app.action(re.compile(r"^share_response_.*")) def handle_share_response(ack, body, client): """Handle share to channel button click""" - ack() - user_id = body["user"]["id"] action = body["actions"][0] response_id = action["value"] @@ -131,17 +129,9 @@ def handle_share_response(ack, body, client): response_data = state_manager.get_user_data(user_id, f"pending_share_{response_id}") if response_data: - # Post the message publicly in the thread (raw content without prefix) - client.chat_postMessage( - channel=response_data["channel_id"], - thread_ts=response_data["thread_ts"], - text=response_data['content'] - ) - - # Update the ephemeral message to hide the share button - client.chat_update( - channel=body["channel"]["id"], - ts=body["message"]["ts"], + # Acknowledge with updated UI (removes button immediately) + ack( + response_action="update", text=response_data["content"], blocks=[ { @@ -163,10 +153,19 @@ def handle_share_response(ack, body, client): ] ) + # Post the message publicly in the thread (raw content without prefix) + client.chat_postMessage( + channel=response_data["channel_id"], + thread_ts=response_data["thread_ts"], + text=response_data['content'] + ) + # Clean up stored data state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) logger.info(f"User {user_id} shared response {response_id} to channel") + else: + ack() @app.action("generate_own_answer") diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index 883202b..e11adc3 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -81,13 +81,15 @@ def handle_message_event( # Get user's OAuth token user_token = state_manager.get_user_token(user_id) - # Send public "working on it" message + # Send public "working on it" message and store its timestamp to delete later + working_msg_ts = None try: - client.chat_postMessage( + result = client.chat_postMessage( channel=channel_id, thread_ts=thread_ts, text=f"Hi <@{user_id}>, I'm asking Databricks agents to respond to your message..." ) + working_msg_ts = result["ts"] except Exception as e: logger.error(f"Failed to send working message: {e}") @@ -113,7 +115,8 @@ def handle_message_event( say=say, client=client, state_manager=state_manager, - logger=logger + logger=logger, + working_msg_ts=working_msg_ts ) except Exception as e: @@ -141,7 +144,8 @@ def send_agent_responses( say, client: WebClient, state_manager, - logger + logger, + working_msg_ts=None ): """Send agent responses to Slack""" # Add user message to history @@ -272,6 +276,17 @@ def send_agent_responses( thread_ts=thread_ts ) + # Delete the "working on it" message now that we've responded + if working_msg_ts: + try: + client.chat_delete( + channel=channel_id, + ts=working_msg_ts + ) + logger.info(f"Deleted working message with ts {working_msg_ts}") + except Exception as e: + logger.error(f"Failed to delete working message: {e}") + logger.info(f"Finished processing {len(response)} response items") diff --git a/slack-bot/oauth_server.py b/slack-bot/oauth_server.py index 08bfce5..b28bbaf 100644 --- a/slack-bot/oauth_server.py +++ b/slack-bot/oauth_server.py @@ -84,20 +84,7 @@ def oauth_callback(): import traceback traceback.print_exc() - # Send success message to Slack - try: - success_text = f"āœ… <@{user_id}> You're now authenticated!" - if pending_message: - success_text += " Processing your message..." - else: - success_text += " Send me a message to get started." - - slack_client.chat_postMessage( - channel=channel_id, - text=success_text, - ) - except Exception as e: - logger.error(f"Failed to send success message to Slack: {e}") + # Don't send a separate success message - the inline "working on it" message is enough # Return success page return _success_page(), 200 From d51d7e07325835b3da918c11a7891c901433f0d0 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Wed, 29 Oct 2025 11:22:25 -0700 Subject: [PATCH 16/21] WIP Signed-off-by: Sid Murching --- slack-bot/handlers/message_handler.py | 18 +++++++++--------- slack-bot/oauth_server.py | 4 ++++ 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index e11adc3..ec42719 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -58,15 +58,6 @@ def handle_message_event( if not auth_handler.is_authenticated(user_id): logger.info(f"User {user_id} not authenticated") - # Store the pending message to auto-respond after OAuth - pending_data = { - "text": text, - "channel_id": channel_id, - "thread_ts": thread_ts, - } - state_manager.set_user_data(user_id, "pending_message", pending_data) - logger.info(f"Stored pending message for user {user_id}: channel={channel_id}, thread_ts={thread_ts}") - auth_message = auth_handler.create_auth_message(user_id, channel_id) # Send ephemeral message (only visible to user) @@ -76,6 +67,15 @@ def handle_message_event( text=auth_message["text"], blocks=auth_message.get("blocks") ) + + # Store the pending message to auto-respond after OAuth + pending_data = { + "text": text, + "channel_id": channel_id, + "thread_ts": thread_ts, + } + state_manager.set_user_data(user_id, "pending_message", pending_data) + logger.info(f"Stored pending message for user {user_id}: channel={channel_id}, thread_ts={thread_ts}") return # Get user's OAuth token diff --git a/slack-bot/oauth_server.py b/slack-bot/oauth_server.py index b28bbaf..e16d61e 100644 --- a/slack-bot/oauth_server.py +++ b/slack-bot/oauth_server.py @@ -63,6 +63,10 @@ def oauth_callback(): pending_message = state_manager.get_user_data(user_id, "pending_message") if pending_message: logger.info(f"Processing pending message for user {user_id}") + + # Note: Ephemeral messages cannot be updated via API after OAuth callback + # The auth message will remain visible to the user until they refresh + try: # Process the pending message handle_message_callback( From a480367b36c739abc722d33fd94abdedaea8031f Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Wed, 29 Oct 2025 23:33:25 -0700 Subject: [PATCH 17/21] Show OAuth prompt in thread context instead of channel MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Add thread_ts parameter to ephemeral OAuth message - Ensures login button appears in thread view, not main channel - Improves UX by keeping authentication flow in context šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/handlers/message_handler.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index ec42719..7595811 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -60,10 +60,11 @@ def handle_message_event( auth_message = auth_handler.create_auth_message(user_id, channel_id) - # Send ephemeral message (only visible to user) + # Send ephemeral message (only visible to user) in the thread client.chat_postEphemeral( channel=channel_id, user=user_id, + thread_ts=thread_ts, text=auth_message["text"], blocks=auth_message.get("blocks") ) From 833aef6a5595b65e06ebad7de02473afa0c75b0a Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Thu, 30 Oct 2025 00:12:18 -0700 Subject: [PATCH 18/21] Improve button UX and add dismiss functionality MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Changes: - Rename "Share to channel" button to "Post in thread" for clarity - Add "Dismiss" button to allow users to dismiss ephemeral messages - Add white_check_mark reaction when posting to thread - Update status message from "Shared to channel" to "Posted in thread" - Add debug logging for button click handlers Note: Ephemeral message dismissal needs further investigation šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/app_socket.py | 33 +++++++++++++++++++++++++-- slack-bot/handlers/message_handler.py | 12 +++++++++- 2 files changed, 42 insertions(+), 3 deletions(-) diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py index c13149c..66ed64b 100644 --- a/slack-bot/app_socket.py +++ b/slack-bot/app_socket.py @@ -121,9 +121,11 @@ def message_handler(message, say, client): @app.action(re.compile(r"^share_response_.*")) def handle_share_response(ack, body, client): """Handle share to channel button click""" + logger.info(f"Share button clicked! Body: {body}") user_id = body["user"]["id"] action = body["actions"][0] response_id = action["value"] + logger.info(f"User {user_id} clicked share button for response {response_id}") # Retrieve the stored response response_data = state_manager.get_user_data(user_id, f"pending_share_{response_id}") @@ -146,7 +148,7 @@ def handle_share_response(ack, body, client): "elements": [ { "type": "mrkdwn", - "text": "āœ… _Shared to channel_" + "text": "āœ… _Posted in thread_" } ] } @@ -154,12 +156,22 @@ def handle_share_response(ack, body, client): ) # Post the message publicly in the thread (raw content without prefix) - client.chat_postMessage( + result = client.chat_postMessage( channel=response_data["channel_id"], thread_ts=response_data["thread_ts"], text=response_data['content'] ) + # Add white_check_mark reaction to the posted message + try: + client.reactions_add( + channel=response_data["channel_id"], + timestamp=result["ts"], + name="white_check_mark" + ) + except Exception as e: + logger.error(f"Failed to add reaction: {e}") + # Clean up stored data state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) @@ -168,6 +180,23 @@ def handle_share_response(ack, body, client): ack() +@app.action(re.compile(r"^dismiss_response_.*")) +def handle_dismiss_response(ack, body, client): + """Handle dismiss button click - delete ephemeral message""" + logger.info(f"Dismiss button clicked! Body: {body}") + user_id = body["user"]["id"] + action = body["actions"][0] + response_id = action["value"] + + # Acknowledge and delete the ephemeral message + ack(response_action="clear") + + # Clean up stored data + state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) + + logger.info(f"User {user_id} dismissed response {response_id}") + + @app.action("generate_own_answer") def handle_generate_own_answer(ack, body, client, say): """Handle Generate my own answer button click""" diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index 7595811..d300d90 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -198,12 +198,22 @@ def send_agent_responses( "type": "button", "text": { "type": "plain_text", - "text": "šŸ“¢ Share to channel", + "text": "Post in thread", "emoji": True }, "style": "primary", "value": response_id, "action_id": f"share_response_{response_id}" + }, + { + "type": "button", + "text": { + "type": "plain_text", + "text": "Dismiss", + "emoji": True + }, + "value": response_id, + "action_id": f"dismiss_response_{response_id}" } ] } From 41b284e08b9987456be3064b7b409f9521ac46c1 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Thu, 30 Oct 2025 00:31:27 -0700 Subject: [PATCH 19/21] Fix ephemeral message dismissal using response_url MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Changes: - Use respond(delete_original=True) instead of ack(response_action="clear") - Both "Post in thread" and "Dismiss" buttons now properly delete ephemeral messages - Add white_check_mark reaction to original user message when posting - Store message_ts in response_data for reaction targeting - Add extensive logging for debugging button interactions Fixes the issue where ephemeral messages weren't being hidden when buttons were clicked. According to Slack docs, ephemeral messages must be deleted via response_url, not response_action. šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- slack-bot/app_socket.py | 85 +++++++++++++++------------ slack-bot/handlers/message_handler.py | 9 ++- 2 files changed, 52 insertions(+), 42 deletions(-) diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py index 66ed64b..1a723be 100644 --- a/slack-bot/app_socket.py +++ b/slack-bot/app_socket.py @@ -119,9 +119,9 @@ def message_handler(message, say, client): @app.action(re.compile(r"^share_response_.*")) -def handle_share_response(ack, body, client): +def handle_share_response(ack, body, client, respond): """Handle share to channel button click""" - logger.info(f"Share button clicked! Body: {body}") + logger.info(f"Share button clicked! Body keys: {body.keys()}") user_id = body["user"]["id"] action = body["actions"][0] response_id = action["value"] @@ -131,29 +131,21 @@ def handle_share_response(ack, body, client): response_data = state_manager.get_user_data(user_id, f"pending_share_{response_id}") if response_data: - # Acknowledge with updated UI (removes button immediately) - ack( - response_action="update", - text=response_data["content"], - blocks=[ - { - "type": "section", - "text": { - "type": "mrkdwn", - "text": response_data["content"] - } - }, - { - "type": "context", - "elements": [ - { - "type": "mrkdwn", - "text": "āœ… _Posted in thread_" - } - ] - } - ] - ) + logger.info(f"Response data found: {response_data.keys()}") + + # Acknowledge first + ack() + logger.info("Sent ack") + + # Delete the ephemeral message + logger.info("Deleting ephemeral message using respond with delete_original") + try: + respond(delete_original=True) + logger.info("Delete request sent successfully") + except Exception as e: + logger.error(f"Failed to delete message: {e}") + import traceback + traceback.print_exc() # Post the message publicly in the thread (raw content without prefix) result = client.chat_postMessage( @@ -161,35 +153,50 @@ def handle_share_response(ack, body, client): thread_ts=response_data["thread_ts"], text=response_data['content'] ) - - # Add white_check_mark reaction to the posted message - try: - client.reactions_add( - channel=response_data["channel_id"], - timestamp=result["ts"], - name="white_check_mark" - ) - except Exception as e: - logger.error(f"Failed to add reaction: {e}") + logger.info(f"Posted message to thread: {result['ts']}") + + # Add white_check_mark reaction to the original user message + if response_data.get("message_ts"): + try: + client.reactions_add( + channel=response_data["channel_id"], + timestamp=response_data["message_ts"], + name="white_check_mark" + ) + logger.info(f"Added reaction to message {response_data['message_ts']}") + except Exception as e: + logger.error(f"Failed to add reaction: {e}") # Clean up stored data state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) logger.info(f"User {user_id} shared response {response_id} to channel") else: + logger.warning(f"No response data found for response_id {response_id}") ack() @app.action(re.compile(r"^dismiss_response_.*")) -def handle_dismiss_response(ack, body, client): +def handle_dismiss_response(ack, body, client, respond): """Handle dismiss button click - delete ephemeral message""" - logger.info(f"Dismiss button clicked! Body: {body}") + logger.info(f"Dismiss button clicked! Body keys: {body.keys()}") user_id = body["user"]["id"] action = body["actions"][0] response_id = action["value"] - # Acknowledge and delete the ephemeral message - ack(response_action="clear") + # Acknowledge first + ack() + logger.info("Sent ack") + + # Delete the ephemeral message using response_url + logger.info("Deleting ephemeral message using respond with delete_original") + try: + respond(delete_original=True) + logger.info("Delete request sent successfully") + except Exception as e: + logger.error(f"Failed to delete ephemeral message: {e}") + import traceback + traceback.print_exc() # Clean up stored data state_manager.set_user_data(user_id, f"pending_share_{response_id}", None) diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index d300d90..b252071 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -117,7 +117,8 @@ def handle_message_event( client=client, state_manager=state_manager, logger=logger, - working_msg_ts=working_msg_ts + working_msg_ts=working_msg_ts, + message_ts=message_ts ) except Exception as e: @@ -146,7 +147,8 @@ def send_agent_responses( client: WebClient, state_manager, logger, - working_msg_ts=None + working_msg_ts=None, + message_ts=None ): """Send agent responses to Slack""" # Add user message to history @@ -167,7 +169,8 @@ def send_agent_responses( state_manager.set_user_data(user_id, f"pending_share_{response_id}", { "content": item["content"], "thread_ts": thread_ts, - "channel_id": channel_id + "channel_id": channel_id, + "message_ts": message_ts # Original user message timestamp for reaction }) # Truncate content if too long for Slack blocks (max 3000 chars) From 2a549c99f4a43b779e980750e59c5c5d693eae52 Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Mon, 3 Nov 2025 07:50:37 -0800 Subject: [PATCH 20/21] WIP updating slackbot Signed-off-by: Sid Murching --- slack-bot/app_socket.py | 21 +++++--- slack-bot/handlers/message_handler.py | 76 ++++++++++++++------------- 2 files changed, 53 insertions(+), 44 deletions(-) diff --git a/slack-bot/app_socket.py b/slack-bot/app_socket.py index 1a723be..98d826b 100644 --- a/slack-bot/app_socket.py +++ b/slack-bot/app_socket.py @@ -205,10 +205,8 @@ def handle_dismiss_response(ack, body, client, respond): @app.action("generate_own_answer") -def handle_generate_own_answer(ack, body, client, say): +def handle_generate_own_answer(ack, body, client, say, respond): """Handle Generate my own answer button click""" - ack() - user_id = body["user"]["id"] action = body["actions"][0] value = action["value"] @@ -218,12 +216,16 @@ def handle_generate_own_answer(ack, body, client, say): thread_ts, original_text = value.split(":", 1) except ValueError: logger.error(f"Invalid action value format: {value}") + ack() return channel_id = body["channel"]["id"] logger.info(f"User {user_id} clicked Generate button for message: {original_text}") + # Acknowledge first + ack() + # Update the message to hide the button and show it was used try: original_message = body["message"] @@ -249,14 +251,16 @@ def handle_generate_own_answer(ack, body, client, say): ] }) - client.chat_update( - channel=channel_id, - ts=original_message["ts"], + respond( + replace_original=True, text=original_message.get("text", ""), blocks=updated_blocks ) + logger.info("Generate button hidden successfully") except Exception as e: logger.error(f"Failed to update Generate button message: {e}") + import traceback + traceback.print_exc() # Create a synthetic message event for this user synthetic_message = { @@ -267,7 +271,7 @@ def handle_generate_own_answer(ack, body, client, say): "thread_ts": thread_ts } - # Process the message for this user + # Process the message for this user (don't show Generate button to avoid cascading) handle_message_event( message=synthetic_message, say=say, @@ -275,7 +279,8 @@ def handle_generate_own_answer(ack, body, client, say): state_manager=state_manager, auth_handler=auth_handler, databricks_client=databricks_client, - logger=logger + logger=logger, + show_generate_button=False ) diff --git a/slack-bot/handlers/message_handler.py b/slack-bot/handlers/message_handler.py index b252071..5ca580f 100644 --- a/slack-bot/handlers/message_handler.py +++ b/slack-bot/handlers/message_handler.py @@ -15,7 +15,8 @@ def handle_message_event( state_manager, auth_handler, databricks_client, - logger + logger, + show_generate_button=True ): """ Handle incoming Slack message events. @@ -118,7 +119,8 @@ def handle_message_event( state_manager=state_manager, logger=logger, working_msg_ts=working_msg_ts, - message_ts=message_ts + message_ts=message_ts, + show_generate_button=show_generate_button ) except Exception as e: @@ -148,7 +150,8 @@ def send_agent_responses( state_manager, logger, working_msg_ts=None, - message_ts=None + message_ts=None, + show_generate_button=True ): """Send agent responses to Slack""" # Add user message to history @@ -225,40 +228,41 @@ def send_agent_responses( logger.info(f"Ephemeral message sent successfully with response_id: {response_id}") logger.info(f"Ephemeral result: {result}") - # Send public message with Generate button for others - try: - client.chat_postMessage( - channel=channel_id, - thread_ts=thread_ts, - text=f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful.", - blocks=[ - { - "type": "section", - "text": { - "type": "mrkdwn", - "text": f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful." - } - }, - { - "type": "actions", - "elements": [ - { - "type": "button", - "text": { - "type": "plain_text", - "text": "šŸŽÆ Generate my own answer", - "emoji": True - }, - "style": "primary", - "value": f"{thread_ts}:{text}", - "action_id": "generate_own_answer" + # Send public message with Generate button for others (only for initial responses, not for Generate clicks) + if show_generate_button: + try: + client.chat_postMessage( + channel=channel_id, + thread_ts=thread_ts, + text=f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful.", + blocks=[ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": f"I've privately shared an answer with <@{user_id}> in this thread. Generate your own personalized answer by clicking the button below, and share it with your teammates if you find it useful." } - ] - } - ] - ) - except Exception as e: - logger.error(f"Failed to send generate button message: {e}") + }, + { + "type": "actions", + "elements": [ + { + "type": "button", + "text": { + "type": "plain_text", + "text": "šŸŽÆ Generate my own answer", + "emoji": True + }, + "style": "primary", + "value": f"{thread_ts}:{text}", + "action_id": "generate_own_answer" + } + ] + } + ] + ) + except Exception as e: + logger.error(f"Failed to send generate button message: {e}") except Exception as e: logger.error(f"Failed to send ephemeral message: {e}") From 19537654aac7cd81ccb8c113f26f76cd4f66414a Mon Sep 17 00:00:00 2001 From: Sid Murching Date: Mon, 3 Nov 2025 08:06:15 -0800 Subject: [PATCH 21/21] Remove HTTP mode and add PR documentation MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Remove app.py and http_mode.py (non-socket mode files) - Update README to focus on Socket Mode only - Add PR_DESCRIPTION.md with high-level context - Update environment variables to include SLACK_APP_TOKEN - Simplify setup instructions šŸ¤– Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude --- PR_DESCRIPTION.md | 78 ++++++++++++++++++++ slack-bot/README.md | 153 +++++++++++++++++++++++---------------- slack-bot/app.py | 51 ------------- slack-bot/http_mode.py | 161 ----------------------------------------- 4 files changed, 169 insertions(+), 274 deletions(-) create mode 100644 PR_DESCRIPTION.md delete mode 100644 slack-bot/app.py delete mode 100644 slack-bot/http_mode.py diff --git a/PR_DESCRIPTION.md b/PR_DESCRIPTION.md new file mode 100644 index 0000000..9e2c0e4 --- /dev/null +++ b/PR_DESCRIPTION.md @@ -0,0 +1,78 @@ +# Add Slack Bot with Glean-like UX and OAuth Authentication + +## Overview + +This PR introduces a Slack bot that provides personalized, authenticated access to Databricks AI agents. The bot implements a Glean-like UX featuring ephemeral messages, sharing capabilities, and multi-user support. + +## Key Features + +- **Glean-like UX**: Bot responds privately to users via ephemeral messages, with options to share responses publicly or dismiss them +- **OAuth Authentication**: Seamless Databricks OAuth flow with automatic response after authentication completes +- **Multi-user Support**: Each user authenticates independently and receives personalized responses +- **Thread-based Conversations**: All interactions happen in threads for clean channel organization +- **Interactive Buttons**: + - "Post in thread" - Share private response publicly + - "Dismiss" - Remove ephemeral message + - "Generate my own answer" - Allow other users to get their own personalized responses +- **Visual Feedback**: Eyes emoji reaction on message receipt, white checkmark when shared +- **Auto-respond After Auth**: Bot stores pending messages and automatically responds after OAuth completes + +## Architecture + +The bot uses Socket Mode for development and consists of: + +- **app_socket.py**: Main Slack app with event handlers and button actions +- **oauth_server.py**: OAuth callback handler running on port 8000 +- **handlers/**: Message processing and authentication logic +- **state/**: In-memory state management (can be replaced with Redis/DynamoDB for production) +- **client/**: Databricks API client for agent calls + +## User Flow + +1. User mentions bot: `@DatabricksBot hello` +2. Bot reacts with šŸ‘€ emoji for immediate acknowledgment +3. **First-time users**: Bot sends ephemeral OAuth prompt in thread +4. **After OAuth or for returning users**: Bot sends "Working on it..." status message +5. Bot sends ephemeral response with "Post in thread" and "Dismiss" buttons (visible only to requesting user) +6. Bot posts public message with "Generate my own answer" button for other users +7. When shared: Ephemeral message disappears, response posted publicly, āœ… reaction added to original message + +## Technical Highlights + +- **Ephemeral Messages**: Uses `chat_postEphemeral` for private responses +- **Response URL Pattern**: Uses `respond(delete_original=True)` for dismissing ephemeral messages +- **Event Deduplication**: Properly handles both `app_mention` and `message` events without duplication +- **PKCE OAuth Flow**: Secure OAuth implementation with proper state management +- **Conversation History**: Per-thread history for context-aware responses +- **Regex Action Patterns**: Dynamic action_id matching for button interactions + +## Testing + +Run the bot locally: +```bash +cd slack-bot +python app_socket.py +``` + +The bot will: +- Connect to Slack via Socket Mode +- Start OAuth callback server on port 8000 +- Log all events and interactions to console + +## Future Enhancements + +- Replace in-memory state storage with Redis/DynamoDB for production +- Implement rate limiting and request validation +- Add token encryption at rest +- Horizontal scaling with HTTP mode and load balancer +- Structured logging with correlation IDs +- Metrics and monitoring + +## Related Documentation + +See `slack-bot/README.md` for: +- Complete setup instructions +- Slack app configuration +- Environment variables +- Development guide +- Troubleshooting tips diff --git a/slack-bot/README.md b/slack-bot/README.md index 6591851..e38ab0a 100644 --- a/slack-bot/README.md +++ b/slack-bot/README.md @@ -1,29 +1,34 @@ # Databricks Slack Bot -A Slack bot that integrates with Databricks agents, providing conversational AI capabilities directly in Slack channels and threads. +A Slack bot that provides personalized, authenticated access to Databricks AI agents with a Glean-like UX featuring ephemeral messages, sharing capabilities, and multi-user support. ## Features -- **OAuth Authentication**: Secure authentication with Databricks on behalf of each user -- **Thread-based Conversations**: Maintains conversation context within Slack threads -- **Ephemeral Messages**: Private authentication prompts visible only to the user +- **OAuth Authentication**: Users authenticate with Databricks once, bot stores tokens securely +- **Auto-respond After Auth**: Bot stores pending messages and automatically responds after OAuth completes +- **Ephemeral Responses**: Private responses visible only to the requesting user with options to share or dismiss +- **Post in Thread**: Share private responses publicly with one click +- **Generate Button**: Other users can generate their own personalized answers +- **Message Acknowledgment**: Bot reacts with :eyes: emoji immediately upon receiving messages +- **Status Updates**: "Working on it" messages that auto-delete after response is ready +- **Thread-based Conversations**: All interactions happen in threads for clean organization - **Tool Call Visualization**: Rich formatting for agent tool calls and results - **Conversation History**: Per-thread conversation history for context-aware responses -- **State Management**: In-memory state storage (extensible to Redis/DynamoDB for production) ## Architecture ``` slack-bot/ -ā”œā”€ā”€ app.py # Main application entry point -ā”œā”€ā”€ config.py # Configuration management +ā”œā”€ā”€ app_socket.py # Main bot application using Socket Mode +ā”œā”€ā”€ oauth_server.py # OAuth callback handler (runs on port 8000) +ā”œā”€ā”€ config.py # Configuration management ā”œā”€ā”€ handlers/ -│ ā”œā”€ā”€ message_handler.py # Message event handling -│ └── auth_handler.py # OAuth flow management -ā”œā”€ā”€ storage/ -│ └── state_manager.py # State and conversation history +│ ā”œā”€ā”€ message_handler.py # Message processing and response logic +│ └── auth_handler.py # OAuth and authentication management +ā”œā”€ā”€ state/ +│ └── state_manager.py # In-memory state management for tokens and history └── client/ - └── databricks_client.py # Databricks API integration + └── databricks_client.py # Databricks API client for agent calls ``` ## Setup @@ -39,7 +44,13 @@ slack-bot/ 1. Go to [api.slack.com/apps](https://api.slack.com/apps) and create a new app 2. Choose "From scratch" and select your workspace -3. Configure **OAuth & Permissions**: + +3. Enable **Socket Mode**: + - Go to **Socket Mode** and enable it + - Create an app-level token with `connections:write` scope + - Save the **App-Level Token** (starts with `xapp-`) + +4. Configure **OAuth & Permissions**: - Add these **Bot Token Scopes**: - `app_mentions:read` - Read mentions - `chat:write` - Send messages @@ -48,9 +59,13 @@ slack-bot/ - `im:history` - Read DM messages - `im:write` - Send DMs - `users:read` - Read user info + - `channels:read` - Read channel info + - `channels:join` - Join channels + - `groups:read` - Read private channel info + - `reactions:write` - Add emoji reactions - Install app to workspace and save the **Bot User OAuth Token** -4. Configure **Event Subscriptions**: +5. Configure **Event Subscriptions**: - Enable events - Subscribe to bot events: - `app_mention` - When bot is mentioned @@ -59,11 +74,6 @@ slack-bot/ - `message.im` - Direct messages - `app_home_opened` - When home tab is opened -5. For development, enable **Socket Mode**: - - Go to **Socket Mode** and enable it - - Create an app-level token with `connections:write` scope - - Save the **App-Level Token** (starts with `xapp-`) - 6. Configure **App Home**: - Enable the Home tab - Enable Messages tab @@ -73,8 +83,9 @@ slack-bot/ Create a `.env` file in the `slack-bot/` directory: ```bash -# Slack Configuration +# Slack Configuration (Socket Mode) SLACK_BOT_TOKEN=xoxb-your-bot-token +SLACK_APP_TOKEN=xapp-your-app-level-token SLACK_SIGNING_SECRET=your-signing-secret SLACK_CLIENT_ID=your-client-id SLACK_CLIENT_SECRET=your-client-secret @@ -84,13 +95,6 @@ DATABRICKS_HOST=https://your-workspace.databricks.com DATABRICKS_CLIENT_ID=your-databricks-oauth-client-id DATABRICKS_CLIENT_SECRET=your-databricks-oauth-client-secret SERVING_ENDPOINT_NAME=your-agent-endpoint-name - -# Optional: State Storage -STATE_STORAGE=memory # or redis, dynamodb -REDIS_URL=redis://localhost:6379 - -# Server Configuration (for HTTP mode) -PORT=3000 ``` ### 4. Install Dependencies @@ -115,21 +119,24 @@ pip install -r requirements.txt ### 6. Run the Bot -**Development (Socket Mode):** ```bash -python app.py +python app_socket.py ``` -**Production (HTTP Mode):** -```bash -# Set up a public endpoint (e.g., using ngrok for testing) -ngrok http 3000 +The bot will: +- Start Socket Mode connection to Slack +- Start OAuth callback server on port 8000 +- Log all events and interactions to console -# Update your Slack app's Event Subscriptions URL to: -# https://your-ngrok-url.ngrok.io/slack/events +**Stop the Bot:** +```bash +pkill -f app_socket.py +``` -# Run the bot -python app.py +**Run in Background with Logs:** +```bash +python app_socket.py > bot.log 2>&1 & +tail -f bot.log ``` ## Usage @@ -138,31 +145,53 @@ python app.py 1. **Invite the bot** to a channel: `/invite @DatabricksBot` -2. **Send a message** in the channel or thread +2. **Mention the bot** in a channel or thread: `@DatabricksBot hello` 3. **First-time authentication**: - - The bot will send you a private (ephemeral) message with a "Sign in to Databricks" button - - Click the button and complete OAuth flow - - Return to Slack and send your message again + - Bot reacts with :eyes: emoji to acknowledge receipt + - Bot sends you a private (ephemeral) login prompt in the thread + - Click "Sign in to Databricks" and complete OAuth flow + - Bot automatically responds to your original message after auth completes -4. **Ask questions**: - ``` - What is the revenue for Q4? - Show me the top 5 customers - Analyze the sales trends - ``` +4. **Using the bot**: + - **Private responses**: Bot sends ephemeral responses visible only to you + - **Post in thread**: Click to share your private response publicly + - **Dismiss**: Click to dismiss the private response + - **Generate**: Other users can click to get their own personalized answer 5. **Commands**: - - `logout` - Sign out from Databricks + - `/databricks-logout` - Sign out from Databricks - `clear` - Clear conversation history for the current thread -### Message Flow - -1. User sends message → Bot checks authentication -2. If not authenticated → Show ephemeral auth prompt -3. If authenticated → Call Databricks agent with conversation history -4. Agent response → Post in thread with formatting -5. Tool calls → Display as rich Slack blocks +### UX Flow + +**First-time User:** +1. User: `@bot write a haiku` +2. Bot: :eyes: reaction (immediate acknowledgment) +3. Bot: Ephemeral login prompt in thread +4. User clicks "Sign in to Databricks" +5. OAuth flow completes +6. Bot: "Hi @user, I'm asking Databricks agents to respond to your message..." +7. Bot: Ephemeral response with "Post in thread" and "Dismiss" buttons +8. Bot: Public message with "Generate my own answer" button for others + +**Returning User:** +1. User: `@bot hello` +2. Bot: :eyes: reaction +3. Bot: "Hi @user, I'm asking Databricks agents to respond to your message..." +4. Bot: Ephemeral response with buttons +5. Bot: Public "Generate" message for others + +**Sharing a Response:** +1. User clicks "Post in thread" on ephemeral message +2. Ephemeral message disappears +3. Bot posts response publicly in thread +4. Bot adds :white_check_mark: reaction to user's original message + +**Other Users Generating Answers:** +1. User clicks "šŸŽÆ Generate my own answer" button +2. Button disappears, replaced with "āœ… @user generated their own answer" +3. User gets their own ephemeral response (no cascading Generate buttons) ## Key Differences from Teams Bot @@ -206,10 +235,9 @@ table = dynamodb.Table('slack-bot-state') ### Scalability -1. **Horizontal Scaling**: Use HTTP mode with load balancer -2. **Background Jobs**: Use Celery/SQS for long-running tasks -3. **Caching**: Cache frequently accessed data -4. **Database**: Use PostgreSQL/MongoDB for persistence +1. **Background Jobs**: Use Celery/SQS for long-running tasks +2. **Caching**: Cache frequently accessed data +3. **Database**: Use PostgreSQL/MongoDB for persistence ### Monitoring @@ -243,12 +271,13 @@ table = dynamodb.Table('slack-bot-state') ### Project Structure -- `app.py` - Main Slack app with event handlers +- `app_socket.py` - Main Slack app with Socket Mode event handlers +- `oauth_server.py` - OAuth callback handler (port 8000) - `config.py` - Environment configuration - `handlers/` - Event handling logic - `message_handler.py` - Process messages and agent calls - `auth_handler.py` - OAuth flow management -- `storage/` - State management +- `state/` - State management - `state_manager.py` - Conversation and user state - `client/` - External integrations - `databricks_client.py` - Databricks API client @@ -257,7 +286,7 @@ table = dynamodb.Table('slack-bot-state') ```bash # Run with test credentials -DATABRICKS_HOST=https://test.databricks.com python app.py +DATABRICKS_HOST=https://test.databricks.com python app_socket.py # Use Slack's test features # Create a test workspace and app diff --git a/slack-bot/app.py b/slack-bot/app.py deleted file mode 100644 index 6dd9ab9..0000000 --- a/slack-bot/app.py +++ /dev/null @@ -1,51 +0,0 @@ -#!/usr/bin/env python3 -""" -Slack Bot Launcher - Auto-detects mode and runs appropriate version. - -Modes: -- Socket Mode (default): For local development, no public URL needed -- HTTP Mode: For production with public HTTPS endpoint - -Usage: - python app.py # Auto-detect based on SLACK_APP_TOKEN - python app_socket.py # Force Socket Mode - python http_mode.py # Force HTTP Mode -""" - -import os -import sys -from pathlib import Path -from dotenv import load_dotenv - -# Load environment variables -env_file = Path(__file__).parent / ".env.local" -if env_file.exists(): - load_dotenv(env_file) - print(f"Loaded environment from: {env_file}") -else: - env_file = Path(__file__).parent / ".env" - if env_file.exists(): - load_dotenv(env_file) - print(f"Loaded environment from: {env_file}") - -# Auto-detect mode based on SLACK_APP_TOKEN -slack_app_token = os.environ.get("SLACK_APP_TOKEN", "") - -if slack_app_token: - print("šŸš€ Socket Mode detected (SLACK_APP_TOKEN found)") - print(" Starting Socket Mode bot...") - print() - from app_socket import main - main() -else: - print("āš ļø HTTP Mode detected (no SLACK_APP_TOKEN)") - print(" HTTP Mode requires public HTTPS URL") - print(" For local development, set SLACK_APP_TOKEN and use Socket Mode") - print() - response = input("Continue with HTTP mode? (y/N): ") - if response.lower() != 'y': - print("Exiting. Set SLACK_APP_TOKEN in .env.local for Socket Mode") - sys.exit(0) - - from http_mode import main - main() diff --git a/slack-bot/http_mode.py b/slack-bot/http_mode.py deleted file mode 100644 index 560ad6e..0000000 --- a/slack-bot/http_mode.py +++ /dev/null @@ -1,161 +0,0 @@ -#!/usr/bin/env python3 -""" -Slack bot using HTTP mode with Events API. - -This version requires: -- Public HTTPS URL for Slack events -- Event Subscriptions configured in Slack app -- For production deployments only - -For local development, use app_socket.py instead. -""" - -import logging -import sys -from pathlib import Path -from dotenv import load_dotenv -from flask import Flask, request -from slack_bolt import App -from slack_bolt.adapter.flask import SlackRequestHandler - -# Load environment variables -env_file = Path(__file__).parent / ".env.local" -if env_file.exists(): - load_dotenv(env_file) -else: - load_dotenv(Path(__file__).parent / ".env") - -from config import DefaultConfig -from storage import StateManager -from handlers import AuthHandler, handle_message_event -from client import DatabricksClient -from oauth_server import create_oauth_server - -# Configure logging -logging.basicConfig( - level=logging.INFO, - format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', - handlers=[logging.StreamHandler(sys.stdout)] -) -logger = logging.getLogger(__name__) - -# Load configuration -CONFIG = DefaultConfig() - -# Validate required configuration -required_configs = [ - "SLACK_BOT_TOKEN", - "SLACK_SIGNING_SECRET", - "DATABRICKS_HOST", - "SERVING_ENDPOINT_NAME" -] -missing_configs = [c for c in required_configs if not getattr(CONFIG, c)] -if missing_configs: - logger.error(f"Missing required configuration: {', '.join(missing_configs)}") - sys.exit(1) - -# Initialize Slack app -app = App( - token=CONFIG.SLACK_BOT_TOKEN, - signing_secret=CONFIG.SLACK_SIGNING_SECRET, -) - -# Initialize components -state_manager = StateManager() -auth_handler = AuthHandler(CONFIG, state_manager) -databricks_client = DatabricksClient( - CONFIG.DATABRICKS_HOST, - CONFIG.SERVING_ENDPOINT_NAME -) - -logger.info("Slack bot initialized successfully") - - -# Event Handlers (same as Socket Mode) - -@app.event("message") -async def message_handler(message, say, client): - """Handle incoming message events""" - await handle_message_event( - message=message, - say=say, - client=client, - state_manager=state_manager, - auth_handler=auth_handler, - databricks_client=databricks_client, - logger=logger - ) - - -@app.event("app_mention") -async def handle_app_mention(event, say, client): - """Handle @mentions of the bot""" - await handle_message_event( - message=event, - say=say, - client=client, - state_manager=state_manager, - auth_handler=auth_handler, - databricks_client=databricks_client, - logger=logger - ) - - -@app.command("/databricks-logout") -async def handle_logout_command(ack, command, respond): - """Handle /databricks-logout slash command""" - await ack() - user_id = command["user_id"] - auth_handler.handle_logout(user_id) - await respond("You have been logged out from Databricks.") - - -@app.error -async def custom_error_handler(error, body, logger): - """Global error handler""" - logger.error(f"Error: {error}") - - -# Create Flask app for HTTP mode -flask_app = Flask(__name__) -slack_handler = SlackRequestHandler(app) - -# OAuth callback server -oauth_app = create_oauth_server(CONFIG, auth_handler, state_manager) -flask_app.register_blueprint(oauth_app) - - -@flask_app.route("/slack/events", methods=["POST"]) -def slack_events(): - """Handle Slack events via HTTP""" - return slack_handler.handle(request) - - -@flask_app.route("/health", methods=["GET"]) -def health_check(): - """Health check endpoint""" - return {"status": "ok"}, 200 - - -def main(): - """Main entry point""" - try: - logger.info(f"Starting bot in HTTP mode on port {CONFIG.PORT}...") - logger.info(f"Slack events endpoint: http://localhost:{CONFIG.PORT}/slack/events") - logger.info(f"OAuth callback: http://localhost:{CONFIG.PORT}/oauth/callback") - logger.warning("āš ļø HTTP mode requires public HTTPS URL for production!") - logger.warning(" For local development, use app_socket.py instead") - - flask_app.run(port=CONFIG.PORT, debug=False) - - except KeyboardInterrupt: - logger.info("Bot stopped by user") - except Exception as e: - logger.error(f"Error starting bot: {e}") - import traceback - traceback.print_exc() - sys.exit(1) - - -if __name__ == "__main__": - main()