Added auth
This commit is contained in:
		
							
								
								
									
										17
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| repos: | ||||
| - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|   rev: v2.3.0 | ||||
|   hooks: | ||||
|   - id: check-yaml | ||||
|   - id: end-of-file-fixer | ||||
|   - id: trailing-whitespace | ||||
| - repo: https://github.com/psf/black | ||||
|   rev: 22.3.0 | ||||
|   hooks: | ||||
|   - id: black | ||||
| - repo: https://github.com/asottile/reorder_python_imports | ||||
|   rev: v3.1.0 | ||||
|   hooks: | ||||
|   - id: reorder-python-imports | ||||
| default_language_version: | ||||
|   python: python3.10 | ||||
							
								
								
									
										0
									
								
								api/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								api/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										14
									
								
								api/database.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								api/database.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| from sqlalchemy import create_engine | ||||
| from sqlalchemy.ext.declarative import declarative_base | ||||
| from sqlalchemy.orm import sessionmaker | ||||
|  | ||||
|  | ||||
| SQLALCHEMY_DATABASE_URL = "sqlite:///./db.sqlite3" | ||||
|  | ||||
| engine = create_engine( | ||||
|     SQLALCHEMY_DATABASE_URL, connect_args={"check_same_thread": False}, future=True | ||||
| ) | ||||
|  | ||||
| SessionLocal = sessionmaker(autocommit=False, autoflush=False, bind=engine, future=True) | ||||
|  | ||||
| Base = declarative_base() | ||||
							
								
								
									
										13
									
								
								api/hasher.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								api/hasher.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| import argon2 | ||||
|  | ||||
|  | ||||
| class Argon2PasswordHasher(argon2.PasswordHasher): | ||||
|     def verify(self, password_hash: str | bytes, password: str | bytes) -> bool: | ||||
|         try: | ||||
|             super().verify(password_hash, password) | ||||
|             return True | ||||
|         except argon2.exceptions.VerifyMismatchError: | ||||
|             return False | ||||
|  | ||||
|  | ||||
| argon2_hasher = Argon2PasswordHasher() | ||||
							
								
								
									
										17
									
								
								api/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								api/models.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| from sqlalchemy import Column | ||||
| from sqlalchemy import Integer | ||||
| from sqlalchemy import VARCHAR | ||||
|  | ||||
| from api.database import Base | ||||
| from api.database import engine | ||||
|  | ||||
|  | ||||
| class UserModel(Base): | ||||
|     __tablename__ = "user" | ||||
|  | ||||
|     id = Column(Integer, primary_key=True, index=True, nullable=False) | ||||
|     name = Column(VARCHAR(length=32), nullable=False, unique=True) | ||||
|     password_hash = Column(VARCHAR, nullable=False) | ||||
|  | ||||
|  | ||||
| Base.metadata.create_all(engine) | ||||
							
								
								
									
										24
									
								
								api/schema/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								api/schema/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| import strawberry | ||||
|  | ||||
| from api.schema.definitions.auth import AuthResult | ||||
| from api.schema.definitions.auth import login | ||||
| from api.schema.definitions.auth import update_me | ||||
| from api.schema.definitions.common import CommonMessage | ||||
| from api.schema.extensions import extensions | ||||
| from api.schema.permissions import IsAuthenticated | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Query: | ||||
|     hello: str | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Mutation: | ||||
|     login: AuthResult = strawberry.field(resolver=login) | ||||
|     update_me: CommonMessage = strawberry.field( | ||||
|         resolver=update_me, permission_classes=[IsAuthenticated] | ||||
|     ) | ||||
|  | ||||
|  | ||||
| schema = strawberry.Schema(query=Query, mutation=Mutation, extensions=extensions) | ||||
							
								
								
									
										83
									
								
								api/schema/definitions/auth.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								api/schema/definitions/auth.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| from typing import TYPE_CHECKING | ||||
|  | ||||
| import strawberry | ||||
| from fastapi import Request | ||||
| from sqlalchemy import true | ||||
| from sqlalchemy.orm import Session | ||||
|  | ||||
| from api.hasher import argon2_hasher | ||||
| from api.models import UserModel | ||||
| from api.schema.definitions.common import CommonError | ||||
| from api.schema.definitions.common import CommonMessage | ||||
| from api.schema.definitions.user import User | ||||
| from api.token import decode_user_token | ||||
| from api.token import encode_user_token | ||||
| from api.token import token_from_headers | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from strawberry.types import Info | ||||
|     from api.schema import Query | ||||
|  | ||||
|  | ||||
| @strawberry.input | ||||
| class LoginInput: | ||||
|     name: str | ||||
|     password: str | ||||
|  | ||||
|  | ||||
| @strawberry.input | ||||
| class UpdateUserInput: | ||||
|     name: str | ||||
|     password: str | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class AuthSuccess: | ||||
|     user: User | ||||
|     token: str | ||||
|  | ||||
|  | ||||
| AuthResult = strawberry.union("AuthResult", (AuthSuccess, CommonError)) | ||||
| LogoutResult = strawberry.union("LogoutResult", (CommonMessage, CommonError)) | ||||
|  | ||||
|  | ||||
| async def login(root: "Query", info: "Info", body: LoginInput) -> AuthResult: | ||||
|     db: Session = info.context["db"] | ||||
|  | ||||
|     stmt = ( | ||||
|         db.query(UserModel.id, UserModel.name, UserModel.password_hash) | ||||
|         .filter(UserModel.name == body.name) | ||||
|         .limit(1) | ||||
|     ) | ||||
|  | ||||
|     user: UserModel | None = db.execute(stmt).first()  # type: ignore | ||||
|     if not user: | ||||
|         return CommonError(message="Invalid credentials") | ||||
|  | ||||
|     if not argon2_hasher.verify(user.password_hash, body.password): | ||||
|         return CommonError(message="Invalid credentials") | ||||
|  | ||||
|     if argon2_hasher.check_needs_rehash(user.password_hash): | ||||
|         user.password_hash = argon2_hasher.hash(body.password) | ||||
|         db.commit() | ||||
|  | ||||
|     return AuthSuccess(user=User.from_instance(user), token=encode_user_token(user)) | ||||
|  | ||||
|  | ||||
| async def update_me( | ||||
|     root: "Query", info: "Info", body: UpdateUserInput | ||||
| ) -> CommonMessage: | ||||
|     db: Session = info.context["db"] | ||||
|     req: Request = info.context["request"] | ||||
|  | ||||
|     _, auth_token = token_from_headers(req.headers) | ||||
|     token = decode_user_token(auth_token) | ||||
|  | ||||
|     updated_user = { | ||||
|         UserModel.name: body.name, | ||||
|         UserModel.password_hash: argon2_hasher.hash(body.password), | ||||
|     } | ||||
|     db.query(UserModel).filter(UserModel.id == token["id"]).update(updated_user) | ||||
|     db.commit() | ||||
|  | ||||
|     return CommonMessage(message="Succesfully updated credentials") | ||||
							
								
								
									
										11
									
								
								api/schema/definitions/common.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								api/schema/definitions/common.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| import strawberry | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class CommonError: | ||||
|     message: str | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class CommonMessage: | ||||
|     message: str | ||||
							
								
								
									
										13
									
								
								api/schema/definitions/user.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								api/schema/definitions/user.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| import strawberry | ||||
|  | ||||
| from api.models import UserModel | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class User: | ||||
|     id: int | ||||
|     name: str | ||||
|  | ||||
|     @classmethod | ||||
|     def from_instance(cls, instance: UserModel): | ||||
|         return cls(id=instance.id, name=instance.name) | ||||
							
								
								
									
										14
									
								
								api/schema/extensions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								api/schema/extensions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| from strawberry.extensions import Extension | ||||
|  | ||||
| from api.database import SessionLocal | ||||
|  | ||||
|  | ||||
| class SQLAlchemySession(Extension): | ||||
|     def on_request_start(self): | ||||
|         self.execution_context.context["db"] = SessionLocal() | ||||
|  | ||||
|     def on_request_end(self): | ||||
|         self.execution_context.context["db"].close() | ||||
|  | ||||
|  | ||||
| extensions = (SQLAlchemySession,) | ||||
							
								
								
									
										27
									
								
								api/schema/permissions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								api/schema/permissions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| import typing | ||||
