from datetime import datetime, timedelta from typing import Any, Union from jose import jwt from passlib.context import CryptContext from app.core.config import settings import string import secrets pwd_context = CryptContext(schemes=["bcrypt"], deprecated="auto") def verify_password(plain_password: str, hashed_password: str) -> bool: return pwd_context.verify(plain_password, hashed_password) def get_password_hash(password: str) -> str: return pwd_context.hash(password) def create_access_token(subject: Union[str, Any], expires_delta: timedelta = None, is_long_term: bool = False) -> str: if expires_delta: expire = datetime.now() + expires_delta else: expire = datetime.now() + timedelta(minutes=settings.ACCESS_TOKEN_EXPIRE_MINUTES) to_encode = {"exp": expire, "sub": str(subject)} if is_long_term: to_encode["long_term"] = True encoded_jwt = jwt.encode(to_encode, settings.SECRET_KEY, algorithm=settings.ALGORITHM) return encoded_jwt def generate_random_password(length: int = 8) -> str: """Generate a random password containing at least one uppercase, one lowercase, one digit and one special char.""" alphabet = string.ascii_letters + string.digits + "!@#$%^&*" while True: password = ''.join(secrets.choice(alphabet) for _ in range(length)) if (any(c.islower() for c in password) and any(c.isupper() for c in password) and any(c.isdigit() for c in password) and any(c in "!@#$%^&*" for c in password)): return password def generate_alphanumeric_password(length: int = 8) -> str: """Generate a random password containing letters and digits only.""" alphabet = string.ascii_letters + string.digits while True: password = ''.join(secrets.choice(alphabet) for _ in range(length)) if (any(c.islower() for c in password) and any(c.isupper() for c in password) and any(c.isdigit() for c in password)): return password def validate_password_strength(password: str) -> bool: """ Validate that the password contains at least one letter and one digit. Returns True if valid, False otherwise. """ has_letter = any(c.isalpha() for c in password) has_digit = any(c.isdigit() for c in password) return has_letter and has_digit