2024-04-02 22:33:15 +03:00
|
|
|
from datetime import timedelta
|
|
|
|
from uuid import UUID
|
|
|
|
|
|
|
|
from jose import JWTError
|
|
|
|
from jose.jwt import decode, encode
|
|
|
|
|
|
|
|
from api.application.protocols.date_time import DateTimeProvider
|
|
|
|
from api.application.protocols.jwt import JwtTokenProcessor
|
2024-04-08 17:32:50 +03:00
|
|
|
from api.domain.user.error import UserInvalidCredentialsError
|
2024-04-02 22:33:15 +03:00
|
|
|
from api.domain.user.model import UserId
|
|
|
|
from api.infrastructure.auth.jwt_settings import JwtSettings
|
|
|
|
|
|
|
|
|
|
|
|
class JoseJwtTokenProcessor(JwtTokenProcessor):
|
2024-04-08 17:32:50 +03:00
|
|
|
def __init__(
|
|
|
|
self, jwt_options: JwtSettings, date_time_provider: DateTimeProvider
|
|
|
|
) -> None:
|
2024-04-02 22:33:15 +03:00
|
|
|
self.jwt_options = jwt_options
|
|
|
|
self.date_time_provider = date_time_provider
|
|
|
|
|
|
|
|
def generate_token(self, user_id: UserId) -> str:
|
|
|
|
issued_at = self.date_time_provider.get_current_time()
|
2024-04-08 00:31:15 +03:00
|
|
|
expiration_time = issued_at + timedelta(minutes=self.jwt_options.expires_in)
|
2024-04-02 22:33:15 +03:00
|
|
|
|
|
|
|
claims = {
|
|
|
|
"iat": issued_at,
|
|
|
|
"exp": expiration_time,
|
|
|
|
"sub": str(user_id.value),
|
|
|
|
}
|
|
|
|
|
|
|
|
return encode(claims, self.jwt_options.secret, self.jwt_options.algorithm)
|
|
|
|
|
|
|
|
def validate_token(self, token: str) -> UserId | None:
|
|
|
|
try:
|
2024-04-08 17:32:50 +03:00
|
|
|
payload = decode(
|
|
|
|
token, self.jwt_options.secret, [self.jwt_options.algorithm]
|
|
|
|
)
|
2024-04-02 22:33:15 +03:00
|
|
|
return UserId(UUID(payload["sub"]))
|
|
|
|
except (JWTError, ValueError, KeyError):
|
|
|
|
return None
|
2024-04-08 00:31:15 +03:00
|
|
|
|
|
|
|
def refresh_token(self, token: str) -> str:
|
2024-04-08 17:32:50 +03:00
|
|
|
user = self.validate_token(token)
|
|
|
|
if user is None:
|
|
|
|
raise UserInvalidCredentialsError("invalid token")
|
|
|
|
return self.generate_token(user)
|