|  | ||||
| from fastapi import Request | ||||
| from strawberry.permission import BasePermission | ||||
|  | ||||
| from api.token import decode_user_token | ||||
| from api.token import token_from_headers | ||||
|  | ||||
| if typing.TYPE_CHECKING: | ||||
|     from strawberry.types import Info | ||||
|  | ||||
|  | ||||
| class IsAuthenticated(BasePermission): | ||||
|     message = "User is not authenticated" | ||||
|  | ||||
|     async def has_permission(self, source: typing.Any, info: "Info", **kwargs) -> bool: | ||||
|         req: Request = info.context["request"] | ||||
|         _, auth_token = token_from_headers(req.headers) | ||||
|  | ||||
|         if len(auth_token) == 0: | ||||
|             return False | ||||
|  | ||||
|         try: | ||||
|             decode_user_token(auth_token) | ||||
|             return True | ||||
|         except Exception: | ||||
|             return False | ||||
							
								
								
									
										14
									
								
								api/seed.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								api/seed.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| from sqlalchemy.orm import Session | ||||
|  | ||||
| from api.database import SessionLocal | ||||
| from api.hasher import argon2_hasher | ||||
| from api.models import UserModel | ||||
|  | ||||
|  | ||||
| def seed(): | ||||
|     db: Session = SessionLocal() | ||||
|  | ||||
|     if db.query(UserModel).count() == 0: | ||||
|         admin = UserModel(name="admin", password_hash=argon2_hasher.hash("admin")) | ||||
|         db.add(admin) | ||||
|         db.commit() | ||||
							
								
								
									
										35
									
								
								api/token.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								api/token.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| import os | ||||
| import typing | ||||
| from datetime import datetime | ||||
| from datetime import timedelta | ||||
| from datetime import timezone | ||||
|  | ||||
| import jwt | ||||
| from fastapi import Request | ||||
| from starlette.datastructures import Headers | ||||
|  | ||||
| from api.models import UserModel | ||||
|  | ||||
| if typing.TYPE_CHECKING: | ||||
|     UserTokenData = dict[str, typing.Any] | ||||
|  | ||||
| JWT_SECRET = os.getenv("JWT_SECRET", "") | ||||
|  | ||||
|  | ||||
| def encode_user_token( | ||||
|     user: UserModel, expire_in: timedelta = timedelta(hours=6) | ||||
| ) -> str: | ||||
|     payload = {} | ||||
|     payload["id"] = user.id | ||||
|     payload["exp"] = datetime.now(tz=timezone.utc) + expire_in | ||||
|     return jwt.encode(payload, JWT_SECRET) | ||||
|  | ||||
|  | ||||
| def decode_user_token(token: str) -> "UserTokenData": | ||||
|     return jwt.decode(token, JWT_SECRET, ["HS256"]) | ||||
|  | ||||
|  | ||||
| def token_from_headers(headers: Headers) -> tuple[str, str]: | ||||
|     auth_header: str = headers.get("authorization", "") | ||||
|     scheme, _, auth_token = auth_header.partition(" ") | ||||
|     return scheme, auth_token | ||||
							
								
								
									
										14
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								main.py
									
									
									
									
									
								
							| @@ -1,11 +1,17 @@ | ||||
| from fastapi import FastAPI | ||||
| from models import schema | ||||
| from strawberry.fastapi import GraphQLRouter | ||||
| from fastapi.staticfiles import StaticFiles | ||||
|  | ||||
| graphql_app = GraphQLRouter(schema=schema) | ||||
| from api.schema import schema | ||||
| from api.seed import seed | ||||
| from routers.frontend import SPAStaticFiles | ||||
|  | ||||
| seed() | ||||
|  | ||||
| graphql_app = GraphQLRouter( | ||||
|     schema=schema, | ||||
| ) | ||||
|  | ||||
| app = FastAPI() | ||||
|  | ||||
| app.include_router(graphql_app, prefix="/graphql") | ||||
| app.mount("/", StaticFiles(directory="frontend/dist"), name="frontend") | ||||
| app.mount("/", SPAStaticFiles(directory="frontend/dist", html=True), name="frontend") | ||||
|   | ||||
							
								
								
									
										53
									
								
								models.py
									
									
									
									
									
								
							
							
						
						
									
										53
									
								
								models.py
									
									
									
									
									
								
							| @@ -1,53 +0,0 @@ | ||||
