2022-01-15 00:23:50 +00:00
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
from typing import Optional, Dict, Any, Tuple
|
|
|
|
|
|
|
|
from app.classes.models.users import users_helper, ApiKeys
|
|
|
|
from app.classes.shared.helpers import helper
|
|
|
|
|
2022-03-08 04:40:44 +00:00
|
|
|
try:
|
|
|
|
import jwt
|
|
|
|
from jwt import PyJWTError
|
2022-01-15 00:23:50 +00:00
|
|
|
|
2022-03-08 04:40:44 +00:00
|
|
|
except ModuleNotFoundError as e:
|
|
|
|
helper.auto_installer_fix(e)
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
2022-01-15 00:23:50 +00:00
|
|
|
|
2022-03-23 02:50:12 +00:00
|
|
|
|
2022-01-15 00:23:50 +00:00
|
|
|
class Authentication:
|
|
|
|
def __init__(self):
|
|
|
|
self.secret = "my secret"
|
2022-03-23 02:50:12 +00:00
|
|
|
self.secret = helper.get_setting("apikey_secret", None)
|
2022-01-15 00:23:50 +00:00
|
|
|
|
2022-03-23 02:50:12 +00:00
|
|
|
if self.secret is None or self.secret == "random":
|
2022-01-15 00:23:50 +00:00
|
|
|
self.secret = helper.random_string_generator(64)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def generate(user_id, extra=None):
|
|
|
|
if extra is None:
|
|
|
|
extra = {}
|
|
|
|
return jwt.encode(
|
2022-03-23 02:50:12 +00:00
|
|
|
{"user_id": user_id, "iat": int(time.time()), **extra},
|
2022-01-15 00:23:50 +00:00
|
|
|
authentication.secret,
|
2022-03-23 02:50:12 +00:00
|
|
|
algorithm="HS256",
|
2022-01-15 00:23:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def read(token):
|
|
|
|
return jwt.decode(token, authentication.secret, algorithms=["HS256"])
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def check_no_iat(token) -> Optional[Dict[str, Any]]:
|
|
|
|
try:
|
|
|
|
return jwt.decode(token, authentication.secret, algorithms=["HS256"])
|
|
|
|
except PyJWTError as error:
|
|
|
|
logger.debug("Error while checking JWT token: ", exc_info=error)
|
|
|
|
return None
|
|
|
|
|
|
|
|
@staticmethod
|
2022-03-23 02:50:12 +00:00
|
|
|
def check(
|
|
|
|
token,
|
|
|
|
) -> Optional[Tuple[Optional[ApiKeys], Dict[str, Any], Dict[str, Any]]]:
|
2022-01-15 00:23:50 +00:00
|
|
|
try:
|
|
|
|
data = jwt.decode(token, authentication.secret, algorithms=["HS256"])
|
|
|
|
except PyJWTError as error:
|
|
|
|
logger.debug("Error while checking JWT token: ", exc_info=error)
|
|
|
|
return None
|
2022-03-23 02:50:12 +00:00
|
|
|
iat: int = data["iat"]
|
2022-01-15 00:23:50 +00:00
|
|
|
key: Optional[ApiKeys] = None
|
2022-03-23 02:50:12 +00:00
|
|
|
if "token_id" in data:
|
|
|
|
key_id = data["token_id"]
|
2022-01-15 00:23:50 +00:00
|
|
|
key = users_helper.get_user_api_key(key_id)
|
|
|
|
if key is None:
|
|
|
|
return None
|
2022-03-23 02:50:12 +00:00
|
|
|
user_id: str = data["user_id"]
|
2022-01-15 00:23:50 +00:00
|
|
|
user = users_helper.get_user(user_id)
|
2022-03-23 06:06:13 +00:00
|
|
|
# TODO: Have a cache or something so we don't constantly
|
|
|
|
# have to query the database
|
2022-03-23 02:50:12 +00:00
|
|
|
if int(user.get("valid_tokens_from").timestamp()) < iat:
|
2022-01-15 00:23:50 +00:00
|
|
|
# Success!
|
|
|
|
return key, data, user
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def check_bool(token) -> bool:
|
|
|
|
return authentication.check(token) is not None
|
|
|
|
|
|
|
|
|
|
|
|
authentication = Authentication()
|