diff --git a/fastapi/security/http.py b/fastapi/security/http.py index 9ab2df3c9..a6cb87b76 100644 --- a/fastapi/security/http.py +++ b/fastapi/security/http.py @@ -2,6 +2,7 @@ import binascii from base64 import b64decode from typing import Optional +from fastapi import HTTPException from fastapi.exceptions import HTTPException from fastapi.openapi.models import HTTPBase as HTTPBaseModel from fastapi.openapi.models import HTTPBearer as HTTPBearerModel @@ -66,32 +67,50 @@ class HTTPAuthorizationCredentials(BaseModel): ] -class HTTPBase(SecurityBase): +class HTTPBasicCredentials(BaseModel): + username: str + password: str + + +class HTTPBasic(SecurityBase): def __init__( self, *, - scheme: str, scheme_name: Optional[str] = None, - description: Optional[str] = None, + realm: str, # REQUIRED parameter (RFC 7617 compliance) auto_error: bool = True, ): - self.model = HTTPBaseModel(scheme=scheme, description=description) self.scheme_name = scheme_name or self.__class__.__name__ + self.realm = realm self.auto_error = auto_error - async def __call__( - self, request: Request - ) -> Optional[HTTPAuthorizationCredentials]: - authorization = request.headers.get("Authorization") - scheme, credentials = get_authorization_scheme_param(authorization) - if not (authorization and scheme and credentials): + async def __call__(self, request: Request) -> Optional[HTTPBasicCredentials]: + authorization_header = request.headers.get("Authorization") + scheme, credentials = get_authorization_scheme_param(authorization_header) + + # Handle missing/invalid scheme + if not authorization_header or scheme.lower() != "basic": if self.auto_error: raise HTTPException( - status_code=HTTP_403_FORBIDDEN, detail="Not authenticated" + status_code=HTTP_401_UNAUTHORIZED, + headers={"WWW-Authenticate": f'Basic realm="{self.realm}"'}, + detail="Invalid authentication credentials", ) - else: - return None - return HTTPAuthorizationCredentials(scheme=scheme, credentials=credentials) + return None + + # Decode credentials (base64) + try: + decoded = base64.b64decode(credentials).decode("utf-8") + username, _, password = decoded.partition(":") + return HTTPBasicCredentials(username=username, password=password) + except (ValueError, UnicodeDecodeError): + if self.auto_error: + raise HTTPException( + status_code=HTTP_401_UNAUTHORIZED, + headers={"WWW-Authenticate": f'Basic realm="{self.realm}"'}, + detail="Invalid authentication credentials", + ) + return None class HTTPBasic(HTTPBase): diff --git a/node_modules/.yarn-integrity b/node_modules/.yarn-integrity new file mode 100644 index 000000000..003bc1b47 --- /dev/null +++ b/node_modules/.yarn-integrity @@ -0,0 +1,10 @@ +{ + "systemParams": "win32-x64-127", + "modulesFolders": [], + "flags": [], + "linkedModules": [], + "topLevelPatterns": [], + "lockfileEntries": {}, + "files": [], + "artifacts": {} +} diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 000000000..4a5801883 --- /dev/null +++ b/yarn.lock @@ -0,0 +1,2 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1