| import strawberry | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class RouterTlsConfig: | ||||
|     cert_resolver: str | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Router: | ||||
|     id: str | ||||
|     entry_points: list[str] | ||||
|     service: str | ||||
|     rule: str | ||||
|     tls_config: "RouterTlsConfig" | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Server: | ||||
|     url: str | ||||
|     port: str | ||||
|     scheme: str | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class LoadBalancer: | ||||
|     servers: list["Server"] | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Service: | ||||
|     id: str | ||||
|     load_balancer: "LoadBalancer" | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class HttpConfiguration: | ||||
|     routers: list["Router"] | ||||
|     services: list["Service"] | ||||
|  | ||||
|  | ||||
| def get_http_configuration() -> "HttpConfiguration": | ||||
|     return HttpConfiguration(routers=[], services=[]) | ||||
|  | ||||
|  | ||||
| @strawberry.type | ||||
| class Query: | ||||
|     http_configuration: "HttpConfiguration" = strawberry.field( | ||||
|         resolver=get_http_configuration | ||||
|     ) | ||||
|  | ||||
|  | ||||
| schema = strawberry.Schema(query=Query) | ||||
							
								
								
									
										505
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										505
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							| @@ -15,6 +15,37 @@ doc = ["packaging", "sphinx-rtd-theme", "sphinx-autodoc-typehints (>=1.2.0)"] | ||||
| test = ["coverage[toml] (>=4.5)", "hypothesis (>=4.0)", "pytest (>=6.0)", "pytest-mock (>=3.6.1)", "trustme", "contextlib2", "uvloop (<0.15)", "mock (>=4)", "uvloop (>=0.15)"] | ||||
| trio = ["trio (>=0.16)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "argon2-cffi" | ||||
| version = "21.3.0" | ||||
| description = "The secure Argon2 password hashing algorithm." | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6" | ||||
|  | ||||
| [package.dependencies] | ||||
| argon2-cffi-bindings = "*" | ||||
|  | ||||
| [package.extras] | ||||
| dev = ["pre-commit", "cogapp", "tomli", "coverage[toml] (>=5.0.2)", "hypothesis", "pytest", "sphinx", "sphinx-notfound-page", "furo"] | ||||
| docs = ["sphinx", "sphinx-notfound-page", "furo"] | ||||
| tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pytest"] | ||||
|  | ||||
| [[package]] | ||||
| name = "argon2-cffi-bindings" | ||||
| version = "21.2.0" | ||||
| description = "Low-level CFFI bindings for Argon2" | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6" | ||||
|  | ||||
| [package.dependencies] | ||||
| cffi = ">=1.0.1" | ||||
|  | ||||
| [package.extras] | ||||
| dev = ["pytest", "cogapp", "pre-commit", "wheel"] | ||||
| tests = ["pytest"] | ||||
|  | ||||
| [[package]] | ||||
| name = "asgiref" | ||||
| version = "3.5.1" | ||||
| @@ -34,6 +65,46 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6.0" | ||||
|  | ||||
| [[package]] | ||||
| name = "black" | ||||
| version = "22.3.0" | ||||
| description = "The uncompromising code formatter." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.6.2" | ||||
|  | ||||
| [package.dependencies] | ||||
| click = ">=8.0.0" | ||||
| mypy-extensions = ">=0.4.3" | ||||
| pathspec = ">=0.9.0" | ||||
| platformdirs = ">=2" | ||||
| tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} | ||||
|  | ||||
| [package.extras] | ||||
| colorama = ["colorama (>=0.4.3)"] | ||||
| d = ["aiohttp (>=3.7.4)"] | ||||
| jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] | ||||
| uvloop = ["uvloop (>=0.15.2)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "cffi" | ||||
| version = "1.15.0" | ||||
| description = "Foreign Function Interface for Python calling C code." | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = "*" | ||||
|  | ||||
| [package.dependencies] | ||||
| pycparser = "*" | ||||
|  | ||||
| [[package]] | ||||
| name = "cfgv" | ||||
| version = "3.3.1" | ||||
| description = "Validate configuration and produce human readable error messages." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.6.1" | ||||
|  | ||||
| [[package]] | ||||
| name = "click" | ||||
| version = "8.1.3" | ||||
| @@ -53,6 +124,14 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" | ||||
|  | ||||
| [[package]] | ||||
| name = "distlib" | ||||
| version = "0.3.4" | ||||
| description = "Distribution utilities" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = "*" | ||||
|  | ||||
| [[package]] | ||||
| name = "fastapi" | ||||
| version = "0.77.1" | ||||
| @@ -71,6 +150,18 @@ dev = ["python-jose[cryptography] (>=3.3.0,<4.0.0)", "passlib[bcrypt] (>=1.7.2,< | ||||
| doc = ["mkdocs (>=1.1.2,<2.0.0)", "mkdocs-material (>=8.1.4,<9.0.0)", "mdx-include (>=1.4.1,<2.0.0)", "mkdocs-markdownextradata-plugin (>=0.1.7,<0.3.0)", "typer (>=0.4.1,<0.5.0)", "pyyaml (>=5.3.1,<7.0.0)"] | ||||
| test = ["pytest (>=6.2.4,<7.0.0)", "pytest-cov (>=2.12.0,<4.0.0)", "mypy (==0.910)", "flake8 (>=3.8.3,<4.0.0)", "black (==22.3.0)", "isort (>=5.0.6,<6.0.0)", "requests (>=2.24.0,<3.0.0)", "httpx (>=0.14.0,<0.19.0)", "email_validator (>=1.1.1,<2.0.0)", "sqlalchemy (>=1.3.18,<1.5.0)", "peewee (>=3.13.3,<4.0.0)", "databases[sqlite] (>=0.3.2,<0.6.0)", "orjson (>=3.2.1,<4.0.0)", "ujson (>=4.0.1,!=4.0.2,!=4.1.0,!=4.2.0,!=4.3.0,!=5.0.0,!=5.1.0,<6.0.0)", "python-multipart (>=0.0.5,<0.0.6)", "flask (>=1.1.2,<3.0.0)", "anyio[trio] (>=3.2.1,<4.0.0)", "types-ujson (==4.2.1)", "types-orjson (==3.6.2)", "types-dataclasses (==0.6.5)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "filelock" | ||||
| version = "3.7.0" | ||||
| description = "A platform independent file lock." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.7" | ||||
|  | ||||
| [package.extras] | ||||
| docs = ["furo (>=2021.8.17b43)", "sphinx (>=4.1)", "sphinx-autodoc-typehints (>=1.12)"] | ||||
| testing = ["covdefaults (>=1.2.0)", "coverage (>=4)", "pytest (>=4)", "pytest-cov", "pytest-timeout (>=1.4.2)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "graphql-core" | ||||
| version = "3.2.1" | ||||
| @@ -79,6 +170,17 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6,<4" | ||||
|  | ||||
| [[package]] | ||||
| name = "greenlet" | ||||
| version = "1.1.2" | ||||
| description = "Lightweight in-process concurrent programming" | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" | ||||
|  | ||||
| [package.extras] | ||||
| docs = ["sphinx"] | ||||
|  | ||||
| [[package]] | ||||
| name = "h11" | ||||
| version = "0.13.0" | ||||
| @@ -98,6 +200,17 @@ python-versions = ">=3.5.0" | ||||
| [package.extras] | ||||
| test = ["Cython (>=0.29.24,<0.30.0)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "identify" | ||||
| version = "2.5.1" | ||||
| description = "File identification library for Python" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.7" | ||||
|  | ||||
| [package.extras] | ||||
| license = ["ukkonen"] | ||||
|  | ||||
| [[package]] | ||||
| name = "idna" | ||||
| version = "3.3" | ||||
| @@ -106,6 +219,66 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.5" | ||||
|  | ||||
| [[package]] | ||||
| name = "mypy-extensions" | ||||
| version = "0.4.3" | ||||
| description = "Experimental type system extensions for programs checked with the mypy typechecker." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = "*" | ||||
|  | ||||
| [[package]] | ||||
| name = "nodeenv" | ||||
| version = "1.6.0" | ||||
| description = "Node.js virtual environment builder" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = "*" | ||||
|  | ||||
| [[package]] | ||||
| name = "pathspec" | ||||
| version = "0.9.0" | ||||
| description = "Utility library for gitignore style pattern matching of file paths." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" | ||||
|  | ||||
| [[package]] | ||||
| name = "platformdirs" | ||||
| version = "2.5.2" | ||||
| description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.7" | ||||
|  | ||||
| [package.extras] | ||||
| docs = ["furo (>=2021.7.5b38)", "proselint (>=0.10.2)", "sphinx-autodoc-typehints (>=1.12)", "sphinx (>=4)"] | ||||
| test = ["appdirs (==1.4.4)", "pytest-cov (>=2.7)", "pytest-mock (>=3.6)", "pytest (>=6)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "pre-commit" | ||||
| version = "2.19.0" | ||||
| description = "A framework for managing and maintaining multi-language pre-commit hooks." | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.7" | ||||
|  | ||||
| [package.dependencies] | ||||
| cfgv = ">=2.0.0" | ||||
| identify = ">=1.0.0" | ||||
| nodeenv = ">=0.11.1" | ||||
| pyyaml = ">=5.1" | ||||
| toml = "*" | ||||
| virtualenv = ">=20.0.8" | ||||
|  | ||||
| [[package]] | ||||
| name = "pycparser" | ||||
| version = "2.21" | ||||
| description = "C parser in Python" | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" | ||||
|  | ||||
| [[package]] | ||||
| name = "pydantic" | ||||
| version = "1.9.0" | ||||
| @@ -129,6 +302,20 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6" | ||||
|  | ||||
| [[package]] | ||||
| name = "pyjwt" | ||||
| version = "2.4.0" | ||||
| description = "JSON Web Token implementation in Python" | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.6" | ||||
|  | ||||
| [package.extras] | ||||
| crypto = ["cryptography (>=3.3.1)"] | ||||
| dev = ["sphinx", "sphinx-rtd-theme", "zope.interface", "cryptography (>=3.3.1)", "pytest (>=6.0.0,<7.0.0)", "coverage[toml] (==5.0.4)", "mypy", "pre-commit"] | ||||
| docs = ["sphinx", "sphinx-rtd-theme", "zope.interface"] | ||||
| tests = ["pytest (>=6.0.0,<7.0.0)", "coverage[toml] (==5.0.4)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "python-dateutil" | ||||
| version = "2.8.2" | ||||
| @@ -186,6 +373,38 @@ category = "main" | ||||
| optional = false | ||||
| python-versions = ">=3.5" | ||||
|  | ||||
| [[package]] | ||||
| name = "sqlalchemy" | ||||
| version = "1.4.36" | ||||
| description = "Database Abstraction Library" | ||||
| category = "main" | ||||
| optional = false | ||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,>=2.7" | ||||
|  | ||||
| [package.dependencies] | ||||
| greenlet = {version = "!=0.4.17", markers = "python_version >= \"3\" and (platform_machine == \"aarch64\" or platform_machine == \"ppc64le\" or platform_machine == \"x86_64\" or platform_machine == \"amd64\" or platform_machine == \"AMD64\" or platform_machine == \"win32\" or platform_machine == \"WIN32\")"} | ||||
|  | ||||
| [package.extras] | ||||
| aiomysql = ["greenlet (!=0.4.17)", "aiomysql"] | ||||
| aiosqlite = ["typing_extensions (!=3.10.0.1)", "greenlet (!=0.4.17)", "aiosqlite"] | ||||
| asyncio = ["greenlet (!=0.4.17)"] | ||||
| asyncmy = ["greenlet (!=0.4.17)", "asyncmy (>=0.2.3,!=0.2.4)"] | ||||
| mariadb_connector = ["mariadb (>=1.0.1)"] | ||||
| mssql = ["pyodbc"] | ||||
| mssql_pymssql = ["pymssql"] | ||||
| mssql_pyodbc = ["pyodbc"] | ||||
| mypy = ["sqlalchemy2-stubs", "mypy (>=0.910)"] | ||||
| mysql = ["mysqlclient (>=1.4.0,<2)", "mysqlclient (>=1.4.0)"] | ||||
| mysql_connector = ["mysql-connector-python"] | ||||
| oracle = ["cx_oracle (>=7,<8)", "cx_oracle (>=7)"] | ||||
| postgresql = ["psycopg2 (>=2.7)"] | ||||
| postgresql_asyncpg = ["greenlet (!=0.4.17)", "asyncpg"] | ||||
| postgresql_pg8000 = ["pg8000 (>=1.16.6)"] | ||||
| postgresql_psycopg2binary = ["psycopg2-binary"] | ||||
| postgresql_psycopg2cffi = ["psycopg2cffi"] | ||||
| pymysql = ["pymysql (<1)", "pymysql"] | ||||
| sqlcipher = ["sqlcipher3-binary"] | ||||
|  | ||||
| [[package]] | ||||
| name = "starlette" | ||||
| version = "0.19.1" | ||||
| @@ -229,6 +448,22 @@ sanic = ["sanic (>=20.12.2,<22.0.0)"] | ||||
| aiohttp = ["aiohttp (>=3.7.4.post0,<4.0.0)"] | ||||
| fastapi = ["fastapi (>=0.65.2)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "toml" | ||||
| version = "0.10.2" | ||||
| description = "Python Library for Tom's Obvious, Minimal Language" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" | ||||
|  | ||||
| [[package]] | ||||
| name = "tomli" | ||||
| version = "2.0.1" | ||||
| description = "A lil' TOML parser" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = ">=3.7" | ||||
|  | ||||
| [[package]] | ||||
| name = "typing-extensions" | ||||
| version = "4.2.0" | ||||
| @@ -273,6 +508,24 @@ dev = ["Cython (>=0.29.24,<0.30.0)", "pytest (>=3.6.0)", "Sphinx (>=4.1.2,<4.2.0 | ||||
| docs = ["Sphinx (>=4.1.2,<4.2.0)", "sphinxcontrib-asyncio (>=0.3.0,<0.4.0)", "sphinx-rtd-theme (>=0.5.2,<0.6.0)"] | ||||
| test = ["aiohttp", "flake8 (>=3.9.2,<3.10.0)", "psutil", "pycodestyle (>=2.7.0,<2.8.0)", "pyOpenSSL (>=19.0.0,<19.1.0)", "mypy (>=0.800)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "virtualenv" | ||||
| version = "20.14.1" | ||||
| description = "Virtual Python Environment builder" | ||||
| category = "dev" | ||||
| optional = false | ||||
| python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" | ||||
|  | ||||
| [package.dependencies] | ||||
| distlib = ">=0.3.1,<1" | ||||
| filelock = ">=3.2,<4" | ||||
| platformdirs = ">=2,<3" | ||||
| six = ">=1.9.0,<2" | ||||
|  | ||||
| [package.extras] | ||||
| docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=21.3)"] | ||||
| testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)"] | ||||
|  | ||||
| [[package]] | ||||
| name = "watchgod" | ||||
| version = "0.8.2" | ||||
| @@ -295,13 +548,40 @@ python-versions = ">=3.7" | ||||
| [metadata] | ||||
| lock-version = "1.1" | ||||
| python-versions = "^3.10" | ||||
| content-hash = "71dea8a7a300caa90736280cbab1ee3490bb574c911c51e0fc81af5ab5e012e3" | ||||
| content-hash = "a162f457a182fac2f738e35556edf5a7e82a416d2e8a9cbac60107ca5a1333b4" | ||||
|  | ||||
| [metadata.files] | ||||
| anyio = [ | ||||
|     {file = "anyio-3.5.0-py3-none-any.whl", hash = "sha256:b5fa16c5ff93fa1046f2eeb5bbff2dad4d3514d6cda61d02816dba34fa8c3c2e"}, | ||||
|     {file = "anyio-3.5.0.tar.gz", hash = "sha256:a0aeffe2fb1fdf374a8e4b471444f0f3ac4fb9f5a5b542b48824475e0042a5a6"}, | ||||
| ] | ||||
| argon2-cffi = [ | ||||
|     {file = "argon2-cffi-21.3.0.tar.gz", hash = "sha256:d384164d944190a7dd7ef22c6aa3ff197da12962bd04b17f64d4e93d934dba5b"}, | ||||
|     {file = "argon2_cffi-21.3.0-py3-none-any.whl", hash = "sha256:8c976986f2c5c0e5000919e6de187906cfd81fb1c72bf9d88c01177e77da7f80"}, | ||||
| ] | ||||
| argon2-cffi-bindings = [ | ||||
|     {file = "argon2-cffi-bindings-21.2.0.tar.gz", hash = "sha256:bb89ceffa6c791807d1305ceb77dbfacc5aa499891d2c55661c6459651fc39e3"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-macosx_10_9_x86_64.whl", hash = "sha256:ccb949252cb2ab3a08c02024acb77cfb179492d5701c7cbdbfd776124d4d2367"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9524464572e12979364b7d600abf96181d3541da11e23ddf565a32e70bd4dc0d"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b746dba803a79238e925d9046a63aa26bf86ab2a2fe74ce6b009a1c3f5c8f2ae"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:58ed19212051f49a523abb1dbe954337dc82d947fb6e5a0da60f7c8471a8476c"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-musllinux_1_1_aarch64.whl", hash = "sha256:bd46088725ef7f58b5a1ef7ca06647ebaf0eb4baff7d1d0d177c6cc8744abd86"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-musllinux_1_1_i686.whl", hash = "sha256:8cd69c07dd875537a824deec19f978e0f2078fdda07fd5c42ac29668dda5f40f"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-musllinux_1_1_x86_64.whl", hash = "sha256:f1152ac548bd5b8bcecfb0b0371f082037e47128653df2e8ba6e914d384f3c3e"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-win32.whl", hash = "sha256:603ca0aba86b1349b147cab91ae970c63118a0f30444d4bc80355937c950c082"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp36-abi3-win_amd64.whl", hash = "sha256:b2ef1c30440dbbcba7a5dc3e319408b59676e2e039e2ae11a8775ecf482b192f"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-cp38-abi3-macosx_10_9_universal2.whl", hash = "sha256:e415e3f62c8d124ee16018e491a009937f8cf7ebf5eb430ffc5de21b900dad93"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:3e385d1c39c520c08b53d63300c3ecc28622f076f4c2b0e6d7e796e9f6502194"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2c3e3cc67fdb7d82c4718f19b4e7a87123caf8a93fde7e23cf66ac0337d3cb3f"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6a22ad9800121b71099d0fb0a65323810a15f2e292f2ba450810a7316e128ee5"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp37-pypy37_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f9f8b450ed0547e3d473fdc8612083fd08dd2120d6ac8f73828df9b7d45bb351"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:93f9bf70084f97245ba10ee36575f0c3f1e7d7724d67d8e5b08e61787c320ed7"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:3b9ef65804859d335dc6b31582cad2c5166f0c3e7975f324d9ffaa34ee7e6583"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d4966ef5848d820776f5f562a7d45fdd70c2f330c961d0d745b784034bd9f48d"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:20ef543a89dee4db46a1a6e206cd015360e5a75822f76df533845c3cbaf72670"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ed2937d286e2ad0cc79a7087d3c272832865f779430e0cc2b4f3718d3159b0cb"}, | ||||
|     {file = "argon2_cffi_bindings-21.2.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:5e00316dabdaea0b2dd82d141cc66889ced0cdcbfa599e8b471cf22c620c329a"}, | ||||
| ] | ||||
| asgiref = [ | ||||
|     {file = "asgiref-3.5.1-py3-none-any.whl", hash = "sha256:45a429524fba18aba9d512498b19d220c4d628e75b40cf5c627524dbaebc5cc1"}, | ||||
|     {file = "asgiref-3.5.1.tar.gz", hash = "sha256:fddeea3c53fa99d0cdb613c3941cc6e52d822491fc2753fba25768fb5bf4e865"}, | ||||
| @@ -310,6 +590,87 @@ asgiref = [ | ||||
|     {file = "backports.cached-property-1.0.1.tar.gz", hash = "sha256:1a5ef1e750f8bc7d0204c807aae8e0f450c655be0cf4b30407a35fd4bb27186c"}, | ||||
|     {file = "backports.cached_property-1.0.1-py3-none-any.whl", hash = "sha256:687b5fe14be40aadcf547cae91337a1fdb84026046a39370274e54d3fe4fb4f9"}, | ||||
| ] | ||||
| black = [ | ||||
|     {file = "black-22.3.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:2497f9c2386572e28921fa8bec7be3e51de6801f7459dffd6e62492531c47e09"}, | ||||
|     {file = "black-22.3.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:5795a0375eb87bfe902e80e0c8cfaedf8af4d49694d69161e5bd3206c18618bb"}, | ||||
|     {file = "black-22.3.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:e3556168e2e5c49629f7b0f377070240bd5511e45e25a4497bb0073d9dda776a"}, | ||||
|     {file = "black-22.3.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:67c8301ec94e3bcc8906740fe071391bce40a862b7be0b86fb5382beefecd968"}, | ||||
|     {file = "black-22.3.0-cp310-cp310-win_amd64.whl", hash = "sha256:fd57160949179ec517d32ac2ac898b5f20d68ed1a9c977346efbac9c2f1e779d"}, | ||||
|     {file = "black-22.3.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:cc1e1de68c8e5444e8f94c3670bb48a2beef0e91dddfd4fcc29595ebd90bb9ce"}, | ||||
|     {file = "black-22.3.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d2fc92002d44746d3e7db7cf9313cf4452f43e9ea77a2c939defce3b10b5c82"}, | ||||
|     {file = "black-22.3.0-cp36-cp36m-win_amd64.whl", hash = "sha256:a6342964b43a99dbc72f72812bf88cad8f0217ae9acb47c0d4f141a6416d2d7b"}, | ||||
|     {file = "black-22.3.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:328efc0cc70ccb23429d6be184a15ce613f676bdfc85e5fe8ea2a9354b4e9015"}, | ||||
|     {file = "black-22.3.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:06f9d8846f2340dfac80ceb20200ea5d1b3f181dd0556b47af4e8e0b24fa0a6b"}, | ||||
|     {file = "black-22.3.0-cp37-cp37m-win_amd64.whl", hash = "sha256:ad4efa5fad66b903b4a5f96d91461d90b9507a812b3c5de657d544215bb7877a"}, | ||||
|     {file = "black-22.3.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:e8477ec6bbfe0312c128e74644ac8a02ca06bcdb8982d4ee06f209be28cdf163"}, | ||||
|     {file = "black-22.3.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:637a4014c63fbf42a692d22b55d8ad6968a946b4a6ebc385c5505d9625b6a464"}, | ||||
|     {file = "black-22.3.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:863714200ada56cbc366dc9ae5291ceb936573155f8bf8e9de92aef51f3ad0f0"}, | ||||
|     {file = "black-22.3.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:10dbe6e6d2988049b4655b2b739f98785a884d4d6b85bc35133a8fb9a2233176"}, | ||||
|     {file = "black-22.3.0-cp38-cp38-win_amd64.whl", hash = "sha256:cee3e11161dde1b2a33a904b850b0899e0424cc331b7295f2a9698e79f9a69a0"}, | ||||
|     {file = "black-22.3.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:5891ef8abc06576985de8fa88e95ab70641de6c1fca97e2a15820a9b69e51b20"}, | ||||
|     {file = "black-22.3.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:30d78ba6bf080eeaf0b7b875d924b15cd46fec5fd044ddfbad38c8ea9171043a"}, | ||||
|     {file = "black-22.3.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ee8f1f7228cce7dffc2b464f07ce769f478968bfb3dd1254a4c2eeed84928aad"}, | ||||
|     {file = "black-22.3.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6ee227b696ca60dd1c507be80a6bc849a5a6ab57ac7352aad1ffec9e8b805f21"}, | ||||
|     {file = "black-22.3.0-cp39-cp39-win_amd64.whl", hash = "sha256:9b542ced1ec0ceeff5b37d69838106a6348e60db7b8fdd245294dc1d26136265"}, | ||||
|     {file = "black-22.3.0-py3-none-any.whl", hash = "sha256:bc58025940a896d7e5356952228b68f793cf5fcb342be703c3a2669a1488cb72"}, | ||||
|     {file = "black-22.3.0.tar.gz", hash = "sha256:35020b8886c022ced9282b51b5a875b6d1ab0c387b31a065b84db7c33085ca79"}, | ||||
| ] | ||||
| cffi = [ | ||||
|     {file = "cffi-1.15.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:c2502a1a03b6312837279c8c1bd3ebedf6c12c4228ddbad40912d671ccc8a962"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:23cfe892bd5dd8941608f93348c0737e369e51c100d03718f108bf1add7bd6d0"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:41d45de54cd277a7878919867c0f08b0cf817605e4eb94093e7516505d3c8d14"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27m-win32.whl", hash = "sha256:4a306fa632e8f0928956a41fa8e1d6243c71e7eb59ffbd165fc0b41e316b2474"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27m-win_amd64.whl", hash = "sha256:e7022a66d9b55e93e1a845d8c9eba2a1bebd4966cd8bfc25d9cd07d515b33fa6"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:14cd121ea63ecdae71efa69c15c5543a4b5fbcd0bbe2aad864baca0063cecf27"}, | ||||
|     {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:d4d692a89c5cf08a8557fdeb329b82e7bf609aadfaed6c0d79f5a449a3c7c023"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:0104fb5ae2391d46a4cb082abdd5c69ea4eab79d8d44eaaf79f1b1fd806ee4c2"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:91ec59c33514b7c7559a6acda53bbfe1b283949c34fe7440bcf917f96ac0723e"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f5c7150ad32ba43a07c4479f40241756145a1f03b43480e058cfd862bf5041c7"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:00c878c90cb53ccfaae6b8bc18ad05d2036553e6d9d1d9dbcf323bbe83854ca3"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:abb9a20a72ac4e0fdb50dae135ba5e77880518e742077ced47eb1499e29a443c"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a5263e363c27b653a90078143adb3d076c1a748ec9ecc78ea2fb916f9b861962"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f54a64f8b0c8ff0b64d18aa76675262e1700f3995182267998c31ae974fbc382"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-win32.whl", hash = "sha256:c21c9e3896c23007803a875460fb786118f0cdd4434359577ea25eb556e34c55"}, | ||||
|     {file = "cffi-1.15.0-cp310-cp310-win_amd64.whl", hash = "sha256:5e069f72d497312b24fcc02073d70cb989045d1c91cbd53979366077959933e0"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:64d4ec9f448dfe041705426000cc13e34e6e5bb13736e9fd62e34a0b0c41566e"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2756c88cbb94231c7a147402476be2c4df2f6078099a6f4a480d239a8817ae39"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b96a311ac60a3f6be21d2572e46ce67f09abcf4d09344c49274eb9e0bf345fc"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:75e4024375654472cc27e91cbe9eaa08567f7fbdf822638be2814ce059f58032"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:59888172256cac5629e60e72e86598027aca6bf01fa2465bdb676d37636573e8"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:27c219baf94952ae9d50ec19651a687b826792055353d07648a5695413e0c605"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-win32.whl", hash = "sha256:4958391dbd6249d7ad855b9ca88fae690783a6be9e86df65865058ed81fc860e"}, | ||||
|     {file = "cffi-1.15.0-cp36-cp36m-win_amd64.whl", hash = "sha256:f6f824dc3bce0edab5f427efcfb1d63ee75b6fcb7282900ccaf925be84efb0fc"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:06c48159c1abed75c2e721b1715c379fa3200c7784271b3c46df01383b593636"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c2051981a968d7de9dd2d7b87bcb9c939c74a34626a6e2f8181455dd49ed69e4"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:fd8a250edc26254fe5b33be00402e6d287f562b6a5b2152dec302fa15bb3e997"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:91d77d2a782be4274da750752bb1650a97bfd8f291022b379bb8e01c66b4e96b"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:45db3a33139e9c8f7c09234b5784a5e33d31fd6907800b316decad50af323ff2"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:263cc3d821c4ab2213cbe8cd8b355a7f72a8324577dc865ef98487c1aeee2bc7"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-win32.whl", hash = "sha256:17771976e82e9f94976180f76468546834d22a7cc404b17c22df2a2c81db0c66"}, | ||||
|     {file = "cffi-1.15.0-cp37-cp37m-win_amd64.whl", hash = "sha256:3415c89f9204ee60cd09b235810be700e993e343a408693e80ce7f6a40108029"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4238e6dab5d6a8ba812de994bbb0a79bddbdf80994e4ce802b6f6f3142fcc880"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0808014eb713677ec1292301ea4c81ad277b6cdf2fdd90fd540af98c0b101d20"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:57e9ac9ccc3101fac9d6014fba037473e4358ef4e89f8e181f8951a2c0162024"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b6c2ea03845c9f501ed1313e78de148cd3f6cad741a75d43a29b43da27f2e1e"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:10dffb601ccfb65262a27233ac273d552ddc4d8ae1bf93b21c94b8511bffe728"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:786902fb9ba7433aae840e0ed609f45c7bcd4e225ebb9c753aa39725bb3e6ad6"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-win32.whl", hash = "sha256:da5db4e883f1ce37f55c667e5c0de439df76ac4cb55964655906306918e7363c"}, | ||||
|     {file = "cffi-1.15.0-cp38-cp38-win_amd64.whl", hash = "sha256:181dee03b1170ff1969489acf1c26533710231c58f95534e3edac87fff06c443"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:45e8636704eacc432a206ac7345a5d3d2c62d95a507ec70d62f23cd91770482a"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:31fb708d9d7c3f49a60f04cf5b119aeefe5644daba1cd2a0fe389b674fd1de37"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6dc2737a3674b3e344847c8686cf29e500584ccad76204efea14f451d4cc669a"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:74fdfdbfdc48d3f47148976f49fab3251e550a8720bebc99bf1483f5bfb5db3e"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffaa5c925128e29efbde7301d8ecaf35c8c60ffbcd6a1ffd3a552177c8e5e796"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f7d084648d77af029acb79a0ff49a0ad7e9d09057a9bf46596dac9514dc07df"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ef1f279350da2c586a69d32fc8733092fd32cc8ac95139a00377841f59a3f8d8"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-win32.whl", hash = "sha256:2a23af14f408d53d5e6cd4e3d9a24ff9e05906ad574822a10563efcef137979a"}, | ||||
|     {file = "cffi-1.15.0-cp39-cp39-win_amd64.whl", hash = "sha256:3773c4d81e6e818df2efbc7dd77325ca0dcb688116050fb2b3011218eda36139"}, | ||||
|     {file = "cffi-1.15.0.tar.gz", hash = "sha256:920f0d66a896c2d99f0adbb391f990a84091179542c205fa53ce5787aff87954"}, | ||||
| ] | ||||
| cfgv = [ | ||||
|     {file = "cfgv-3.3.1-py2.py3-none-any.whl", hash = "sha256:c6a0883f3917a037485059700b9e75da2464e6c27051014ad85ba6aaa5884426"}, | ||||
|     {file = "cfgv-3.3.1.tar.gz", hash = "sha256:f5a830efb9ce7a445376bb66ec94c638a9787422f96264c98edc6bdeed8ab736"}, | ||||
| ] | ||||
| click = [ | ||||
|     {file = "click-8.1.3-py3-none-any.whl", hash = "sha256:bb4d8133cb15a609f44e8213d9b391b0809795062913b383c62be0ee95b1db48"}, | ||||
|     {file = "click-8.1.3.tar.gz", hash = "sha256:7682dc8afb30297001674575ea00d1814d808d6a36af415a82bd481d37ba7b8e"}, | ||||
| @@ -318,14 +679,74 @@ colorama = [ | ||||
|     {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"}, | ||||
|     {file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"}, | ||||
| ] | ||||
| distlib = [ | ||||
|     {file = "distlib-0.3.4-py2.py3-none-any.whl", hash = "sha256:6564fe0a8f51e734df6333d08b8b94d4ea8ee6b99b5ed50613f731fd4089f34b"}, | ||||
|     {file = "distlib-0.3.4.zip", hash = "sha256:e4b58818180336dc9c529bfb9a0b58728ffc09ad92027a3f30b7cd91e3458579"}, | ||||
| ] | ||||
| fastapi = [ | ||||
|     {file = "fastapi-0.77.1-py3-none-any.whl", hash = "sha256:041d8935a13c8f22a7ef80c76d8eea54953c44476d7057e7734138ee8951b9dc"}, | ||||
|     {file = "fastapi-0.77.1.tar.gz", hash = "sha256:6dfa715a334d1fafedcea33d74c7382afd2ce123093f83be2f4e322e89d45e22"}, | ||||
| ] | ||||
| filelock = [ | ||||
|     {file = "filelock-3.7.0-py3-none-any.whl", hash = "sha256:c7b5fdb219b398a5b28c8e4c1893ef5f98ece6a38c6ab2c22e26ec161556fed6"}, | ||||
|     {file = "filelock-3.7.0.tar.gz", hash = "sha256:b795f1b42a61bbf8ec7113c341dad679d772567b936fbd1bf43c9a238e673e20"}, | ||||
| ] | ||||
| graphql-core = [ | ||||
|     {file = "graphql-core-3.2.1.tar.gz", hash = "sha256:9d1bf141427b7d54be944587c8349df791ce60ade2e3cccaf9c56368c133c201"}, | ||||
|     {file = "graphql_core-3.2.1-py3-none-any.whl", hash = "sha256:f83c658e4968998eed1923a2e3e3eddd347e005ac0315fbb7ca4d70ea9156323"}, | ||||
| ] | ||||
| greenlet = [ | ||||
|     {file = "greenlet-1.1.2-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:58df5c2a0e293bf665a51f8a100d3e9956febfbf1d9aaf8c0677cf70218910c6"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:aec52725173bd3a7b56fe91bc56eccb26fbdff1386ef123abb63c84c5b43b63a"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:833e1551925ed51e6b44c800e71e77dacd7e49181fdc9ac9a0bf3714d515785d"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27m-win32.whl", hash = "sha256:aa5b467f15e78b82257319aebc78dd2915e4c1436c3c0d1ad6f53e47ba6e2713"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27m-win_amd64.whl", hash = "sha256:40b951f601af999a8bf2ce8c71e8aaa4e8c6f78ff8afae7b808aae2dc50d4c40"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:95e69877983ea39b7303570fa6760f81a3eec23d0e3ab2021b7144b94d06202d"}, | ||||
|     {file = "greenlet-1.1.2-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:356b3576ad078c89a6107caa9c50cc14e98e3a6c4874a37c3e0273e4baf33de8"}, | ||||
|     {file = "greenlet-1.1.2-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:8639cadfda96737427330a094476d4c7a56ac03de7265622fcf4cfe57c8ae18d"}, | ||||
|     {file = "greenlet-1.1.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:97e5306482182170ade15c4b0d8386ded995a07d7cc2ca8f27958d34d6736497"}, | ||||
|     {file = "greenlet-1.1.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e6a36bb9474218c7a5b27ae476035497a6990e21d04c279884eb10d9b290f1b1"}, | ||||
|     {file = "greenlet-1.1.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:abb7a75ed8b968f3061327c433a0fbd17b729947b400747c334a9c29a9af6c58"}, | ||||
|     {file = "greenlet-1.1.2-cp310-cp310-win_amd64.whl", hash = "sha256:14d4f3cd4e8b524ae9b8aa567858beed70c392fdec26dbdb0a8a418392e71708"}, | ||||
|     {file = "greenlet-1.1.2-cp35-cp35m-macosx_10_14_x86_64.whl", hash = "sha256:17ff94e7a83aa8671a25bf5b59326ec26da379ace2ebc4411d690d80a7fbcf23"}, | ||||
|     {file = "greenlet-1.1.2-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:9f3cba480d3deb69f6ee2c1825060177a22c7826431458c697df88e6aeb3caee"}, | ||||
|     {file = "greenlet-1.1.2-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:fa877ca7f6b48054f847b61d6fa7bed5cebb663ebc55e018fda12db09dcc664c"}, | ||||
|     {file = "greenlet-1.1.2-cp35-cp35m-win32.whl", hash = "sha256:7cbd7574ce8e138bda9df4efc6bf2ab8572c9aff640d8ecfece1b006b68da963"}, | ||||
|     {file = "greenlet-1.1.2-cp35-cp35m-win_amd64.whl", hash = "sha256:903bbd302a2378f984aef528f76d4c9b1748f318fe1294961c072bdc7f2ffa3e"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:049fe7579230e44daef03a259faa24511d10ebfa44f69411d99e6a184fe68073"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:dd0b1e9e891f69e7675ba5c92e28b90eaa045f6ab134ffe70b52e948aa175b3c"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:7418b6bfc7fe3331541b84bb2141c9baf1ec7132a7ecd9f375912eca810e714e"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f9d29ca8a77117315101425ec7ec2a47a22ccf59f5593378fc4077ac5b754fce"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:21915eb821a6b3d9d8eefdaf57d6c345b970ad722f856cd71739493ce003ad08"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eff9d20417ff9dcb0d25e2defc2574d10b491bf2e693b4e491914738b7908168"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-win32.whl", hash = "sha256:32ca72bbc673adbcfecb935bb3fb1b74e663d10a4b241aaa2f5a75fe1d1f90aa"}, | ||||
|     {file = "greenlet-1.1.2-cp36-cp36m-win_amd64.whl", hash = "sha256:f0214eb2a23b85528310dad848ad2ac58e735612929c8072f6093f3585fd342d"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:b92e29e58bef6d9cfd340c72b04d74c4b4e9f70c9fa7c78b674d1fec18896dc4"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:fdcec0b8399108577ec290f55551d926d9a1fa6cad45882093a7a07ac5ec147b"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:93f81b134a165cc17123626ab8da2e30c0455441d4ab5576eed73a64c025b25c"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1e12bdc622676ce47ae9abbf455c189e442afdde8818d9da983085df6312e7a1"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8c790abda465726cfb8bb08bd4ca9a5d0a7bd77c7ac1ca1b839ad823b948ea28"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f276df9830dba7a333544bd41070e8175762a7ac20350786b322b714b0e654f5"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-win32.whl", hash = "sha256:64e6175c2e53195278d7388c454e0b30997573f3f4bd63697f88d855f7a6a1fc"}, | ||||
|     {file = "greenlet-1.1.2-cp37-cp37m-win_amd64.whl", hash = "sha256:b11548073a2213d950c3f671aa88e6f83cda6e2fb97a8b6317b1b5b33d850e06"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:9633b3034d3d901f0a46b7939f8c4d64427dfba6bbc5a36b1a67364cf148a1b0"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:eb6ea6da4c787111adf40f697b4e58732ee0942b5d3bd8f435277643329ba627"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:f3acda1924472472ddd60c29e5b9db0cec629fbe3c5c5accb74d6d6d14773478"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e859fcb4cbe93504ea18008d1df98dee4f7766db66c435e4882ab35cf70cac43"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:00e44c8afdbe5467e4f7b5851be223be68adb4272f44696ee71fe46b7036a711"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ec8c433b3ab0419100bd45b47c9c8551248a5aee30ca5e9d399a0b57ac04651b"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-win32.whl", hash = "sha256:288c6a76705dc54fba69fbcb59904ae4ad768b4c768839b8ca5fdadec6dd8cfd"}, | ||||
|     {file = "greenlet-1.1.2-cp38-cp38-win_amd64.whl", hash = "sha256:8d2f1fb53a421b410751887eb4ff21386d119ef9cde3797bf5e7ed49fb51a3b3"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:166eac03e48784a6a6e0e5f041cfebb1ab400b394db188c48b3a84737f505b67"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:572e1787d1460da79590bf44304abbc0a2da944ea64ec549188fa84d89bba7ab"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:be5f425ff1f5f4b3c1e33ad64ab994eed12fc284a6ea71c5243fd564502ecbe5"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b1692f7d6bc45e3200844be0dba153612103db241691088626a33ff1f24a0d88"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7227b47e73dedaa513cdebb98469705ef0d66eb5a1250144468e9c3097d6b59b"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7ff61ff178250f9bb3cd89752df0f1dd0e27316a8bd1465351652b1b4a4cdfd3"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-win32.whl", hash = "sha256:f70a9e237bb792c7cc7e44c531fd48f5897961701cdaa06cf22fc14965c496cf"}, | ||||
|     {file = "greenlet-1.1.2-cp39-cp39-win_amd64.whl", hash = "sha256:013d61294b6cd8fe3242932c1c5e36e5d1db2c8afb58606c5a67efce62c1f5fd"}, | ||||
|     {file = "greenlet-1.1.2.tar.gz", hash = "sha256:e30f5ea4ae2346e62cedde8794a56858a67b878dd79f7df76a0767e356b1744a"}, | ||||
| ] | ||||
| h11 = [ | ||||
|     {file = "h11-0.13.0-py3-none-any.whl", hash = "sha256:8ddd78563b633ca55346c8cd41ec0af27d3c79931828beffb46ce70a379e7442"}, | ||||
|     {file = "h11-0.13.0.tar.gz", hash = "sha256:70813c1135087a248a4d38cc0e1a0181ffab2188141a93eaf567940c3957ff06"}, | ||||
| @@ -366,10 +787,38 @@ httptools = [ | ||||
|     {file = "httptools-0.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:34d2903dd2a3dd85d33705b6fde40bf91fc44411661283763fd0746723963c83"}, | ||||
|     {file = "httptools-0.4.0.tar.gz", hash = "sha256:2c9a930c378b3d15d6b695fb95ebcff81a7395b4f9775c4f10a076beb0b2c1ff"}, | ||||
| ] | ||||
| identify = [ | ||||
|     {file = "identify-2.5.1-py2.py3-none-any.whl", hash = "sha256:0dca2ea3e4381c435ef9c33ba100a78a9b40c0bab11189c7cf121f75815efeaa"}, | ||||
|     {file = "identify-2.5.1.tar.gz", hash = "sha256:3d11b16f3fe19f52039fb7e39c9c884b21cb1b586988114fbe42671f03de3e82"}, | ||||
| ] | ||||
| idna = [ | ||||
|     {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"}, | ||||
|     {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"}, | ||||
| ] | ||||
| mypy-extensions = [ | ||||
|     {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, | ||||
|     {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, | ||||
| ] | ||||
| nodeenv = [ | ||||
|     {file = "nodeenv-1.6.0-py2.py3-none-any.whl", hash = "sha256:621e6b7076565ddcacd2db0294c0381e01fd28945ab36bcf00f41c5daf63bef7"}, | ||||
|     {file = "nodeenv-1.6.0.tar.gz", hash = "sha256:3ef13ff90291ba2a4a7a4ff9a979b63ffdd00a464dbe04acf0ea6471517a4c2b"}, | ||||
| ] | ||||
| pathspec = [ | ||||
|     {file = "pathspec-0.9.0-py2.py3-none-any.whl", hash = "sha256:7d15c4ddb0b5c802d161efc417ec1a2558ea2653c2e8ad9c19098201dc1c993a"}, | ||||
|     {file = "pathspec-0.9.0.tar.gz", hash = "sha256:e564499435a2673d586f6b2130bb5b95f04a3ba06f81b8f895b651a3c76aabb1"}, | ||||
| ] | ||||
| platformdirs = [ | ||||
|     {file = "platformdirs-2.5.2-py3-none-any.whl", hash = "sha256:027d8e83a2d7de06bbac4e5ef7e023c02b863d7ea5d079477e722bb41ab25788"}, | ||||
|     {file = "platformdirs-2.5.2.tar.gz", hash = "sha256:58c8abb07dcb441e6ee4b11d8df0ac856038f944ab98b7be6b27b2a3c7feef19"}, | ||||
| ] | ||||
| pre-commit = [ | ||||
|     {file = "pre_commit-2.19.0-py2.py3-none-any.whl", hash = "sha256:10c62741aa5704faea2ad69cb550ca78082efe5697d6f04e5710c3c229afdd10"}, | ||||
|     {file = "pre_commit-2.19.0.tar.gz", hash = "sha256:4233a1e38621c87d9dda9808c6606d7e7ba0e087cd56d3fe03202a01d2919615"}, | ||||
| ] | ||||
| pycparser = [ | ||||
|     {file = "pycparser-2.21-py2.py3-none-any.whl", hash = "sha256:8ee45429555515e1f6b185e78100aea234072576aa43ab53aefcae078162fca9"}, | ||||
|     {file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206"}, | ||||
| ] | ||||
| pydantic = [ | ||||
|     {file = "pydantic-1.9.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:cb23bcc093697cdea2708baae4f9ba0e972960a835af22560f6ae4e7e47d33f5"}, | ||||
|     {file = "pydantic-1.9.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:1d5278bd9f0eee04a44c712982343103bba63507480bfd2fc2790fa70cd64cf4"}, | ||||
| @@ -411,6 +860,10 @@ pygments = [ | ||||
|     {file = "Pygments-2.12.0-py3-none-any.whl", hash = "sha256:dc9c10fb40944260f6ed4c688ece0cd2048414940f1cea51b8b226318411c519"}, | ||||
|     {file = "Pygments-2.12.0.tar.gz", hash = "sha256:5eb116118f9612ff1ee89ac96437bb6b49e8f04d8a13b514ba26f620208e26eb"}, | ||||
| ] | ||||
| pyjwt = [ | ||||
|     {file = "PyJWT-2.4.0-py3-none-any.whl", hash = "sha256:72d1d253f32dbd4f5c88eaf1fdc62f3a19f676ccbadb9dbc5d07e951b2b26daf"}, | ||||
|     {file = "PyJWT-2.4.0.tar.gz", hash = "sha256:d42908208c699b3b973cbeb01a969ba6a96c821eefb1c5bfe4c390c01d67abba"}, | ||||
| ] | ||||
| python-dateutil = [ | ||||
|     {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"}, | ||||
|     {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"}, | ||||
| @@ -465,6 +918,44 @@ sniffio = [ | ||||
|     {file = "sniffio-1.2.0-py3-none-any.whl", hash = "sha256:471b71698eac1c2112a40ce2752bb2f4a4814c22a54a3eed3676bc0f5ca9f663"}, | ||||
|     {file = "sniffio-1.2.0.tar.gz", hash = "sha256:c4666eecec1d3f50960c6bdf61ab7bc350648da6c126e3cf6898d8cd4ddcd3de"}, | ||||
| ] | ||||
| sqlalchemy = [ | ||||
|     {file = "SQLAlchemy-1.4.36-cp27-cp27m-macosx_10_14_x86_64.whl", hash = "sha256:81e53bd383c2c33de9d578bfcc243f559bd3801a0e57f2bcc9a943c790662e0c"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp27-cp27m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:6e1fe00ee85c768807f2a139b83469c1e52a9ffd58a6eb51aa7aeb524325ab18"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp27-cp27m-win32.whl", hash = "sha256:d57ac32f8dc731fddeb6f5d1358b4ca5456e72594e664769f0a9163f13df2a31"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp27-cp27m-win_amd64.whl", hash = "sha256:fca8322e04b2dde722fcb0558682740eebd3bd239bea7a0d0febbc190e99dc15"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp27-cp27mu-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:53d2d9ee93970c969bc4e3c78b1277d7129554642f6ffea039c282c7dc4577bc"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:f0394a3acfb8925db178f7728adb38c027ed7e303665b225906bfa8099dc1ce8"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:09c606d8238feae2f360b8742ffbe67741937eb0a05b57f536948d198a3def96"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8d07fe2de0325d06e7e73281e9a9b5e259fbd7cbfbe398a0433cbb0082ad8fa7"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5041474dcab7973baa91ec1f3112049a9dd4652898d6a95a6a895ff5c58beb6b"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-win32.whl", hash = "sha256:be094460930087e50fd08297db9d7aadaed8408ad896baf758e9190c335632da"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp310-cp310-win_amd64.whl", hash = "sha256:64d796e9af522162f7f2bf7a3c5531a0a550764c426782797bbeed809d0646c5"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:a0ae3aa2e86a4613f2d4c49eb7da23da536e6ce80b2bfd60bbb2f55fc02b0b32"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5d50cb71c1dbed70646d521a0975fb0f92b7c3f84c61fa59e07be23a1aaeecfc"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:16abf35af37a3d5af92725fc9ec507dd9e9183d261c2069b6606d60981ed1c6e"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5864a83bd345871ad9699ce466388f836db7572003d67d9392a71998092210e3"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-win32.whl", hash = "sha256:fbf8c09fe9728168f8cc1b40c239eab10baf9c422c18be7f53213d70434dea43"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp36-cp36m-win_amd64.whl", hash = "sha256:6e859fa96605027bd50d8e966db1c4e1b03e7b3267abbc4b89ae658c99393c58"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:166a3887ec355f7d2f12738f7fa25dc8ac541867147a255f790f2f41f614cb44"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2e885548da361aa3f8a9433db4cfb335b2107e533bf314359ae3952821d84b3e"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:5c90ef955d429966d84326d772eb34333178737ebb669845f1d529eb00c75e72"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7a052bd9f53004f8993c624c452dfad8ec600f572dd0ed0445fbe64b22f5570e"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-win32.whl", hash = "sha256:dce3468bf1fc12374a1a732c9efd146ce034f91bb0482b602a9311cb6166a920"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp37-cp37m-win_amd64.whl", hash = "sha256:6cb4c4f57a20710cea277edf720d249d514e587f796b75785ad2c25e1c0fed26"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:e74ce103b81c375c3853b436297952ef8d7863d801dcffb6728d01544e5191b5"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b20c4178ead9bc398be479428568ff31b6c296eb22e75776273781a6551973f"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:af2587ae11400157753115612d6c6ad255143efba791406ad8a0cbcccf2edcb3"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:83cf3077712be9f65c9aaa0b5bc47bc1a44789fd45053e2e3ecd59ff17c63fe9"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-win32.whl", hash = "sha256:ce20f5da141f8af26c123ebaa1b7771835ca6c161225ce728962a79054f528c3"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp38-cp38-win_amd64.whl", hash = "sha256:316c7e5304dda3e3ad711569ac5d02698bbc71299b168ac56a7076b86259f7ea"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:f522214f6749bc073262529c056f7dfd660f3b5ec4180c5354d985eb7219801e"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2ecac4db8c1aa4a269f5829df7e706639a24b780d2ac46b3e485cbbd27ec0028"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b3db741beaa983d4cbf9087558620e7787106319f7e63a066990a70657dd6b35"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2ec89bf98cc6a0f5d1e28e3ad28e9be6f3b4bdbd521a4053c7ae8d5e1289a8a1"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-win32.whl", hash = "sha256:e12532c4d3f614678623da5d852f038ace1f01869b89f003ed6fe8c793f0c6a3"}, | ||||
|     {file = "SQLAlchemy-1.4.36-cp39-cp39-win_amd64.whl", hash = "sha256:cb441ca461bf97d00877b607f132772644b623518b39ced54da433215adce691"}, | ||||
|     {file = "SQLAlchemy-1.4.36.tar.gz", hash = "sha256:64678ac321d64a45901ef2e24725ec5e783f1f4a588305e196431447e7ace243"}, | ||||
| ] | ||||
| starlette = [ | ||||
|     {file = "starlette-0.19.1-py3-none-any.whl", hash = "sha256:5a60c5c2d051f3a8eb546136aa0c9399773a689595e099e0877704d5888279bf"}, | ||||
|     {file = "starlette-0.19.1.tar.gz", hash = "sha256:c6d21096774ecb9639acad41b86b7706e52ba3bf1dc13ea4ed9ad593d47e24c7"}, | ||||
| @@ -473,6 +964,14 @@ strawberry-graphql = [ | ||||
|     {file = "strawberry-graphql-0.113.0.tar.gz", hash = "sha256:ff6743aa2c84ba56d9390460a84311c43aa3a1558a261c2fecea88cd70d0797a"}, | ||||
|     {file = "strawberry_graphql-0.113.0-py3-none-any.whl", hash = "sha256:eb131a33a51fb53d16eadcbe9651e45744ecfaeeea2d15167bea336a6e1ff9be"}, | ||||
| ] | ||||
| toml = [ | ||||
|     {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"}, | ||||
|     {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"}, | ||||
| ] | ||||
| tomli = [ | ||||
|     {file = "tomli-2.0.1-py3-none-any.whl", hash = "sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc"}, | ||||
|     {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"}, | ||||
| ] | ||||
| typing-extensions = [ | ||||
|     {file = "typing_extensions-4.2.0-py3-none-any.whl", hash = "sha256:6657594ee297170d19f67d55c05852a874e7eb634f4f753dbd667855e07c1708"}, | ||||
|     {file = "typing_extensions-4.2.0.tar.gz", hash = "sha256:f1c24655a0da0d1b67f07e17a5e6b2a105894e6824b92096378bb3668ef02376"}, | ||||
| @@ -499,6 +998,10 @@ uvloop = [ | ||||
|     {file = "uvloop-0.16.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1e5f2e2ff51aefe6c19ee98af12b4ae61f5be456cd24396953244a30880ad861"}, | ||||
|     {file = "uvloop-0.16.0.tar.gz", hash = "sha256:f74bc20c7b67d1c27c72601c78cf95be99d5c2cdd4514502b4f3eb0933ff1228"}, | ||||
| ] | ||||
| virtualenv = [ | ||||
|     {file = "virtualenv-20.14.1-py2.py3-none-any.whl", hash = "sha256:e617f16e25b42eb4f6e74096b9c9e37713cf10bf30168fb4a739f3fa8f898a3a"}, | ||||
|     {file = "virtualenv-20.14.1.tar.gz", hash = "sha256:ef589a79795589aada0c1c5b319486797c03b67ac3984c48c669c0e4f50df3a5"}, | ||||
| ] | ||||
| watchgod = [ | ||||
|     {file = "watchgod-0.8.2-py3-none-any.whl", hash = "sha256:2f3e8137d98f493ff58af54ea00f4d1433a6afe2ed08ab331a657df468c6bfce"}, | ||||
|     {file = "watchgod-0.8.2.tar.gz", hash = "sha256:cb11ff66657befba94d828e3b622d5fb76f22fbda1376f355f3e6e51e97d9450"}, | ||||
|   | ||||
| @@ -10,8 +10,13 @@ python = "^3.10" | ||||
| fastapi = "^0.77.1" | ||||
| uvicorn = {extras = ["standard"], version = "^0.17.6"} | ||||
| strawberry-graphql = "^0.113.0" | ||||
| SQLAlchemy = "^1.4.36" | ||||
| argon2-cffi = "^21.3.0" | ||||
| PyJWT = "^2.4.0" | ||||
|  | ||||
| [tool.poetry.dev-dependencies] | ||||
| black = {version = "^22.3.0", allow-prereleases = true} | ||||
| pre-commit = "^2.19.0" | ||||
|  | ||||
| [build-system] | ||||
| requires = ["poetry-core>=1.0.0"] | ||||
|   | ||||
							
								
								
									
										14
									
								
								routers/frontend.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								routers/frontend.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import typing | ||||
|  | ||||
| from fastapi.staticfiles import StaticFiles | ||||
|  | ||||
|  | ||||
| class SPAStaticFiles(StaticFiles): | ||||
|     async def get_response(self, path: str, scope: typing.Any): | ||||
|         response = await super().get_response(path, scope) | ||||
|  | ||||
|         # Default back to `index.html` for our SPA | ||||
|         if response.status_code == 404: | ||||
|             response = await super().get_response(".", scope) | ||||
|  | ||||
|         return response | ||||
		Reference in New Issue
	
	Block a user